decap-cms-core 3.10.1 → 3.12.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.
Files changed (36) hide show
  1. package/dist/decap-cms-core.js +18 -18
  2. package/dist/decap-cms-core.js.map +1 -1
  3. package/dist/esm/actions/deploys.js +3 -2
  4. package/dist/esm/backend.js +13 -5
  5. package/dist/esm/bootstrap.js +2 -2
  6. package/dist/esm/components/App/StatusBar.js +1 -1
  7. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +1 -0
  8. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +6 -4
  9. package/dist/esm/components/Editor/EditorToolbar.js +52 -34
  10. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +7 -8
  11. package/dist/esm/components/UI/ErrorBoundary.js +10 -10
  12. package/dist/esm/constants/configSchema.js +9 -0
  13. package/dist/esm/formats/yaml.js +11 -2
  14. package/dist/esm/lib/formatters.js +14 -3
  15. package/dist/esm/lib/i18n.js +8 -3
  16. package/dist/esm/reducers/deploys.js +8 -3
  17. package/index.d.ts +6 -0
  18. package/package.json +2 -3
  19. package/src/actions/deploys.ts +4 -3
  20. package/src/backend.ts +12 -1
  21. package/src/components/Editor/EditorPreviewPane/EditorPreviewContent.js +1 -0
  22. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +3 -0
  23. package/src/components/Editor/EditorToolbar.js +40 -3
  24. package/src/components/Editor/__tests__/EditorToolbar.spec.js +46 -0
  25. package/src/components/MediaLibrary/MediaLibraryButtons.js +2 -3
  26. package/src/components/UI/ErrorBoundary.js +5 -5
  27. package/src/constants/configSchema.js +6 -0
  28. package/src/formats/__tests__/frontmatter.spec.js +21 -0
  29. package/src/formats/__tests__/yaml.spec.js +27 -0
  30. package/src/formats/yaml.ts +16 -1
  31. package/src/lib/__tests__/formatters.spec.js +21 -0
  32. package/src/lib/formatters.ts +15 -3
  33. package/src/lib/i18n.ts +14 -6
  34. package/src/reducers/__tests__/deploys.spec.ts +111 -0
  35. package/src/reducers/deploys.ts +5 -3
  36. package/src/types/redux.ts +6 -0
@@ -47,9 +47,10 @@ export function loadDeployPreview(collection, slug, entry, published, opts) {
47
47
  const backend = currentBackend(state.config);
48
48
  const collectionName = collection.get('name');
49
49
 
50
- // Exit if currently fetching
50
+ // Exit if currently fetching, unless the caller provides a signal
51
+ // (indicating it manages cancellation of the previous poll externally).
51
52
  const deployState = selectDeployPreview(state, collectionName, slug);
52
- if (deployState && deployState.isFetching) {
53
+ if (deployState && deployState.isFetching && !opts?.signal) {
53
54
  return;
54
55
  }
55
56
  dispatch(deployPreviewLoading(collectionName, slug));
@@ -793,7 +793,8 @@ export class Backend {
793
793
  */
794
794
  async getDeployPreview(collection, slug, entry, {
795
795
  maxAttempts = 1,
796
- interval = 5000
796
+ interval = 5000,
797
+ signal
797
798
  } = {}) {
798
799
  /**
799
800
  * If the registered backend does not provide a `getDeployPreview` method, or
@@ -810,6 +811,9 @@ export class Backend {
810
811
  let deployPreview,
811
812
  count = 0;
812
813
  while (!deployPreview && count < maxAttempts) {
814
+ if (signal?.aborted) {
815
+ return;
816
+ }
813
817
  count++;
814
818
  deployPreview = await this.implementation.getDeployPreview(collection.get('name'), slug);
815
819
  if (!deployPreview) {
@@ -893,7 +897,8 @@ export class Backend {
893
897
  slug,
894
898
  path,
895
899
  authorLogin: user.login,
896
- authorName: user.name
900
+ authorName: user.name,
901
+ authorEmail: user.email
897
902
  }, user.useOpenAuthoring);
898
903
  const collectionName = collection.get('name');
899
904
  const updatedOptions = {
@@ -960,7 +965,8 @@ export class Backend {
960
965
  commitMessage: commitMessageFormatter('uploadMedia', config, {
961
966
  path: file.path,
962
967
  authorLogin: user.login,
963
- authorName: user.name
968
+ authorName: user.name,
969
+ authorEmail: user.email
964
970
  }, user.useOpenAuthoring)
965
971
  };
966
972
  return this.implementation.persistMedia(file, options);
@@ -978,7 +984,8 @@ export class Backend {
978
984
  slug,
979
985
  path,
980
986
  authorLogin: user.login,
981
- authorName: user.name
987
+ authorName: user.name,
988
+ authorEmail: user.email
982
989
  }, user.useOpenAuthoring);
983
990
  const entry = selectEntry(state.entries, collection.get('name'), slug);
984
991
  await this.invokePreUnpublishEvent(entry);
@@ -994,7 +1001,8 @@ export class Backend {
994
1001
  const commitMessage = commitMessageFormatter('deleteMedia', config, {
995
1002
  path,
996
1003
  authorLogin: user.login,
997
- authorName: user.name
1004
+ authorName: user.name,
1005
+ authorEmail: user.email
998
1006
  }, user.useOpenAuthoring);
999
1007
  return this.implementation.deleteFiles([path], commitMessage);
1000
1008
  }
@@ -49,8 +49,8 @@ function bootstrap(opts = {}) {
49
49
  /**
50
50
  * Log the version number.
51
51
  */
52
- if (typeof "3.10.1" === 'string') {
53
- console.log(`decap-cms-core ${"3.10.1"}`);
52
+ if (typeof "3.12.0" === 'string') {
53
+ console.log(`decap-cms-core ${"3.12.0"}`);
54
54
  }
55
55
 
56
56
  /**
@@ -30,7 +30,7 @@ function StatusBar({
30
30
  backendName,
31
31
  t
32
32
  }) {
33
- return ___EmotionJSX(StatusBarContainer, null, typeof "3.10.0" === 'string' && ___EmotionJSX("span", null, "decap-cms-app ", "3.10.0"), backendName && ___EmotionJSX("span", null, backendName, " ", t('app.statusBar.backend')), rateLimitInfo && ___EmotionJSX("span", null, rateLimitInfo.used, " / ", rateLimitInfo.limit, " (", formatPercentage(rateLimitInfo.used, rateLimitInfo.limit), "%)", ' ', t('app.statusBar.requestsUsed'), ", ", t('app.statusBar.resetAt'), ' ', formatResetTime(rateLimitInfo.reset)));
33
+ return ___EmotionJSX(StatusBarContainer, null, typeof "3.10.1" === 'string' && ___EmotionJSX("span", null, "decap-cms-app ", "3.10.1"), backendName && ___EmotionJSX("span", null, backendName, " ", t('app.statusBar.backend')), rateLimitInfo && ___EmotionJSX("span", null, rateLimitInfo.used, " / ", rateLimitInfo.limit, " (", formatPercentage(rateLimitInfo.used, rateLimitInfo.limit), "%)", ' ', t('app.statusBar.requestsUsed'), ", ", t('app.statusBar.resetAt'), ' ', formatResetTime(rateLimitInfo.reset)));
34
34
  }
35
35
  function mapStateToProps(state) {
36
36
  return {
@@ -60,6 +60,7 @@ class PreviewContent extends React.Component {
60
60
  }
61
61
  PreviewContent.propTypes = {
62
62
  previewComponent: PropTypes.func.isRequired,
63
+ getEditorComponents: PropTypes.func,
63
64
  previewProps: PropTypes.object,
64
65
  onFieldClick: PropTypes.func
65
66
  };
@@ -7,7 +7,7 @@ import Frame, { FrameContextConsumer } from 'react-frame-component';
7
7
  import { lengths } from 'decap-cms-ui-default';
8
8
  import { connect } from 'react-redux';
9
9
  import { encodeEntry } from '../../../lib/stega';
10
- import { resolveWidget, getPreviewTemplate, getPreviewStyles, getRemarkPlugins } from '../../../lib/registry';
10
+ import { resolveWidget, getPreviewTemplate, getPreviewStyles, getRemarkPlugins, getEditorComponents } from '../../../lib/registry';
11
11
  import { getAllEntries, tryLoadEntry } from '../../../actions/entries';
12
12
  import { ErrorBoundary } from '../../UI';
13
13
  import { selectTemplateName, selectInferredField, selectField } from '../../../reducers/collections';
@@ -21,7 +21,7 @@ import { jsx as ___EmotionJSX } from "@emotion/react";
21
21
  const PreviewPaneFrame = /*#__PURE__*/_styled(Frame, {
22
22
  target: "enus48h0",
23
23
  label: "PreviewPaneFrame"
24
- })("width:100%;height:100%;border:none;background:#fff;border-radius:", lengths.borderRadius, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorPreviewPane/EditorPreviewPane.js"],"names":[],"mappings":"AA8BsC","file":"../../../../../src/components/Editor/EditorPreviewPane/EditorPreviewPane.js","sourcesContent":["import PropTypes from 'prop-types';\nimport React from 'react';\nimport styled from '@emotion/styled';\nimport { List, Map } from 'immutable';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport Frame, { FrameContextConsumer } from 'react-frame-component';\nimport { lengths } from 'decap-cms-ui-default';\nimport { connect } from 'react-redux';\n\nimport { encodeEntry } from '../../../lib/stega';\nimport {\n  resolveWidget,\n  getPreviewTemplate,\n  getPreviewStyles,\n  getRemarkPlugins,\n} from '../../../lib/registry';\nimport { getAllEntries, tryLoadEntry } from '../../../actions/entries';\nimport { ErrorBoundary } from '../../UI';\nimport {\n  selectTemplateName,\n  selectInferredField,\n  selectField,\n} from '../../../reducers/collections';\nimport { boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { INFERABLE_FIELDS } from '../../../constants/fieldInference';\nimport EditorPreviewContent from './EditorPreviewContent.js';\nimport PreviewHOC from './PreviewHOC';\nimport EditorPreview from './EditorPreview';\n\nconst PreviewPaneFrame = styled(Frame)`\n  width: 100%;\n  height: 100%;\n  border: none;\n  background: #fff;\n  border-radius: ${lengths.borderRadius};\n`;\n\nexport class PreviewPane extends React.Component {\n  getWidget = (field, value, metadata, props, idx = null) => {\n    const { getAsset, entry } = props;\n    const widget = resolveWidget(field.get('widget'));\n    const key = idx ? field.get('name') + '_' + idx : field.get('name');\n    const valueIsInMap = value && !widget.allowMapValue && Map.isMap(value);\n\n    /**\n     * Use an HOC to provide conditional updates for all previews.\n     */\n    return !widget.preview ? null : (\n      <PreviewHOC\n        previewComponent={widget.preview}\n        key={key}\n        field={field}\n        getAsset={getAsset}\n        value={valueIsInMap ? value.get(field.get('name')) : value}\n        entry={entry}\n        fieldsMetaData={metadata}\n        resolveWidget={resolveWidget}\n        getRemarkPlugins={getRemarkPlugins}\n      />\n    );\n  };\n\n  inferredFields = {};\n\n  inferFields() {\n    const titleField = selectInferredField(this.props.collection, 'title');\n    const shortTitleField = selectInferredField(this.props.collection, 'shortTitle');\n    const authorField = selectInferredField(this.props.collection, 'author');\n\n    this.inferredFields = {};\n    if (titleField) this.inferredFields[titleField] = INFERABLE_FIELDS.title;\n    if (shortTitleField) this.inferredFields[shortTitleField] = INFERABLE_FIELDS.shortTitle;\n    if (authorField) this.inferredFields[authorField] = INFERABLE_FIELDS.author;\n  }\n\n  /**\n   * Returns the widget component for a named field, and makes recursive calls\n   * to retrieve components for nested and deeply nested fields, which occur in\n   * object and list type fields. Used internally to retrieve widgets, and also\n   * exposed for use in custom preview templates.\n   */\n  widgetFor = (\n    name,\n    fields = this.props.fields,\n    values = this.props.entry.get('data'),\n    fieldsMetaData = this.props.fieldsMetaData,\n  ) => {\n    // We retrieve the field by name so that this function can also be used in\n    // custom preview templates, where the field object can't be passed in.\n    let field = fields && fields.find(f => f.get('name') === name);\n    let value = Map.isMap(values) && values.get(field.get('name'));\n    if (field.get('meta')) {\n      value = this.props.entry.getIn(['meta', field.get('name')]);\n    }\n\n    const nestedFields = field.get('fields');\n    const singleField = field.get('field');\n    const metadata = fieldsMetaData && fieldsMetaData.get(field.get('name'), Map());\n\n    if (nestedFields) {\n      field = field.set('fields', this.getNestedWidgets(nestedFields, value, metadata));\n    }\n\n    if (singleField) {\n      field = field.set('field', this.getSingleNested(singleField, value, metadata));\n    }\n\n    const labelledWidgets = ['string', 'text', 'number'];\n    const inferredField = Object.entries(this.inferredFields)\n      .filter(([key]) => {\n        const fieldToMatch = selectField(this.props.collection, key);\n        return fieldToMatch === field;\n      })\n      .map(([, value]) => value)[0];\n\n    if (inferredField) {\n      value = inferredField.defaultPreview(value);\n    } else if (\n      value &&\n      labelledWidgets.indexOf(field.get('widget')) !== -1 &&\n      value.toString().length < 50\n    ) {\n      value = (\n        <div>\n          <strong>{field.get('label', field.get('name'))}:</strong> {value}\n        </div>\n      );\n    }\n\n    return value ? this.getWidget(field, value, metadata, this.props) : null;\n  };\n\n  /**\n   * Retrieves widgets for nested fields (children of object/list fields)\n   */\n  getNestedWidgets = (fields, values, fieldsMetaData) => {\n    // Fields nested within a list field will be paired with a List of value Maps.\n    if (List.isList(values)) {\n      return values.map(value => this.widgetsForNestedFields(fields, value, fieldsMetaData));\n    }\n    // Fields nested within an object field will be paired with a single Map of values.\n    return this.widgetsForNestedFields(fields, values, fieldsMetaData);\n  };\n\n  getSingleNested = (field, values, fieldsMetaData) => {\n    if (List.isList(values)) {\n      return values.map((value, idx) =>\n        this.getWidget(field, value, fieldsMetaData.get(field.get('name')), this.props, idx),\n      );\n    }\n    return this.getWidget(field, values, fieldsMetaData.get(field.get('name')), this.props);\n  };\n\n  /**\n   * Use widgetFor as a mapping function for recursive widget retrieval\n   */\n  widgetsForNestedFields = (fields, values, fieldsMetaData) => {\n    return fields.map(field => this.widgetFor(field.get('name'), fields, values, fieldsMetaData));\n  };\n\n  /**\n   * This function exists entirely to expose nested widgets for object and list\n   * fields to custom preview templates.\n   *\n   * TODO: see if widgetFor can now provide this functionality for preview templates\n   */\n  widgetsFor = name => {\n    const { fields, entry, fieldsMetaData } = this.props;\n    const field = fields.find(f => f.get('name') === name);\n    const nestedFields = field && field.get('fields');\n    const variableTypes = field && field.get('types');\n    const value = entry.getIn(['data', field.get('name')]);\n    const metadata = fieldsMetaData.get(field.get('name'), Map());\n\n    // Variable Type lists\n    if (List.isList(value) && variableTypes) {\n      return value.map(val => {\n        const valueType = variableTypes.find(t => t.get('name') === val.get('type'));\n        const typeFields = valueType && valueType.get('fields');\n        const widgets =\n          typeFields &&\n          Map(\n            typeFields.map((f, i) => [\n              f.get('name'),\n              <div key={i}>{this.getWidget(f, val, metadata.get(f.get('name')), this.props)}</div>,\n            ]),\n          );\n        return Map({ data: val, widgets });\n      });\n    }\n\n    // List widgets\n    if (List.isList(value)) {\n      return value.map(val => {\n        const widgets =\n          nestedFields &&\n          Map(\n            nestedFields.map((f, i) => [\n              f.get('name'),\n              <div key={i}>{this.getWidget(f, val, metadata.get(f.get('name')), this.props)}</div>,\n            ]),\n          );\n        return Map({ data: val, widgets });\n      });\n    }\n\n    return Map({\n      data: value,\n      widgets:\n        nestedFields &&\n        Map(\n          nestedFields.map(f => [\n            f.get('name'),\n            this.getWidget(f, value, metadata.get(f.get('name')), this.props),\n          ]),\n        ),\n    });\n  };\n\n  /**\n   * This function exists entirely to expose collections from outside of this entry\n   *\n   */\n  getCollection = async (collectionName, slug) => {\n    const { state } = this.props;\n    const selectedCollection = state.collections.get(collectionName);\n\n    if (typeof slug === 'undefined') {\n      const entries = await getAllEntries(state, selectedCollection);\n      return entries.map(entry => Map().set('data', entry.data));\n    }\n\n    const entry = await tryLoadEntry(state, selectedCollection, slug);\n    return Map().set('data', entry.data);\n  };\n\n  render() {\n    const { entry, collection, config } = this.props;\n\n    if (!entry || !entry.get('data')) {\n      return null;\n    }\n\n    const previewComponent =\n      getPreviewTemplate(selectTemplateName(collection, entry.get('slug'))) || EditorPreview;\n\n    this.inferFields();\n\n    const visualEditing = collection.getIn(['editor', 'visualEditing'], false);\n\n    // Only encode entry data if visual editing is enabled\n    const previewEntry = visualEditing\n      ? entry.set('data', encodeEntry(entry.get('data'), this.props.fields))\n      : entry;\n\n    const previewProps = {\n      ...this.props,\n      entry: previewEntry,\n      widgetFor: (name, fields, values = previewEntry.get('data'), fieldsMetaData) =>\n        this.widgetFor(name, fields, values, fieldsMetaData),\n      widgetsFor: this.widgetsFor,\n      getCollection: this.getCollection,\n    };\n\n    const styleEls = getPreviewStyles().map((style, i) => {\n      if (style.raw) {\n        return <style key={i}>{style.value}</style>;\n      }\n      return <link key={i} href={style.value} type=\"text/css\" rel=\"stylesheet\" />;\n    });\n\n    if (!collection) {\n      <PreviewPaneFrame id=\"preview-pane\" head={styleEls} />;\n    }\n\n    const initialContent = `\n<!DOCTYPE html>\n<html>\n  <head><base target=\"_blank\"/></head>\n  <body><div></div></body>\n</html>\n`;\n\n    return (\n      <ErrorBoundary config={config}>\n        <PreviewPaneFrame id=\"preview-pane\" head={styleEls} initialContent={initialContent}>\n          <FrameContextConsumer>\n            {({ document, window }) => {\n              return (\n                <EditorPreviewContent\n                  {...{ previewComponent, previewProps: { ...previewProps, document, window } }}\n                  onFieldClick={this.props.onFieldClick}\n                />\n              );\n            }}\n          </FrameContextConsumer>\n        </PreviewPaneFrame>\n      </ErrorBoundary>\n    );\n  }\n}\n\nPreviewPane.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  fields: ImmutablePropTypes.list.isRequired,\n  entry: ImmutablePropTypes.map.isRequired,\n  fieldsMetaData: ImmutablePropTypes.map.isRequired,\n  getAsset: PropTypes.func.isRequired,\n  onFieldClick: PropTypes.func,\n};\n\nfunction mapStateToProps(state) {\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n  return { isLoadingAsset, config: state.config, state };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  return {\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    getAsset: dispatchProps.boundGetAsset(ownProps.collection, ownProps.entry),\n  };\n}\n\nexport default connect(mapStateToProps, mapDispatchToProps, mergeProps)(PreviewPane);\n"]} */"));
24
+ })("width:100%;height:100%;border:none;background:#fff;border-radius:", lengths.borderRadius, ";" + (process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../../src/components/Editor/EditorPreviewPane/EditorPreviewPane.js"],"names":[],"mappings":"AA+BsC","file":"../../../../../src/components/Editor/EditorPreviewPane/EditorPreviewPane.js","sourcesContent":["import PropTypes from 'prop-types';\nimport React from 'react';\nimport styled from '@emotion/styled';\nimport { List, Map } from 'immutable';\nimport ImmutablePropTypes from 'react-immutable-proptypes';\nimport Frame, { FrameContextConsumer } from 'react-frame-component';\nimport { lengths } from 'decap-cms-ui-default';\nimport { connect } from 'react-redux';\n\nimport { encodeEntry } from '../../../lib/stega';\nimport {\n  resolveWidget,\n  getPreviewTemplate,\n  getPreviewStyles,\n  getRemarkPlugins,\n  getEditorComponents,\n} from '../../../lib/registry';\nimport { getAllEntries, tryLoadEntry } from '../../../actions/entries';\nimport { ErrorBoundary } from '../../UI';\nimport {\n  selectTemplateName,\n  selectInferredField,\n  selectField,\n} from '../../../reducers/collections';\nimport { boundGetAsset } from '../../../actions/media';\nimport { selectIsLoadingAsset } from '../../../reducers/medias';\nimport { INFERABLE_FIELDS } from '../../../constants/fieldInference';\nimport EditorPreviewContent from './EditorPreviewContent.js';\nimport PreviewHOC from './PreviewHOC';\nimport EditorPreview from './EditorPreview';\n\nconst PreviewPaneFrame = styled(Frame)`\n  width: 100%;\n  height: 100%;\n  border: none;\n  background: #fff;\n  border-radius: ${lengths.borderRadius};\n`;\n\nexport class PreviewPane extends React.Component {\n  getWidget = (field, value, metadata, props, idx = null) => {\n    const { getAsset, entry } = props;\n    const widget = resolveWidget(field.get('widget'));\n    const key = idx ? field.get('name') + '_' + idx : field.get('name');\n    const valueIsInMap = value && !widget.allowMapValue && Map.isMap(value);\n\n    /**\n     * Use an HOC to provide conditional updates for all previews.\n     */\n    return !widget.preview ? null : (\n      <PreviewHOC\n        previewComponent={widget.preview}\n        key={key}\n        field={field}\n        getAsset={getAsset}\n        value={valueIsInMap ? value.get(field.get('name')) : value}\n        entry={entry}\n        fieldsMetaData={metadata}\n        resolveWidget={resolveWidget}\n        getRemarkPlugins={getRemarkPlugins}\n        getEditorComponents={getEditorComponents}\n      />\n    );\n  };\n\n  inferredFields = {};\n\n  inferFields() {\n    const titleField = selectInferredField(this.props.collection, 'title');\n    const shortTitleField = selectInferredField(this.props.collection, 'shortTitle');\n    const authorField = selectInferredField(this.props.collection, 'author');\n\n    this.inferredFields = {};\n    if (titleField) this.inferredFields[titleField] = INFERABLE_FIELDS.title;\n    if (shortTitleField) this.inferredFields[shortTitleField] = INFERABLE_FIELDS.shortTitle;\n    if (authorField) this.inferredFields[authorField] = INFERABLE_FIELDS.author;\n  }\n\n  /**\n   * Returns the widget component for a named field, and makes recursive calls\n   * to retrieve components for nested and deeply nested fields, which occur in\n   * object and list type fields. Used internally to retrieve widgets, and also\n   * exposed for use in custom preview templates.\n   */\n  widgetFor = (\n    name,\n    fields = this.props.fields,\n    values = this.props.entry.get('data'),\n    fieldsMetaData = this.props.fieldsMetaData,\n  ) => {\n    // We retrieve the field by name so that this function can also be used in\n    // custom preview templates, where the field object can't be passed in.\n    let field = fields && fields.find(f => f.get('name') === name);\n    let value = Map.isMap(values) && values.get(field.get('name'));\n    if (field.get('meta')) {\n      value = this.props.entry.getIn(['meta', field.get('name')]);\n    }\n\n    const nestedFields = field.get('fields');\n    const singleField = field.get('field');\n    const metadata = fieldsMetaData && fieldsMetaData.get(field.get('name'), Map());\n\n    if (nestedFields) {\n      field = field.set('fields', this.getNestedWidgets(nestedFields, value, metadata));\n    }\n\n    if (singleField) {\n      field = field.set('field', this.getSingleNested(singleField, value, metadata));\n    }\n\n    const labelledWidgets = ['string', 'text', 'number'];\n    const inferredField = Object.entries(this.inferredFields)\n      .filter(([key]) => {\n        const fieldToMatch = selectField(this.props.collection, key);\n        return fieldToMatch === field;\n      })\n      .map(([, value]) => value)[0];\n\n    if (inferredField) {\n      value = inferredField.defaultPreview(value);\n    } else if (\n      value &&\n      labelledWidgets.indexOf(field.get('widget')) !== -1 &&\n      value.toString().length < 50\n    ) {\n      value = (\n        <div>\n          <strong>{field.get('label', field.get('name'))}:</strong> {value}\n        </div>\n      );\n    }\n\n    return value ? this.getWidget(field, value, metadata, this.props) : null;\n  };\n\n  /**\n   * Retrieves widgets for nested fields (children of object/list fields)\n   */\n  getNestedWidgets = (fields, values, fieldsMetaData) => {\n    // Fields nested within a list field will be paired with a List of value Maps.\n    if (List.isList(values)) {\n      return values.map(value => this.widgetsForNestedFields(fields, value, fieldsMetaData));\n    }\n    // Fields nested within an object field will be paired with a single Map of values.\n    return this.widgetsForNestedFields(fields, values, fieldsMetaData);\n  };\n\n  getSingleNested = (field, values, fieldsMetaData) => {\n    if (List.isList(values)) {\n      return values.map((value, idx) =>\n        this.getWidget(field, value, fieldsMetaData.get(field.get('name')), this.props, idx),\n      );\n    }\n    return this.getWidget(field, values, fieldsMetaData.get(field.get('name')), this.props);\n  };\n\n  /**\n   * Use widgetFor as a mapping function for recursive widget retrieval\n   */\n  widgetsForNestedFields = (fields, values, fieldsMetaData) => {\n    return fields.map(field => this.widgetFor(field.get('name'), fields, values, fieldsMetaData));\n  };\n\n  /**\n   * This function exists entirely to expose nested widgets for object and list\n   * fields to custom preview templates.\n   *\n   * TODO: see if widgetFor can now provide this functionality for preview templates\n   */\n  widgetsFor = name => {\n    const { fields, entry, fieldsMetaData } = this.props;\n    const field = fields.find(f => f.get('name') === name);\n    const nestedFields = field && field.get('fields');\n    const variableTypes = field && field.get('types');\n    const value = entry.getIn(['data', field.get('name')]);\n    const metadata = fieldsMetaData.get(field.get('name'), Map());\n\n    // Variable Type lists\n    if (List.isList(value) && variableTypes) {\n      return value.map(val => {\n        const valueType = variableTypes.find(t => t.get('name') === val.get('type'));\n        const typeFields = valueType && valueType.get('fields');\n        const widgets =\n          typeFields &&\n          Map(\n            typeFields.map((f, i) => [\n              f.get('name'),\n              <div key={i}>{this.getWidget(f, val, metadata.get(f.get('name')), this.props)}</div>,\n            ]),\n          );\n        return Map({ data: val, widgets });\n      });\n    }\n\n    // List widgets\n    if (List.isList(value)) {\n      return value.map(val => {\n        const widgets =\n          nestedFields &&\n          Map(\n            nestedFields.map((f, i) => [\n              f.get('name'),\n              <div key={i}>{this.getWidget(f, val, metadata.get(f.get('name')), this.props)}</div>,\n            ]),\n          );\n        return Map({ data: val, widgets });\n      });\n    }\n\n    return Map({\n      data: value,\n      widgets:\n        nestedFields &&\n        Map(\n          nestedFields.map(f => [\n            f.get('name'),\n            this.getWidget(f, value, metadata.get(f.get('name')), this.props),\n          ]),\n        ),\n    });\n  };\n\n  /**\n   * This function exists entirely to expose collections from outside of this entry\n   *\n   */\n  getCollection = async (collectionName, slug) => {\n    const { state } = this.props;\n    const selectedCollection = state.collections.get(collectionName);\n\n    if (typeof slug === 'undefined') {\n      const entries = await getAllEntries(state, selectedCollection);\n      return entries.map(entry => Map().set('data', entry.data));\n    }\n\n    const entry = await tryLoadEntry(state, selectedCollection, slug);\n    return Map().set('data', entry.data);\n  };\n\n  render() {\n    const { entry, collection, config } = this.props;\n\n    if (!entry || !entry.get('data')) {\n      return null;\n    }\n\n    const previewComponent =\n      getPreviewTemplate(selectTemplateName(collection, entry.get('slug'))) || EditorPreview;\n\n    this.inferFields();\n\n    const visualEditing = collection.getIn(['editor', 'visualEditing'], false);\n\n    // Only encode entry data if visual editing is enabled\n    const previewEntry = visualEditing\n      ? entry.set('data', encodeEntry(entry.get('data'), this.props.fields))\n      : entry;\n\n    const previewProps = {\n      ...this.props,\n      entry: previewEntry,\n      widgetFor: (name, fields, values = previewEntry.get('data'), fieldsMetaData) =>\n        this.widgetFor(name, fields, values, fieldsMetaData),\n      widgetsFor: this.widgetsFor,\n      getCollection: this.getCollection,\n      getEditorComponents,\n    };\n\n    const styleEls = getPreviewStyles().map((style, i) => {\n      if (style.raw) {\n        return <style key={i}>{style.value}</style>;\n      }\n      return <link key={i} href={style.value} type=\"text/css\" rel=\"stylesheet\" />;\n    });\n\n    if (!collection) {\n      <PreviewPaneFrame id=\"preview-pane\" head={styleEls} />;\n    }\n\n    const initialContent = `\n<!DOCTYPE html>\n<html>\n  <head><base target=\"_blank\"/></head>\n  <body><div></div></body>\n</html>\n`;\n\n    return (\n      <ErrorBoundary config={config}>\n        <PreviewPaneFrame id=\"preview-pane\" head={styleEls} initialContent={initialContent}>\n          <FrameContextConsumer>\n            {({ document, window }) => {\n              return (\n                <EditorPreviewContent\n                  {...{ previewComponent, previewProps: { ...previewProps, document, window } }}\n                  onFieldClick={this.props.onFieldClick}\n                />\n              );\n            }}\n          </FrameContextConsumer>\n        </PreviewPaneFrame>\n      </ErrorBoundary>\n    );\n  }\n}\n\nPreviewPane.propTypes = {\n  collection: ImmutablePropTypes.map.isRequired,\n  fields: ImmutablePropTypes.list.isRequired,\n  entry: ImmutablePropTypes.map.isRequired,\n  fieldsMetaData: ImmutablePropTypes.map.isRequired,\n  getAsset: PropTypes.func.isRequired,\n  onFieldClick: PropTypes.func,\n};\n\nfunction mapStateToProps(state) {\n  const isLoadingAsset = selectIsLoadingAsset(state.medias);\n  return { isLoadingAsset, config: state.config, state };\n}\n\nfunction mapDispatchToProps(dispatch) {\n  return {\n    boundGetAsset: (collection, entry) => boundGetAsset(dispatch, collection, entry),\n  };\n}\n\nfunction mergeProps(stateProps, dispatchProps, ownProps) {\n  return {\n    ...stateProps,\n    ...dispatchProps,\n    ...ownProps,\n    getAsset: dispatchProps.boundGetAsset(ownProps.collection, ownProps.entry),\n  };\n}\n\nexport default connect(mapStateToProps, mapDispatchToProps, mergeProps)(PreviewPane);\n"]} */"));
25
25
  export class PreviewPane extends React.Component {
26
26
  getWidget = (field, value, metadata, props, idx = null) => {
27
27
  const {
@@ -44,7 +44,8 @@ export class PreviewPane extends React.Component {
44
44
  entry: entry,
45
45
  fieldsMetaData: metadata,
46
46
  resolveWidget: resolveWidget,
47
- getRemarkPlugins: getRemarkPlugins
47
+ getRemarkPlugins: getRemarkPlugins,
48
+ getEditorComponents: getEditorComponents
48
49
  });
49
50
  };
50
51
  inferredFields = {};
@@ -206,7 +207,8 @@ export class PreviewPane extends React.Component {
206
207
  entry: previewEntry,
207
208
  widgetFor: (name, fields, values = previewEntry.get('data'), fieldsMetaData) => this.widgetFor(name, fields, values, fieldsMetaData),
208
209
  widgetsFor: this.widgetsFor,
209
- getCollection: this.getCollection
210
+ getCollection: this.getCollection,
211
+ getEditorComponents
210
212
  };
211
213
  const styleEls = getPreviewStyles().map((style, i) => {
212
214
  if (style.raw) {