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.
- package/dist/decap-cms-core.js +25 -25
- package/dist/decap-cms-core.js.LICENSE.txt +14 -8
- package/dist/decap-cms-core.js.map +1 -1
- package/dist/esm/actions/config.js +57 -49
- package/dist/esm/actions/editorialWorkflow.js +4 -4
- package/dist/esm/actions/entries.js +8 -14
- package/dist/esm/actions/mediaLibrary.js +6 -11
- package/dist/esm/actions/search.js +2 -2
- package/dist/esm/actions/status.js +2 -8
- package/dist/esm/backend.js +70 -79
- package/dist/esm/bootstrap.js +3 -2
- package/dist/esm/components/App/App.js +28 -34
- package/dist/esm/components/App/Header.js +32 -39
- package/dist/esm/components/Collection/Collection.js +45 -48
- package/dist/esm/components/Collection/CollectionSearch.js +76 -81
- package/dist/esm/components/Collection/CollectionTop.js +1 -2
- package/dist/esm/components/Collection/Entries/Entries.js +2 -4
- package/dist/esm/components/Collection/Entries/EntriesCollection.js +25 -29
- package/dist/esm/components/Collection/Entries/EntriesSearch.js +34 -38
- package/dist/esm/components/Collection/Entries/EntryCard.js +8 -13
- package/dist/esm/components/Collection/Entries/EntryListing.js +72 -76
- package/dist/esm/components/Collection/FilterControl.js +1 -1
- package/dist/esm/components/Collection/GroupControl.js +1 -1
- package/dist/esm/components/Collection/NestedCollection.js +50 -53
- package/dist/esm/components/Collection/Sidebar.js +35 -38
- package/dist/esm/components/Collection/SortControl.js +3 -3
- package/dist/esm/components/Collection/ViewStyleControl.js +1 -2
- package/dist/esm/components/Editor/Editor.js +197 -201
- package/dist/esm/components/Editor/EditorControlPane/EditorControl.js +79 -87
- package/dist/esm/components/Editor/EditorControlPane/EditorControlPane.js +75 -86
- package/dist/esm/components/Editor/EditorControlPane/Widget.js +226 -228
- package/dist/esm/components/Editor/EditorInterface.js +69 -80
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreview.js +1 -2
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewContent.js +20 -28
- package/dist/esm/components/Editor/EditorPreviewPane/EditorPreviewPane.js +163 -161
- package/dist/esm/components/Editor/EditorPreviewPane/PreviewHOC.js +4 -8
- package/dist/esm/components/Editor/EditorToolbar.js +335 -347
- package/dist/esm/components/Editor/withWorkflow.js +5 -6
- package/dist/esm/components/MediaLibrary/MediaLibrary.js +304 -294
- package/dist/esm/components/MediaLibrary/MediaLibraryButtons.js +40 -46
- package/dist/esm/components/MediaLibrary/MediaLibraryCard.js +1 -2
- package/dist/esm/components/MediaLibrary/MediaLibraryCardGrid.js +8 -13
- package/dist/esm/components/MediaLibrary/MediaLibraryModal.js +3 -3
- package/dist/esm/components/MediaLibrary/MediaLibrarySearch.js +1 -2
- package/dist/esm/components/MediaLibrary/MediaLibraryTop.js +3 -6
- package/dist/esm/components/UI/DragDrop.js +15 -23
- package/dist/esm/components/UI/ErrorBoundary.js +23 -25
- package/dist/esm/components/UI/Modal.js +10 -12
- package/dist/esm/components/UI/Notifications.js +4 -8
- package/dist/esm/components/UI/SettingsDropdown.js +4 -8
- package/dist/esm/components/Workflow/Workflow.js +19 -20
- package/dist/esm/components/Workflow/WorkflowCard.js +2 -4
- package/dist/esm/components/Workflow/WorkflowList.js +105 -113
- package/dist/esm/constants/configSchema.js +18 -16
- package/dist/esm/formats/formats.js +11 -12
- package/dist/esm/formats/frontmatter.js +17 -21
- package/dist/esm/formats/toml.js +2 -2
- package/dist/esm/formats/yaml.js +2 -6
- package/dist/esm/index.js +3 -7
- package/dist/esm/integrations/providers/algolia/implementation.js +12 -14
- package/dist/esm/integrations/providers/assetStore/implementation.js +10 -12
- package/dist/esm/lib/formatters.js +13 -17
- package/dist/esm/lib/i18n.js +35 -33
- package/dist/esm/lib/phrases.js +2 -2
- package/dist/esm/lib/polyfill.js +8 -0
- package/dist/esm/lib/registry.js +35 -35
- package/dist/esm/lib/serializeEntryValues.js +3 -3
- package/dist/esm/lib/stega.js +142 -0
- package/dist/esm/lib/urlHelper.js +16 -18
- package/dist/esm/mediaLibrary.js +3 -4
- package/dist/esm/reducers/collections.js +26 -42
- package/dist/esm/reducers/combinedReducer.js +3 -6
- package/dist/esm/reducers/config.js +3 -7
- package/dist/esm/reducers/editorialWorkflow.js +5 -9
- package/dist/esm/reducers/entries.js +33 -35
- package/dist/esm/reducers/entryDraft.js +2 -2
- package/dist/esm/reducers/integrations.js +8 -14
- package/dist/esm/reducers/mediaLibrary.js +18 -20
- package/dist/esm/reducers/notifications.js +4 -8
- package/dist/esm/types/immutable.js +7 -1
- package/dist/esm/valueObjects/AssetProxy.js +1 -9
- package/dist/esm/valueObjects/EditorComponent.js +18 -25
- package/dist/esm/valueObjects/Entry.js +2 -2
- package/index.d.ts +2 -0
- package/package.json +14 -11
- package/src/actions/__tests__/config.spec.js +3 -3
- package/src/actions/config.ts +3 -1
- package/src/actions/editorialWorkflow.ts +1 -1
- package/src/actions/entries.ts +1 -1
- package/src/actions/search.ts +1 -1
- package/src/backend.ts +8 -1
- package/src/bootstrap.js +1 -0
- package/src/components/App/App.js +5 -0
- package/src/components/App/Header.js +3 -0
- package/src/components/Collection/Collection.js +5 -0
- package/src/components/Collection/CollectionSearch.js +5 -0
- package/src/components/Collection/Entries/EntriesCollection.js +4 -1
- package/src/components/Collection/Entries/EntriesSearch.js +4 -1
- package/src/components/Collection/Entries/EntryListing.js +5 -0
- package/src/components/Collection/Entries/__tests__/__snapshots__/EntriesCollection.spec.js.snap +0 -4
- package/src/components/Collection/NestedCollection.js +6 -1
- package/src/components/Collection/Sidebar.js +5 -0
- package/src/components/Editor/Editor.js +4 -1
- package/src/components/Editor/EditorControlPane/EditorControl.js +7 -1
- package/src/components/Editor/EditorControlPane/Widget.js +5 -0
- package/src/components/Editor/EditorPreviewPane/EditorPreviewPane.js +1 -1
- package/src/components/Editor/EditorToolbar.js +3 -0
- package/src/components/Editor/__tests__/Editor.spec.js +3 -4
- package/src/components/Editor/__tests__/__snapshots__/Editor.spec.js.snap +5 -5
- package/src/components/Editor/__tests__/__snapshots__/EditorToolbar.spec.js.snap +708 -393
- package/src/components/MediaLibrary/MediaLibrary.js +5 -1
- package/src/components/MediaLibrary/MediaLibraryModal.js +1 -1
- package/src/components/UI/ErrorBoundary.js +6 -1
- package/src/components/UI/Modal.js +3 -0
- package/src/components/Workflow/Workflow.js +3 -0
- package/src/components/Workflow/WorkflowList.js +5 -0
- package/src/constants/__tests__/configSchema.spec.js +1 -1
- package/src/formats/formats.ts +1 -1
- package/src/formats/toml.ts +2 -2
- package/src/integrations/providers/algolia/implementation.js +2 -2
- package/src/integrations/providers/assetStore/implementation.js +2 -1
- package/src/lib/formatters.ts +4 -1
- package/src/lib/i18n.ts +3 -1
- package/src/lib/phrases.js +1 -1
- package/src/lib/polyfill.js +9 -0
- package/src/lib/serializeEntryValues.js +1 -1
- package/src/lib/stega.ts +145 -0
- package/src/lib/urlHelper.ts +4 -1
- package/src/mediaLibrary.ts +1 -1
- package/src/reducers/collections.ts +2 -1
- package/src/reducers/editorialWorkflow.ts +1 -1
- package/src/reducers/entries.ts +6 -1
- package/src/reducers/entryDraft.js +1 -1
- package/src/types/immutable.ts +10 -0
- package/src/types/redux.ts +2 -0
- package/src/valueObjects/EditorComponent.js +1 -1
- 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
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
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
|
|
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 '
|
|
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
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
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
|
-
*
|
|
168
|
-
*
|
|
36
|
+
* Use an HOC to provide conditional updates for all previews.
|
|
169
37
|
*/
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
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 =
|
|
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:
|
|
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
|
|
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
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
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
|
}
|