decap-cms-core 3.6.3 → 3.7.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (137) hide show
  1. package/dist/decap-cms-core.js +25 -25
  2. package/dist/decap-cms-core.js.LICENSE.txt +14 -8
  3. package/dist/decap-cms-core.js.map +1 -1
  4. package/dist/esm/actions/config.js +57 -49
  5. package/dist/esm/actions/editorialWorkflow.js +4 -4
  6. package/dist/esm/actions/entries.js +8 -14
  7. package/dist/esm/actions/mediaLibrary.js +6 -11
  8. package/dist/esm/actions/search.js +2 -2
  9. package/dist/esm/actions/status.js +2 -8
  10. package/dist/esm/backend.js +70 -79
  11. package/dist/esm/bootstrap.js +3 -2
  12. package/dist/esm/components/App/App.js +28 -34
  13. package/dist/esm/components/App/Header.js +32 -39
  14. package/dist/esm/components/Collection/Collection.js +45 -48
  15. package/dist/esm/components/Collection/CollectionSearch.js +76 -81
  16. package/dist/esm/components/Collection/CollectionTop.js +1 -2
  17. package/dist/esm/components/Collection/Entries/Entries.js +2 -4
  18. package/dist/esm/components/Collection/Entries/EntriesCollection.js +25 -29
  19. package/dist/esm/components/Collection/Entries/EntriesSearch.js +34 -38
  20. package/dist/esm/components/Collection/Entries/EntryCard.js +8 -13
  21. package/dist/esm/components/Collection/Entries/EntryListing.js +72 -76
  22. package/dist/esm/components/Collection/FilterControl.js +1 -1
  23. package/dist/esm/components/Collection/GroupControl.js +1 -1
  24. package/dist/esm/components/Collection/NestedCollection.js +50 -53
  25. package/dist/esm/components/Collection/Sidebar.js +35 -38
  26. package/dist/esm/components/Collection/SortControl.js +3 -3
  27. package/dist/esm/components/Collection/ViewStyleControl.js +1 -2
  28. package/dist/esm/components/Editor/Editor.js +197 -201
  29. package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +79 -87
  30. package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +75 -86
  31. package/dist/esm/components/Editor/EditorControlPane/Widget.js +226 -228
  32. package/dist/esm/components/Editor/EditorInterface.js +69 -80
  33. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +1 -2
  34. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +20 -28
  35. package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +163 -161
  36. package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +4 -8
  37. package/dist/esm/components/Editor/EditorToolbar.js +335 -347
  38. package/dist/esm/components/Editor/withWorkflow.js +5 -6
  39. package/dist/esm/components/MediaLibrary/MediaLibrary.js +304 -294
  40. package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +40 -46
  41. package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +1 -2
  42. package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +8 -13
  43. package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +3 -3
  44. package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +1 -2
  45. package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +3 -6
  46. package/dist/esm/components/UI/DragDrop.js +15 -23
  47. package/dist/esm/components/UI/ErrorBoundary.js +23 -25
  48. package/dist/esm/components/UI/Modal.js +10 -12
  49. package/dist/esm/components/UI/Notifications.js +4 -8
  50. package/dist/esm/components/UI/SettingsDropdown.js +4 -8
  51. package/dist/esm/components/Workflow/Workflow.js +19 -20
  52. package/dist/esm/components/Workflow/WorkflowCard.js +2 -4
  53. package/dist/esm/components/Workflow/WorkflowList.js +105 -113
  54. package/dist/esm/constants/configSchema.js +18 -16
  55. package/dist/esm/formats/formats.js +11 -12
  56. package/dist/esm/formats/frontmatter.js +17 -21
  57. package/dist/esm/formats/toml.js +2 -2
  58. package/dist/esm/formats/yaml.js +2 -6
  59. package/dist/esm/index.js +3 -7
  60. package/dist/esm/integrations/providers/algolia/implementation.js +12 -14
  61. package/dist/esm/integrations/providers/assetStore/implementation.js +10 -12
  62. package/dist/esm/lib/formatters.js +13 -17
  63. package/dist/esm/lib/i18n.js +35 -33
  64. package/dist/esm/lib/phrases.js +2 -2
  65. package/dist/esm/lib/polyfill.js +8 -0
  66. package/dist/esm/lib/registry.js +35 -35
  67. package/dist/esm/lib/serializeEntryValues.js +3 -3
  68. package/dist/esm/lib/stega.js +142 -0
  69. package/dist/esm/lib/urlHelper.js +16 -18
  70. package/dist/esm/mediaLibrary.js +3 -4
  71. package/dist/esm/reducers/collections.js +26 -42
  72. package/dist/esm/reducers/combinedReducer.js +3 -6
  73. package/dist/esm/reducers/config.js +3 -7
  74. package/dist/esm/reducers/editorialWorkflow.js +5 -9
  75. package/dist/esm/reducers/entries.js +33 -35
  76. package/dist/esm/reducers/entryDraft.js +2 -2
  77. package/dist/esm/reducers/integrations.js +8 -14
  78. package/dist/esm/reducers/mediaLibrary.js +18 -20
  79. package/dist/esm/reducers/notifications.js +4 -8
  80. package/dist/esm/types/immutable.js +7 -1
  81. package/dist/esm/valueObjects/AssetProxy.js +1 -9
  82. package/dist/esm/valueObjects/EditorComponent.js +18 -25
  83. package/dist/esm/valueObjects/Entry.js +2 -2
  84. package/index.d.ts +2 -0
  85. package/package.json +14 -11
  86. package/src/actions/__tests__/config.spec.js +3 -3
  87. package/src/actions/config.ts +3 -1
  88. package/src/actions/editorialWorkflow.ts +1 -1
  89. package/src/actions/entries.ts +1 -1
  90. package/src/actions/search.ts +1 -1
  91. package/src/backend.ts +8 -1
  92. package/src/bootstrap.js +1 -0
  93. package/src/components/App/App.js +5 -0
  94. package/src/components/App/Header.js +3 -0
  95. package/src/components/Collection/Collection.js +5 -0
  96. package/src/components/Collection/CollectionSearch.js +5 -0
  97. package/src/components/Collection/Entries/EntriesCollection.js +4 -1
  98. package/src/components/Collection/Entries/EntriesSearch.js +4 -1
  99. package/src/components/Collection/Entries/EntryListing.js +5 -0
  100. package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +0 -4
  101. package/src/components/Collection/NestedCollection.js +6 -1
  102. package/src/components/Collection/Sidebar.js +5 -0
  103. package/src/components/Editor/Editor.js +4 -1
  104. package/src/components/Editor/EditorControlPane/EditorControl.js +7 -1
  105. package/src/components/Editor/EditorControlPane/Widget.js +5 -0
  106. package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +1 -1
  107. package/src/components/Editor/EditorToolbar.js +3 -0
  108. package/src/components/Editor/__tests__/Editor.spec.js +3 -4
  109. package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +5 -5
  110. package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +708 -393
  111. package/src/components/MediaLibrary/MediaLibrary.js +5 -1
  112. package/src/components/MediaLibrary/MediaLibraryModal.js +1 -1
  113. package/src/components/UI/ErrorBoundary.js +6 -1
  114. package/src/components/UI/Modal.js +3 -0
  115. package/src/components/Workflow/Workflow.js +3 -0
  116. package/src/components/Workflow/WorkflowList.js +5 -0
  117. package/src/constants/__tests__/configSchema.spec.js +1 -1
  118. package/src/formats/formats.ts +1 -1
  119. package/src/formats/toml.ts +2 -2
  120. package/src/integrations/providers/algolia/implementation.js +2 -2
  121. package/src/integrations/providers/assetStore/implementation.js +2 -1
  122. package/src/lib/formatters.ts +4 -1
  123. package/src/lib/i18n.ts +3 -1
  124. package/src/lib/phrases.js +1 -1
  125. package/src/lib/polyfill.js +9 -0
  126. package/src/lib/serializeEntryValues.js +1 -1
  127. package/src/lib/stega.ts +145 -0
  128. package/src/lib/urlHelper.ts +4 -1
  129. package/src/mediaLibrary.ts +1 -1
  130. package/src/reducers/collections.ts +2 -1
  131. package/src/reducers/editorialWorkflow.ts +1 -1
  132. package/src/reducers/entries.ts +6 -1
  133. package/src/reducers/entryDraft.js +1 -1
  134. package/src/types/immutable.ts +10 -0
  135. package/src/types/redux.ts +2 -0
  136. package/src/valueObjects/EditorComponent.js +1 -1
  137. package/src/valueObjects/Entry.ts +1 -1
@@ -1,6 +1,3 @@
1
- function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
2
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
3
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
4
1
  import PropTypes from 'prop-types';
5
2
  import React from 'react';
6
3
  import { isElement } from 'react-is';
@@ -14,31 +11,27 @@ import { vercelStegaDecode } from '@vercel/stega';
14
11
  */
15
12
  import { jsx as ___EmotionJSX } from "@emotion/react";
16
13
  class PreviewContent extends React.Component {
17
- constructor(...args) {
18
- super(...args);
19
- _defineProperty(this, "handleClick", e => {
20
- var _previewProps$collect;
21
- const {
22
- previewProps,
23
- onFieldClick
24
- } = this.props;
25
- const visualEditing = previewProps === null || previewProps === void 0 ? void 0 : (_previewProps$collect = previewProps.collection) === null || _previewProps$collect === void 0 ? void 0 : _previewProps$collect.getIn(['editor', 'visualEditing'], false);
26
- if (!visualEditing) {
27
- return;
28
- }
29
- try {
30
- const text = e.target.textContent;
31
- const decoded = vercelStegaDecode(text);
32
- if (decoded !== null && decoded !== void 0 && decoded.decap) {
33
- if (onFieldClick) {
34
- onFieldClick(decoded.decap);
35
- }
14
+ handleClick = e => {
15
+ const {
16
+ previewProps,
17
+ onFieldClick
18
+ } = this.props;
19
+ const visualEditing = previewProps?.collection?.getIn(['editor', 'visualEditing'], false);
20
+ if (!visualEditing) {
21
+ return;
22
+ }
23
+ try {
24
+ const text = e.target.textContent;
25
+ const decoded = vercelStegaDecode(text);
26
+ if (decoded?.decap) {
27
+ if (onFieldClick) {
28
+ onFieldClick(decoded.decap);
36
29
  }
37
- } catch (err) {
38
- console.log('Visual editing error:', err);
39
30
  }
40
- });
41
- }
31
+ } catch (err) {
32
+ console.log('Visual editing error:', err);
33
+ }
34
+ };
42
35
  renderPreview() {
43
36
  const {
44
37
  previewComponent,
@@ -49,11 +42,10 @@ class PreviewContent extends React.Component {
49
42
  }, isElement(previewComponent) ? /*#__PURE__*/React.cloneElement(previewComponent, previewProps) : /*#__PURE__*/React.createElement(previewComponent, previewProps));
50
43
  }
51
44
  render() {
52
- var _previewProps$collect2;
53
45
  const {
54
46
  previewProps
55
47
  } = this.props;
56
- const visualEditing = previewProps === null || previewProps === void 0 ? void 0 : (_previewProps$collect2 = previewProps.collection) === null || _previewProps$collect2 === void 0 ? void 0 : _previewProps$collect2.getIn(['editor', 'visualEditing'], false);
48
+ const visualEditing = previewProps?.collection?.getIn(['editor', 'visualEditing'], false);
57
49
  const showScrollSync = !visualEditing;
58
50
  return ___EmotionJSX(FrameContextConsumer, null, context => {
59
51
  const preview = this.renderPreview();
@@ -1,9 +1,4 @@
1
1
  import _styled from "@emotion/styled/base";
2
- function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
3
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
4
- function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
5
- function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : String(i); }
6
- function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
7
2
  import PropTypes from 'prop-types';
8
3
  import React from 'react';
9
4
  import { List, Map } from 'immutable';
@@ -11,7 +6,7 @@ import ImmutablePropTypes from 'react-immutable-proptypes';
11
6
  import Frame, { FrameContextConsumer } from 'react-frame-component';
12
7
  import { lengths } from 'decap-cms-ui-default';
13
8
  import { connect } from 'react-redux';
14
- import { encodeEntry } from 'decap-cms-lib-util/src/stega';
9
+ import { encodeEntry } from '../../../lib/stega';
15
10
  import { resolveWidget, getPreviewTemplate, getPreviewStyles, getRemarkPlugins } from '../../../lib/registry';
16
11
  import { getAllEntries, tryLoadEntry } from '../../../actions/entries';
17
12
  import { ErrorBoundary } from '../../UI';
@@ -26,160 +21,33 @@ import { jsx as ___EmotionJSX } from "@emotion/react";
26
21
  const PreviewPaneFrame = /*#__PURE__*/_styled(Frame, {
27
22
  target: "enus48h0",
28
23
  label: "PreviewPaneFrame"
29
- })("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';\nimport { encodeEntry } from 'decap-cms-lib-util/src/stega';\n\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":"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"]} */"));
30
25
  export class PreviewPane extends React.Component {
31
- constructor(...args) {
32
- super(...args);
33
- _defineProperty(this, "getWidget", (field, value, metadata, props, idx = null) => {
34
- const {
35
- getAsset,
36
- entry
37
- } = props;
38
- const widget = resolveWidget(field.get('widget'));
39
- const key = idx ? field.get('name') + '_' + idx : field.get('name');
40
- const valueIsInMap = value && !widget.allowMapValue && Map.isMap(value);
41
-
42
- /**
43
- * Use an HOC to provide conditional updates for all previews.
44
- */
45
- return !widget.preview ? null : ___EmotionJSX(PreviewHOC, {
46
- previewComponent: widget.preview,
47
- key: key,
48
- field: field,
49
- getAsset: getAsset,
50
- value: valueIsInMap ? value.get(field.get('name')) : value,
51
- entry: entry,
52
- fieldsMetaData: metadata,
53
- resolveWidget: resolveWidget,
54
- getRemarkPlugins: getRemarkPlugins
55
- });
56
- });
57
- _defineProperty(this, "inferredFields", {});
58
- /**
59
- * Returns the widget component for a named field, and makes recursive calls
60
- * to retrieve components for nested and deeply nested fields, which occur in
61
- * object and list type fields. Used internally to retrieve widgets, and also
62
- * exposed for use in custom preview templates.
63
- */
64
- _defineProperty(this, "widgetFor", (name, fields = this.props.fields, values = this.props.entry.get('data'), fieldsMetaData = this.props.fieldsMetaData) => {
65
- // We retrieve the field by name so that this function can also be used in
66
- // custom preview templates, where the field object can't be passed in.
67
- let field = fields && fields.find(f => f.get('name') === name);
68
- let value = Map.isMap(values) && values.get(field.get('name'));
69
- if (field.get('meta')) {
70
- value = this.props.entry.getIn(['meta', field.get('name')]);
71
- }
72
- const nestedFields = field.get('fields');
73
- const singleField = field.get('field');
74
- const metadata = fieldsMetaData && fieldsMetaData.get(field.get('name'), Map());
75
- if (nestedFields) {
76
- field = field.set('fields', this.getNestedWidgets(nestedFields, value, metadata));
77
- }
78
- if (singleField) {
79
- field = field.set('field', this.getSingleNested(singleField, value, metadata));
80
- }
81
- const labelledWidgets = ['string', 'text', 'number'];
82
- const inferredField = Object.entries(this.inferredFields).filter(([key]) => {
83
- const fieldToMatch = selectField(this.props.collection, key);
84
- return fieldToMatch === field;
85
- }).map(([, value]) => value)[0];
86
- if (inferredField) {
87
- value = inferredField.defaultPreview(value);
88
- } else if (value && labelledWidgets.indexOf(field.get('widget')) !== -1 && value.toString().length < 50) {
89
- value = ___EmotionJSX("div", null, ___EmotionJSX("strong", null, field.get('label', field.get('name')), ":"), " ", value);
90
- }
91
- return value ? this.getWidget(field, value, metadata, this.props) : null;
92
- });
93
- /**
94
- * Retrieves widgets for nested fields (children of object/list fields)
95
- */
96
- _defineProperty(this, "getNestedWidgets", (fields, values, fieldsMetaData) => {
97
- // Fields nested within a list field will be paired with a List of value Maps.
98
- if (List.isList(values)) {
99
- return values.map(value => this.widgetsForNestedFields(fields, value, fieldsMetaData));
100
- }
101
- // Fields nested within an object field will be paired with a single Map of values.
102
- return this.widgetsForNestedFields(fields, values, fieldsMetaData);
103
- });
104
- _defineProperty(this, "getSingleNested", (field, values, fieldsMetaData) => {
105
- if (List.isList(values)) {
106
- return values.map((value, idx) => this.getWidget(field, value, fieldsMetaData.get(field.get('name')), this.props, idx));
107
- }
108
- return this.getWidget(field, values, fieldsMetaData.get(field.get('name')), this.props);
109
- });
110
- /**
111
- * Use widgetFor as a mapping function for recursive widget retrieval
112
- */
113
- _defineProperty(this, "widgetsForNestedFields", (fields, values, fieldsMetaData) => {
114
- return fields.map(field => this.widgetFor(field.get('name'), fields, values, fieldsMetaData));
115
- });
116
- /**
117
- * This function exists entirely to expose nested widgets for object and list
118
- * fields to custom preview templates.
119
- *
120
- * TODO: see if widgetFor can now provide this functionality for preview templates
121
- */
122
- _defineProperty(this, "widgetsFor", name => {
123
- const {
124
- fields,
125
- entry,
126
- fieldsMetaData
127
- } = this.props;
128
- const field = fields.find(f => f.get('name') === name);
129
- const nestedFields = field && field.get('fields');
130
- const variableTypes = field && field.get('types');
131
- const value = entry.getIn(['data', field.get('name')]);
132
- const metadata = fieldsMetaData.get(field.get('name'), Map());
133
-
134
- // Variable Type lists
135
- if (List.isList(value) && variableTypes) {
136
- return value.map(val => {
137
- const valueType = variableTypes.find(t => t.get('name') === val.get('type'));
138
- const typeFields = valueType && valueType.get('fields');
139
- const widgets = typeFields && Map(typeFields.map((f, i) => [f.get('name'), ___EmotionJSX("div", {
140
- key: i
141
- }, this.getWidget(f, val, metadata.get(f.get('name')), this.props))]));
142
- return Map({
143
- data: val,
144
- widgets
145
- });
146
- });
147
- }
26
+ getWidget = (field, value, metadata, props, idx = null) => {
27
+ const {
28
+ getAsset,
29
+ entry
30
+ } = props;
31
+ const widget = resolveWidget(field.get('widget'));
32
+ const key = idx ? field.get('name') + '_' + idx : field.get('name');
33
+ const valueIsInMap = value && !widget.allowMapValue && Map.isMap(value);
148
34
 
149
- // List widgets
150
- if (List.isList(value)) {
151
- return value.map(val => {
152
- const widgets = nestedFields && Map(nestedFields.map((f, i) => [f.get('name'), ___EmotionJSX("div", {
153
- key: i
154
- }, this.getWidget(f, val, metadata.get(f.get('name')), this.props))]));
155
- return Map({
156
- data: val,
157
- widgets
158
- });
159
- });
160
- }
161
- return Map({
162
- data: value,
163
- widgets: nestedFields && Map(nestedFields.map(f => [f.get('name'), this.getWidget(f, value, metadata.get(f.get('name')), this.props)]))
164
- });
165
- });
166
35
  /**
167
- * This function exists entirely to expose collections from outside of this entry
168
- *
36
+ * Use an HOC to provide conditional updates for all previews.
169
37
  */
170
- _defineProperty(this, "getCollection", async (collectionName, slug) => {
171
- const {
172
- state
173
- } = this.props;
174
- const selectedCollection = state.collections.get(collectionName);
175
- if (typeof slug === 'undefined') {
176
- const entries = await getAllEntries(state, selectedCollection);
177
- return entries.map(entry => Map().set('data', entry.data));
178
- }
179
- const entry = await tryLoadEntry(state, selectedCollection, slug);
180
- return Map().set('data', entry.data);
38
+ return !widget.preview ? null : ___EmotionJSX(PreviewHOC, {
39
+ previewComponent: widget.preview,
40
+ key: key,
41
+ field: field,
42
+ getAsset: getAsset,
43
+ value: valueIsInMap ? value.get(field.get('name')) : value,
44
+ entry: entry,
45
+ fieldsMetaData: metadata,
46
+ resolveWidget: resolveWidget,
47
+ getRemarkPlugins: getRemarkPlugins
181
48
  });
182
- }
49
+ };
50
+ inferredFields = {};
183
51
  inferFields() {
184
52
  const titleField = selectInferredField(this.props.collection, 'title');
185
53
  const shortTitleField = selectInferredField(this.props.collection, 'shortTitle');
@@ -189,6 +57,135 @@ export class PreviewPane extends React.Component {
189
57
  if (shortTitleField) this.inferredFields[shortTitleField] = INFERABLE_FIELDS.shortTitle;
190
58
  if (authorField) this.inferredFields[authorField] = INFERABLE_FIELDS.author;
191
59
  }
60
+
61
+ /**
62
+ * Returns the widget component for a named field, and makes recursive calls
63
+ * to retrieve components for nested and deeply nested fields, which occur in
64
+ * object and list type fields. Used internally to retrieve widgets, and also
65
+ * exposed for use in custom preview templates.
66
+ */
67
+ widgetFor = (name, fields = this.props.fields, values = this.props.entry.get('data'), fieldsMetaData = this.props.fieldsMetaData) => {
68
+ // We retrieve the field by name so that this function can also be used in
69
+ // custom preview templates, where the field object can't be passed in.
70
+ let field = fields && fields.find(f => f.get('name') === name);
71
+ let value = Map.isMap(values) && values.get(field.get('name'));
72
+ if (field.get('meta')) {
73
+ value = this.props.entry.getIn(['meta', field.get('name')]);
74
+ }
75
+ const nestedFields = field.get('fields');
76
+ const singleField = field.get('field');
77
+ const metadata = fieldsMetaData && fieldsMetaData.get(field.get('name'), Map());
78
+ if (nestedFields) {
79
+ field = field.set('fields', this.getNestedWidgets(nestedFields, value, metadata));
80
+ }
81
+ if (singleField) {
82
+ field = field.set('field', this.getSingleNested(singleField, value, metadata));
83
+ }
84
+ const labelledWidgets = ['string', 'text', 'number'];
85
+ const inferredField = Object.entries(this.inferredFields).filter(([key]) => {
86
+ const fieldToMatch = selectField(this.props.collection, key);
87
+ return fieldToMatch === field;
88
+ }).map(([, value]) => value)[0];
89
+ if (inferredField) {
90
+ value = inferredField.defaultPreview(value);
91
+ } else if (value && labelledWidgets.indexOf(field.get('widget')) !== -1 && value.toString().length < 50) {
92
+ value = ___EmotionJSX("div", null, ___EmotionJSX("strong", null, field.get('label', field.get('name')), ":"), " ", value);
93
+ }
94
+ return value ? this.getWidget(field, value, metadata, this.props) : null;
95
+ };
96
+
97
+ /**
98
+ * Retrieves widgets for nested fields (children of object/list fields)
99
+ */
100
+ getNestedWidgets = (fields, values, fieldsMetaData) => {
101
+ // Fields nested within a list field will be paired with a List of value Maps.
102
+ if (List.isList(values)) {
103
+ return values.map(value => this.widgetsForNestedFields(fields, value, fieldsMetaData));
104
+ }
105
+ // Fields nested within an object field will be paired with a single Map of values.
106
+ return this.widgetsForNestedFields(fields, values, fieldsMetaData);
107
+ };
108
+ getSingleNested = (field, values, fieldsMetaData) => {
109
+ if (List.isList(values)) {
110
+ return values.map((value, idx) => this.getWidget(field, value, fieldsMetaData.get(field.get('name')), this.props, idx));
111
+ }
112
+ return this.getWidget(field, values, fieldsMetaData.get(field.get('name')), this.props);
113
+ };
114
+
115
+ /**
116
+ * Use widgetFor as a mapping function for recursive widget retrieval
117
+ */
118
+ widgetsForNestedFields = (fields, values, fieldsMetaData) => {
119
+ return fields.map(field => this.widgetFor(field.get('name'), fields, values, fieldsMetaData));
120
+ };
121
+
122
+ /**
123
+ * This function exists entirely to expose nested widgets for object and list
124
+ * fields to custom preview templates.
125
+ *
126
+ * TODO: see if widgetFor can now provide this functionality for preview templates
127
+ */
128
+ widgetsFor = name => {
129
+ const {
130
+ fields,
131
+ entry,
132
+ fieldsMetaData
133
+ } = this.props;
134
+ const field = fields.find(f => f.get('name') === name);
135
+ const nestedFields = field && field.get('fields');
136
+ const variableTypes = field && field.get('types');
137
+ const value = entry.getIn(['data', field.get('name')]);
138
+ const metadata = fieldsMetaData.get(field.get('name'), Map());
139
+
140
+ // Variable Type lists
141
+ if (List.isList(value) && variableTypes) {
142
+ return value.map(val => {
143
+ const valueType = variableTypes.find(t => t.get('name') === val.get('type'));
144
+ const typeFields = valueType && valueType.get('fields');
145
+ const widgets = typeFields && Map(typeFields.map((f, i) => [f.get('name'), ___EmotionJSX("div", {
146
+ key: i
147
+ }, this.getWidget(f, val, metadata.get(f.get('name')), this.props))]));
148
+ return Map({
149
+ data: val,
150
+ widgets
151
+ });
152
+ });
153
+ }
154
+
155
+ // List widgets
156
+ if (List.isList(value)) {
157
+ return value.map(val => {
158
+ const widgets = nestedFields && Map(nestedFields.map((f, i) => [f.get('name'), ___EmotionJSX("div", {
159
+ key: i
160
+ }, this.getWidget(f, val, metadata.get(f.get('name')), this.props))]));
161
+ return Map({
162
+ data: val,
163
+ widgets
164
+ });
165
+ });
166
+ }
167
+ return Map({
168
+ data: value,
169
+ widgets: nestedFields && Map(nestedFields.map(f => [f.get('name'), this.getWidget(f, value, metadata.get(f.get('name')), this.props)]))
170
+ });
171
+ };
172
+
173
+ /**
174
+ * This function exists entirely to expose collections from outside of this entry
175
+ *
176
+ */
177
+ getCollection = async (collectionName, slug) => {
178
+ const {
179
+ state
180
+ } = this.props;
181
+ const selectedCollection = state.collections.get(collectionName);
182
+ if (typeof slug === 'undefined') {
183
+ const entries = await getAllEntries(state, selectedCollection);
184
+ return entries.map(entry => Map().set('data', entry.data));
185
+ }
186
+ const entry = await tryLoadEntry(state, selectedCollection, slug);
187
+ return Map().set('data', entry.data);
188
+ };
192
189
  render() {
193
190
  const {
194
191
  entry,
@@ -204,12 +201,13 @@ export class PreviewPane extends React.Component {
204
201
 
205
202
  // Only encode entry data if visual editing is enabled
206
203
  const previewEntry = visualEditing ? entry.set('data', encodeEntry(entry.get('data'), this.props.fields)) : entry;
207
- const previewProps = _objectSpread(_objectSpread({}, this.props), {}, {
204
+ const previewProps = {
205
+ ...this.props,
208
206
  entry: previewEntry,
209
207
  widgetFor: (name, fields, values = previewEntry.get('data'), fieldsMetaData) => this.widgetFor(name, fields, values, fieldsMetaData),
210
208
  widgetsFor: this.widgetsFor,
211
209
  getCollection: this.getCollection
212
- });
210
+ };
213
211
  const styleEls = getPreviewStyles().map((style, i) => {
214
212
  if (style.raw) {
215
213
  return ___EmotionJSX("style", {
@@ -248,10 +246,11 @@ export class PreviewPane extends React.Component {
248
246
  }) => {
249
247
  return ___EmotionJSX(EditorPreviewContent, {
250
248
  previewComponent,
251
- previewProps: _objectSpread(_objectSpread({}, previewProps), {}, {
249
+ previewProps: {
250
+ ...previewProps,
252
251
  document,
253
252
  window
254
- }),
253
+ },
255
254
  onFieldClick: this.props.onFieldClick
256
255
  });
257
256
  })));
@@ -279,8 +278,11 @@ function mapDispatchToProps(dispatch) {
279
278
  };
280
279
  }
281
280
  function mergeProps(stateProps, dispatchProps, ownProps) {
282
- return _objectSpread(_objectSpread(_objectSpread(_objectSpread({}, stateProps), dispatchProps), ownProps), {}, {
281
+ return {
282
+ ...stateProps,
283
+ ...dispatchProps,
284
+ ...ownProps,
283
285
  getAsset: dispatchProps.boundGetAsset(ownProps.collection, ownProps.entry)
284
- });
286
+ };
285
287
  }
286
288
  export default connect(mapStateToProps, mapDispatchToProps, mergeProps)(PreviewPane);
@@ -1,6 +1,3 @@
1
- const _excluded = ["previewComponent"];
2
- function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
3
- function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
4
1
  import React from 'react';
5
2
  import PropTypes from 'prop-types';
6
3
  import ImmutablePropTypes from 'react-immutable-proptypes';
@@ -15,11 +12,10 @@ class PreviewHOC extends React.Component {
15
12
  return isWidgetContainer || this.props.value !== nextProps.value || this.props.fieldsMetaData !== nextProps.fieldsMetaData || this.props.getAsset !== nextProps.getAsset;
16
13
  }
17
14
  render() {
18
- const _this$props = this.props,
19
- {
20
- previewComponent
21
- } = _this$props,
22
- props = _objectWithoutProperties(_this$props, _excluded);
15
+ const {
16
+ previewComponent,
17
+ ...props
18
+ } = this.props;
23
19
  return /*#__PURE__*/React.createElement(previewComponent, props);
24
20
  }
25
21
  }