payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
224 lines (223 loc) • 24.7 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "LivePreviewView", {
enumerable: true,
get: function() {
return LivePreviewView;
}
});
const _react = /*#__PURE__*/ _interop_require_wildcard(require("react"));
const _reacti18next = require("react-i18next");
const _getTranslation = require("../../../../utilities/getTranslation");
const _DocumentControls = require("../../elements/DocumentControls");
const _DocumentFields = require("../../elements/DocumentFields");
const _LeaveWithoutSaving = require("../../modals/LeaveWithoutSaving");
const _ActionsProvider = require("../../utilities/ActionsProvider");
const _Config = require("../../utilities/Config");
const _DocumentInfo = require("../../utilities/DocumentInfo");
const _Locale = require("../../utilities/Locale");
const _Meta = /*#__PURE__*/ _interop_require_default(require("../../utilities/Meta"));
const _SetStepNav = require("../collections/Edit/SetStepNav");
const _Context = require("./Context");
const _context = require("./Context/context");
const _Preview = require("./Preview");
require("./index.scss");
const _usePopupWindow = require("./usePopupWindow");
function _interop_require_default(obj) {
return obj && obj.__esModule ? obj : {
default: obj
};
}
function _getRequireWildcardCache(nodeInterop) {
if (typeof WeakMap !== "function") return null;
var cacheBabelInterop = new WeakMap();
var cacheNodeInterop = new WeakMap();
return (_getRequireWildcardCache = function(nodeInterop) {
return nodeInterop ? cacheNodeInterop : cacheBabelInterop;
})(nodeInterop);
}
function _interop_require_wildcard(obj, nodeInterop) {
if (!nodeInterop && obj && obj.__esModule) {
return obj;
}
if (obj === null || typeof obj !== "object" && typeof obj !== "function") {
return {
default: obj
};
}
var cache = _getRequireWildcardCache(nodeInterop);
if (cache && cache.has(obj)) {
return cache.get(obj);
}
var newObj = {
__proto__: null
};
var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor;
for(var key in obj){
if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) {
var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null;
if (desc && (desc.get || desc.set)) {
Object.defineProperty(newObj, key, desc);
} else {
newObj[key] = obj[key];
}
}
}
newObj.default = obj;
if (cache) {
cache.set(obj, newObj);
}
return newObj;
}
const baseClass = 'live-preview';
const PreviewView = (props)=>{
const { i18n, t } = (0, _reacti18next.useTranslation)('general');
const { previewWindowType } = (0, _context.useLivePreviewContext)();
const { apiURL, data, fieldTypes, permissions } = props;
let collection;
let global;
let disableActions;
let disableLeaveWithoutSaving;
let hasSavePermission;
let isEditing;
let id;
let fields = [];
let label;
let description;
if ('collection' in props) {
collection = props?.collection;
disableActions = props?.disableActions;
disableLeaveWithoutSaving = props?.disableLeaveWithoutSaving;
hasSavePermission = props?.hasSavePermission;
isEditing = props?.isEditing;
id = props?.id;
fields = props?.collection?.fields;
}
if ('global' in props) {
global = props?.global;
fields = props?.global?.fields;
label = props?.global?.label;
description = props?.global?.admin?.description;
hasSavePermission = permissions?.update?.permission;
}
return /*#__PURE__*/ _react.default.createElement(_react.Fragment, null, collection && /*#__PURE__*/ _react.default.createElement(_Meta.default, {
description: t('editing'),
keywords: `${(0, _getTranslation.getTranslation)(collection.labels.singular, i18n)}, Payload, CMS`,
title: `${isEditing ? t('editing') : t('creating')} - ${(0, _getTranslation.getTranslation)(collection.labels.singular, i18n)}`
}), global && /*#__PURE__*/ _react.default.createElement(_Meta.default, {
description: (0, _getTranslation.getTranslation)(label, i18n),
keywords: `${(0, _getTranslation.getTranslation)(label, i18n)}, Payload, CMS`,
title: (0, _getTranslation.getTranslation)(label, i18n)
}), (collection && !(collection.versions?.drafts && collection.versions?.drafts?.autosave) || global && !(global.versions?.drafts && global.versions?.drafts?.autosave)) && !disableLeaveWithoutSaving && /*#__PURE__*/ _react.default.createElement(_LeaveWithoutSaving.LeaveWithoutSaving, null), /*#__PURE__*/ _react.default.createElement(_SetStepNav.SetStepNav, {
collection: collection,
global: global,
id: id,
isEditing: isEditing,
view: t('livePreview')
}), /*#__PURE__*/ _react.default.createElement(_DocumentControls.DocumentControls, {
apiURL: apiURL,
collection: collection,
data: data,
disableActions: disableActions,
global: global,
hasSavePermission: hasSavePermission,
id: id,
isEditing: isEditing,
permissions: permissions
}), /*#__PURE__*/ _react.default.createElement("div", {
className: [
baseClass,
previewWindowType === 'popup' && `${baseClass}--detached`
].filter(Boolean).join(' ')
}, /*#__PURE__*/ _react.default.createElement("div", {
className: [
`${baseClass}__main`,
previewWindowType === 'popup' && `${baseClass}__main--popup-open`
].filter(Boolean).join(' ')
}, /*#__PURE__*/ _react.default.createElement(_DocumentFields.DocumentFields, {
description: description,
fieldTypes: fieldTypes,
fields: fields,
forceSidebarWrap: true,
hasSavePermission: hasSavePermission,
permissions: permissions
})), /*#__PURE__*/ _react.default.createElement(_Preview.LivePreview, props)));
};
const LivePreviewView = (props)=>{
const { data } = props;
const config = (0, _Config.useConfig)();
const documentInfo = (0, _DocumentInfo.useDocumentInfo)();
const locale = (0, _Locale.useLocale)();
const { setViewActions } = (0, _ActionsProvider.useActions)();
const collection = documentInfo.collection;
const global = documentInfo.global;
let livePreviewConfig = config?.admin?.livePreview;
if ('collection' in props) {
livePreviewConfig = {
...livePreviewConfig || {},
...props?.collection.admin.livePreview || {}
};
}
if ('global' in props) {
livePreviewConfig = {
...livePreviewConfig || {},
...props?.global.admin.livePreview || {}
};
}
const [url, setURL] = _react.default.useState(()=>{
if (typeof livePreviewConfig?.url === 'string') return livePreviewConfig?.url;
});
(0, _react.useEffect)(()=>{
const getURL = async ()=>{
const newURL = typeof livePreviewConfig?.url === 'function' ? await livePreviewConfig.url({
data,
documentInfo,
locale
}) : livePreviewConfig?.url;
setURL(newURL);
};
getURL() // eslint-disable-line @typescript-eslint/no-floating-promises
;
}, [
data,
documentInfo,
locale,
livePreviewConfig
]);
(0, _react.useEffect)(()=>{
const editConfig = (collection || global)?.admin?.components?.views?.Edit;
const livePreviewActions = editConfig && 'LivePreview' in editConfig && 'actions' in editConfig.LivePreview ? editConfig.LivePreview.actions : [];
setViewActions(livePreviewActions);
return ()=>{
setViewActions([]);
};
}, [
collection,
global,
setViewActions
]);
const breakpoints = [
...livePreviewConfig?.breakpoints || [],
{
name: 'responsive',
height: '100%',
label: 'Responsive',
width: '100%'
}
];
const { isPopupOpen, openPopupWindow, popupRef } = (0, _usePopupWindow.usePopupWindow)({
eventType: 'payload-live-preview',
url
});
return /*#__PURE__*/ _react.default.createElement(_Context.LivePreviewProvider, {
...props,
breakpoints: breakpoints,
isPopupOpen: isPopupOpen,
openPopupWindow: openPopupWindow,
popupRef: popupRef,
url: url
}, /*#__PURE__*/ _react.default.createElement(PreviewView, props));
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../src/admin/components/views/LivePreview/index.tsx"],"sourcesContent":["import React, { Fragment, useEffect } from 'react'\nimport { useTranslation } from 'react-i18next'\n\nimport type { SanitizedCollectionConfig } from '../../../../collections/config/types'\nimport type { LivePreviewConfig } from '../../../../exports/config'\nimport type { Field } from '../../../../fields/config/types'\nimport type { SanitizedGlobalConfig } from '../../../../globals/config/types'\nimport type { FieldTypes } from '../../forms/field-types'\nimport type { EditViewProps } from '../types'\n\nimport { getTranslation } from '../../../../utilities/getTranslation'\nimport { DocumentControls } from '../../elements/DocumentControls'\nimport { DocumentFields } from '../../elements/DocumentFields'\nimport { LeaveWithoutSaving } from '../../modals/LeaveWithoutSaving'\nimport { useActions } from '../../utilities/ActionsProvider'\nimport { useConfig } from '../../utilities/Config'\nimport { useDocumentInfo } from '../../utilities/DocumentInfo'\nimport { useLocale } from '../../utilities/Locale'\nimport Meta from '../../utilities/Meta'\nimport { SetStepNav } from '../collections/Edit/SetStepNav'\nimport { LivePreviewProvider } from './Context'\nimport { useLivePreviewContext } from './Context/context'\nimport { LivePreview } from './Preview'\nimport './index.scss'\nimport { usePopupWindow } from './usePopupWindow'\n\nconst baseClass = 'live-preview'\n\nconst PreviewView: React.FC<\n  EditViewProps & {\n    fieldTypes: FieldTypes\n  }\n> = (props) => {\n  const { i18n, t } = useTranslation('general')\n  const { previewWindowType } = useLivePreviewContext()\n\n  const { apiURL, data, fieldTypes, permissions } = props\n\n  let collection: SanitizedCollectionConfig\n  let global: SanitizedGlobalConfig\n  let disableActions: boolean\n  let disableLeaveWithoutSaving: boolean\n  let hasSavePermission: boolean\n  let isEditing: boolean\n  let id: string\n  let fields: Field[] = []\n  let label: SanitizedGlobalConfig['label']\n  let description: SanitizedGlobalConfig['admin']['description']\n\n  if ('collection' in props) {\n    collection = props?.collection\n    disableActions = props?.disableActions\n    disableLeaveWithoutSaving = props?.disableLeaveWithoutSaving\n    hasSavePermission = props?.hasSavePermission\n    isEditing = props?.isEditing\n    id = props?.id\n    fields = props?.collection?.fields\n  }\n\n  if ('global' in props) {\n    global = props?.global\n    fields = props?.global?.fields\n    label = props?.global?.label\n    description = props?.global?.admin?.description\n    hasSavePermission = permissions?.update?.permission\n  }\n\n  return (\n    <Fragment>\n      {collection && (\n        <Meta\n          description={t('editing')}\n          keywords={`${getTranslation(collection.labels.singular, i18n)}, Payload, CMS`}\n          title={`${isEditing ? t('editing') : t('creating')} - ${getTranslation(\n            collection.labels.singular,\n            i18n,\n          )}`}\n        />\n      )}\n      {global && (\n        <Meta\n          description={getTranslation(label, i18n)}\n          keywords={`${getTranslation(label, i18n)}, Payload, CMS`}\n          title={getTranslation(label, i18n)}\n        />\n      )}\n      {((collection && !(collection.versions?.drafts && collection.versions?.drafts?.autosave)) ||\n        (global && !(global.versions?.drafts && global.versions?.drafts?.autosave))) &&\n        !disableLeaveWithoutSaving && <LeaveWithoutSaving />}\n      <SetStepNav\n        collection={collection}\n        global={global}\n        id={id}\n        isEditing={isEditing}\n        view={t('livePreview')}\n      />\n      <DocumentControls\n        apiURL={apiURL}\n        collection={collection}\n        data={data}\n        disableActions={disableActions}\n        global={global}\n        hasSavePermission={hasSavePermission}\n        id={id}\n        isEditing={isEditing}\n        permissions={permissions}\n      />\n      <div\n        className={[baseClass, previewWindowType === 'popup' && `${baseClass}--detached`]\n          .filter(Boolean)\n          .join(' ')}\n      >\n        <div\n          className={[\n            `${baseClass}__main`,\n            previewWindowType === 'popup' && `${baseClass}__main--popup-open`,\n          ]\n            .filter(Boolean)\n            .join(' ')}\n        >\n          <DocumentFields\n            description={description}\n            fieldTypes={fieldTypes}\n            fields={fields}\n            forceSidebarWrap\n            hasSavePermission={hasSavePermission}\n            permissions={permissions}\n          />\n        </div>\n        <LivePreview {...props} />\n      </div>\n    </Fragment>\n  )\n}\n\nexport const LivePreviewView: React.FC<\n  EditViewProps & {\n    fieldTypes: FieldTypes\n  }\n> = (props) => {\n  const { data } = props\n  const config = useConfig()\n  const documentInfo = useDocumentInfo()\n  const locale = useLocale()\n\n  const { setViewActions } = useActions()\n\n  const collection = documentInfo.collection\n  const global = documentInfo.global\n\n  let livePreviewConfig: LivePreviewConfig = config?.admin?.livePreview\n\n  if ('collection' in props) {\n    livePreviewConfig = {\n      ...(livePreviewConfig || {}),\n      ...(props?.collection.admin.livePreview || {}),\n    }\n  }\n\n  if ('global' in props) {\n    livePreviewConfig = {\n      ...(livePreviewConfig || {}),\n      ...(props?.global.admin.livePreview || {}),\n    }\n  }\n\n  const [url, setURL] = React.useState<string | undefined>(() => {\n    if (typeof livePreviewConfig?.url === 'string') return livePreviewConfig?.url\n  })\n\n  useEffect(() => {\n    const getURL = async () => {\n      const newURL =\n        typeof livePreviewConfig?.url === 'function'\n          ? await livePreviewConfig.url({\n              data,\n              documentInfo,\n              locale,\n            })\n          : livePreviewConfig?.url\n\n      setURL(newURL)\n    }\n\n    getURL() // eslint-disable-line @typescript-eslint/no-floating-promises\n  }, [data, documentInfo, locale, livePreviewConfig])\n\n  useEffect(() => {\n    const editConfig = (collection || global)?.admin?.components?.views?.Edit\n    const livePreviewActions =\n      editConfig && 'LivePreview' in editConfig && 'actions' in editConfig.LivePreview\n        ? editConfig.LivePreview.actions\n        : []\n\n    setViewActions(livePreviewActions)\n\n    return () => {\n      setViewActions([])\n    }\n  }, [collection, global, setViewActions])\n\n  const breakpoints: LivePreviewConfig['breakpoints'] = [\n    ...(livePreviewConfig?.breakpoints || []),\n    {\n      name: 'responsive',\n      height: '100%',\n      label: 'Responsive',\n      width: '100%',\n    },\n  ]\n\n  const { isPopupOpen, openPopupWindow, popupRef } = usePopupWindow({\n    eventType: 'payload-live-preview',\n    url,\n  })\n\n  return (\n    <LivePreviewProvider\n      {...props}\n      breakpoints={breakpoints}\n      isPopupOpen={isPopupOpen}\n      openPopupWindow={openPopupWindow}\n      popupRef={popupRef}\n      url={url}\n    >\n      <PreviewView {...props} />\n    </LivePreviewProvider>\n  )\n}\n"],"names":["LivePreviewView","baseClass","PreviewView","props","i18n","t","useTranslation","previewWindowType","useLivePreviewContext","apiURL","data","fieldTypes","permissions","collection","global","disableActions","disableLeaveWithoutSaving","hasSavePermission","isEditing","id","fields","label","description","admin","update","permission","Fragment","Meta","keywords","getTranslation","labels","singular","title","versions","drafts","autosave","LeaveWithoutSaving","SetStepNav","view","DocumentControls","div","className","filter","Boolean","join","DocumentFields","forceSidebarWrap","LivePreview","config","useConfig","documentInfo","useDocumentInfo","locale","useLocale","setViewActions","useActions","livePreviewConfig","livePreview","url","setURL","React","useState","useEffect","getURL","newURL","editConfig","components","views","Edit","livePreviewActions","actions","breakpoints","name","height","width","isPopupOpen","openPopupWindow","popupRef","usePopupWindow","eventType","LivePreviewProvider"],"mappings":"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"}