payload
Version:
Node, React and MongoDB Headless CMS and Application Framework
208 lines (207 loc) • 20.6 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
Object.defineProperty(exports, "LivePreviewProvider", {
enumerable: true,
get: function() {
return LivePreviewProvider;
}
});
const _core = require("@dnd-kit/core");
const _react = /*#__PURE__*/ _interop_require_wildcard(require("react"));
const _fieldSchemaToJSON = require("../../../../../utilities/fieldSchemaToJSON");
const _collisionDetection = require("./collisionDetection");
const _context = require("./context");
const _sizeReducer = require("./sizeReducer");
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 LivePreviewProvider = (props)=>{
const { breakpoints, children, isPopupOpen, openPopupWindow, popupRef, url } = props;
const [previewWindowType, setPreviewWindowType] = (0, _react.useState)('iframe');
const [appIsReady, setAppIsReady] = (0, _react.useState)(false);
const iframeRef = _react.default.useRef(null);
const [iframeHasLoaded, setIframeHasLoaded] = (0, _react.useState)(false);
const [zoom, setZoom] = (0, _react.useState)(1);
const [position, setPosition] = (0, _react.useState)({
x: 0,
y: 0
});
const [size, setSize] = _react.default.useReducer(_sizeReducer.sizeReducer, {
height: 0,
width: 0
});
const [measuredDeviceSize, setMeasuredDeviceSize] = (0, _react.useState)({
height: 0,
width: 0
});
const [breakpoint, setBreakpoint] = _react.default.useState('responsive');
const [fieldSchemaJSON] = (0, _react.useState)(()=>{
let fields;
if ('collection' in props) {
const { collection } = props;
fields = collection.fields;
}
if ('global' in props) {
const { global } = props;
fields = global.fields;
}
return (0, _fieldSchemaToJSON.fieldSchemaToJSON)(fields);
});
// The toolbar needs to freely drag and drop around the page
const handleDragEnd = (ev)=>{
// only update position if the toolbar is completely within the preview area
// otherwise reset it back to the previous position
// TODO: reset to the nearest edge of the preview area
if (ev.over && ev.over.id === 'live-preview-area') {
const newPos = {
x: position.x + ev.delta.x,
y: position.y + ev.delta.y
};
setPosition(newPos);
} else {
// reset
}
};
const setWidth = (0, _react.useCallback)((width)=>{
setSize({
type: 'width',
value: width
});
}, [
setSize
]);
const setHeight = (0, _react.useCallback)((height)=>{
setSize({
type: 'height',
value: height
});
}, [
setSize
]);
// explicitly set new width and height when as new breakpoints are selected
// exclude `custom` breakpoint as it is handled by the `setWidth` and `setHeight` directly
(0, _react.useEffect)(()=>{
const foundBreakpoint = breakpoints?.find((bp)=>bp.name === breakpoint);
if (foundBreakpoint && breakpoint !== 'responsive' && breakpoint !== 'custom' && typeof foundBreakpoint?.width === 'number' && typeof foundBreakpoint?.height === 'number') {
setSize({
type: 'reset',
value: {
height: foundBreakpoint.height,
width: foundBreakpoint.width
}
});
}
}, [
breakpoint,
breakpoints
]);
// Receive the `ready` message from the popup window
// This indicates that the app is ready to receive `window.postMessage` events
// This is also the only cross-origin way of detecting when a popup window has loaded
// Unlike iframe elements which have an `onLoad` handler, there is no way to access `window.open` on popups
(0, _react.useEffect)(()=>{
const handleMessage = (event)=>{
if (url?.startsWith(event.origin) && event.data && typeof event.data === 'object' && event.data.type === 'payload-live-preview') {
if (event.data.ready) {
setAppIsReady(true);
}
}
};
window.addEventListener('message', handleMessage);
return ()=>{
window.removeEventListener('message', handleMessage);
};
}, [
url
]);
const handleWindowChange = (0, _react.useCallback)((type)=>{
setAppIsReady(false);
setPreviewWindowType(type);
if (type === 'popup') openPopupWindow();
}, [
openPopupWindow
]);
// when the user closes the popup window, switch back to the iframe
// the `usePopupWindow` reports the `isPopupOpen` state for us to use here
(0, _react.useEffect)(()=>{
if (!isPopupOpen) {
handleWindowChange('iframe');
}
}, [
isPopupOpen,
handleWindowChange
]);
return /*#__PURE__*/ _react.default.createElement(_context.LivePreviewContext.Provider, {
value: {
appIsReady,
breakpoint,
breakpoints,
fieldSchemaJSON,
iframeHasLoaded,
iframeRef,
isPopupOpen,
measuredDeviceSize,
openPopupWindow,
popupRef,
previewWindowType,
setAppIsReady,
setBreakpoint,
setHeight,
setIframeHasLoaded,
setMeasuredDeviceSize,
setPreviewWindowType: handleWindowChange,
setSize,
setToolbarPosition: setPosition,
setWidth,
setZoom,
size,
toolbarPosition: position,
url,
zoom
}
}, /*#__PURE__*/ _react.default.createElement(_core.DndContext, {
collisionDetection: _collisionDetection.customCollisionDetection,
onDragEnd: handleDragEnd
}, children));
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../../src/admin/components/views/LivePreview/Context/index.tsx"],"sourcesContent":["import { DndContext } from '@dnd-kit/core'\nimport React, { useCallback, useEffect, useState } from 'react'\n\nimport type { LivePreviewConfig } from '../../../../../exports/config'\nimport type { Field } from '../../../../../fields/config/types'\nimport type { EditViewProps } from '../../types'\nimport type { usePopupWindow } from '../usePopupWindow'\n\nimport { fieldSchemaToJSON } from '../../../../../utilities/fieldSchemaToJSON'\nimport { customCollisionDetection } from './collisionDetection'\nimport { LivePreviewContext } from './context'\nimport { sizeReducer } from './sizeReducer'\n\nexport type LivePreviewProviderProps = EditViewProps & {\n  appIsReady?: boolean\n  breakpoints?: LivePreviewConfig['breakpoints']\n  children: React.ReactNode\n  deviceSize?: {\n    height: number\n    width: number\n  }\n  isPopupOpen?: boolean\n  openPopupWindow?: ReturnType<typeof usePopupWindow>['openPopupWindow']\n  popupRef?: React.MutableRefObject<Window>\n  url?: string\n}\n\nexport const LivePreviewProvider: React.FC<LivePreviewProviderProps> = (props) => {\n  const { breakpoints, children, isPopupOpen, openPopupWindow, popupRef, url } = props\n\n  const [previewWindowType, setPreviewWindowType] = useState<'iframe' | 'popup'>('iframe')\n\n  const [appIsReady, setAppIsReady] = useState(false)\n\n  const iframeRef = React.useRef<HTMLIFrameElement>(null)\n\n  const [iframeHasLoaded, setIframeHasLoaded] = useState(false)\n\n  const [zoom, setZoom] = useState(1)\n\n  const [position, setPosition] = useState({ x: 0, y: 0 })\n\n  const [size, setSize] = React.useReducer(sizeReducer, { height: 0, width: 0 })\n\n  const [measuredDeviceSize, setMeasuredDeviceSize] = useState({\n    height: 0,\n    width: 0,\n  })\n\n  const [breakpoint, setBreakpoint] =\n    React.useState<LivePreviewConfig['breakpoints'][0]['name']>('responsive')\n\n  const [fieldSchemaJSON] = useState(() => {\n    let fields: Field[]\n\n    if ('collection' in props) {\n      const { collection } = props\n      fields = collection.fields\n    }\n\n    if ('global' in props) {\n      const { global } = props\n      fields = global.fields\n    }\n\n    return fieldSchemaToJSON(fields)\n  })\n\n  // The toolbar needs to freely drag and drop around the page\n  const handleDragEnd = (ev) => {\n    // only update position if the toolbar is completely within the preview area\n    // otherwise reset it back to the previous position\n    // TODO: reset to the nearest edge of the preview area\n    if (ev.over && ev.over.id === 'live-preview-area') {\n      const newPos = {\n        x: position.x + ev.delta.x,\n        y: position.y + ev.delta.y,\n      }\n\n      setPosition(newPos)\n    } else {\n      // reset\n    }\n  }\n\n  const setWidth = useCallback(\n    (width) => {\n      setSize({ type: 'width', value: width })\n    },\n    [setSize],\n  )\n\n  const setHeight = useCallback(\n    (height) => {\n      setSize({ type: 'height', value: height })\n    },\n    [setSize],\n  )\n\n  // explicitly set new width and height when as new breakpoints are selected\n  // exclude `custom` breakpoint as it is handled by the `setWidth` and `setHeight` directly\n  useEffect(() => {\n    const foundBreakpoint = breakpoints?.find((bp) => bp.name === breakpoint)\n\n    if (\n      foundBreakpoint &&\n      breakpoint !== 'responsive' &&\n      breakpoint !== 'custom' &&\n      typeof foundBreakpoint?.width === 'number' &&\n      typeof foundBreakpoint?.height === 'number'\n    ) {\n      setSize({\n        type: 'reset',\n        value: {\n          height: foundBreakpoint.height,\n          width: foundBreakpoint.width,\n        },\n      })\n    }\n  }, [breakpoint, breakpoints])\n\n  // Receive the `ready` message from the popup window\n  // This indicates that the app is ready to receive `window.postMessage` events\n  // This is also the only cross-origin way of detecting when a popup window has loaded\n  // Unlike iframe elements which have an `onLoad` handler, there is no way to access `window.open` on popups\n  useEffect(() => {\n    const handleMessage = (event: MessageEvent) => {\n      if (\n        url?.startsWith(event.origin) &&\n        event.data &&\n        typeof event.data === 'object' &&\n        event.data.type === 'payload-live-preview'\n      ) {\n        if (event.data.ready) {\n          setAppIsReady(true)\n        }\n      }\n    }\n\n    window.addEventListener('message', handleMessage)\n\n    return () => {\n      window.removeEventListener('message', handleMessage)\n    }\n  }, [url])\n\n  const handleWindowChange = useCallback(\n    (type: 'iframe' | 'popup') => {\n      setAppIsReady(false)\n      setPreviewWindowType(type)\n      if (type === 'popup') openPopupWindow()\n    },\n    [openPopupWindow],\n  )\n\n  // when the user closes the popup window, switch back to the iframe\n  // the `usePopupWindow` reports the `isPopupOpen` state for us to use here\n  useEffect(() => {\n    if (!isPopupOpen) {\n      handleWindowChange('iframe')\n    }\n  }, [isPopupOpen, handleWindowChange])\n\n  return (\n    <LivePreviewContext.Provider\n      value={{\n        appIsReady,\n        breakpoint,\n        breakpoints,\n        fieldSchemaJSON,\n        iframeHasLoaded,\n        iframeRef,\n        isPopupOpen,\n        measuredDeviceSize,\n        openPopupWindow,\n        popupRef,\n        previewWindowType,\n        setAppIsReady,\n        setBreakpoint,\n        setHeight,\n        setIframeHasLoaded,\n        setMeasuredDeviceSize,\n        setPreviewWindowType: handleWindowChange,\n        setSize,\n        setToolbarPosition: setPosition,\n        setWidth,\n        setZoom,\n        size,\n        toolbarPosition: position,\n        url,\n        zoom,\n      }}\n    >\n      <DndContext collisionDetection={customCollisionDetection} onDragEnd={handleDragEnd}>\n        {children}\n      </DndContext>\n    </LivePreviewContext.Provider>\n  )\n}\n"],"names":["LivePreviewProvider","props","breakpoints","children","isPopupOpen","openPopupWindow","popupRef","url","previewWindowType","setPreviewWindowType","useState","appIsReady","setAppIsReady","iframeRef","React","useRef","iframeHasLoaded","setIframeHasLoaded","zoom","setZoom","position","setPosition","x","y","size","setSize","useReducer","sizeReducer","height","width","measuredDeviceSize","setMeasuredDeviceSize","breakpoint","setBreakpoint","fieldSchemaJSON","fields","collection","global","fieldSchemaToJSON","handleDragEnd","ev","over","id","newPos","delta","setWidth","useCallback","type","value","setHeight","useEffect","foundBreakpoint","find","bp","name","handleMessage","event","startsWith","origin","data","ready","window","addEventListener","removeEventListener","handleWindowChange","LivePreviewContext","Provider","setToolbarPosition","toolbarPosition","DndContext","collisionDetection","customCollisionDetection","onDragEnd"],"mappings":"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"}