UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

208 lines (207 loc) • 20.6 kB
"use strict"; 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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"}