UNPKG

payload

Version:

Node, React and MongoDB Headless CMS and Application Framework

111 lines (110 loc) 12.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); Object.defineProperty(exports, "usePopupWindow", { enumerable: true, get: function() { return usePopupWindow; } }); const _react = require("react"); const _Config = require("../../utilities/Config"); const usePopupWindow = (props)=>{ const { eventType, onMessage, url } = props; const isReceivingMessage = (0, _react.useRef)(false); const [isOpen, setIsOpen] = (0, _react.useState)(false); const { serverURL } = (0, _Config.useConfig)(); const popupRef = (0, _react.useRef)(null); // Optionally broadcast messages back out to the parent component (0, _react.useEffect)(()=>{ const receiveMessage = async (event)=>{ if (event.origin !== window.location.origin || event.origin !== url || event.origin !== serverURL) { // console.warn(`Message received by ${event.origin}; IGNORED.`) // eslint-disable-line no-console return; } if (typeof onMessage === 'function' && event.data?.type === eventType && !isReceivingMessage.current) { isReceivingMessage.current = true; await onMessage(event.data?.searchParams); isReceivingMessage.current = false; } }; if (isOpen && popupRef.current) { window.addEventListener('message', receiveMessage, false); } return ()=>{ window.removeEventListener('message', receiveMessage); }; }, [ onMessage, eventType, url, serverURL, isOpen ]); // Customize the size, position, and style of the popup window const openPopupWindow = (0, _react.useCallback)((e)=>{ if (e) { e.preventDefault(); } const features = { height: 700, left: 'auto', menubar: 'no', popup: 'yes', toolbar: 'no', top: 'auto', width: 800 }; const popupOptions = Object.entries(features).reduce((str, [key, value])=>{ let strCopy = str; if (value === 'auto') { if (key === 'top') { const v = Math.round(window.innerHeight / 2 - features.height / 2); strCopy += `top=${v},`; } else if (key === 'left') { const v = Math.round(window.innerWidth / 2 - features.width / 2); strCopy += `left=${v},`; } return strCopy; } strCopy += `${key}=${value},`; return strCopy; }, '').slice(0, -1) // remove last ',' (comma) ; const newWindow = window.open(url, '_blank', popupOptions); popupRef.current = newWindow; setIsOpen(true); }, [ url ]); // this is the most stable and widely supported way to check if a popup window is no longer open // we poll its ref every x ms and use the popup window's `closed` property (0, _react.useEffect)(()=>{ let timer; if (isOpen) { timer = setInterval(function() { if (popupRef.current.closed) { clearInterval(timer); setIsOpen(false); } }, 1000); } else { clearInterval(timer); } return ()=>{ if (timer) { clearInterval(timer); } }; }, [ isOpen, popupRef ]); return { isPopupOpen: isOpen, openPopupWindow, popupRef }; }; //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../../../src/admin/components/views/LivePreview/usePopupWindow.ts"],"sourcesContent":["import { useCallback, useEffect, useRef, useState } from 'react'\n\nimport { useConfig } from '../../utilities/Config'\n\nexport interface PopupMessage {\n  searchParams: {\n    [key: string]: string | undefined\n    code: string\n    installation_id: string\n    state: string\n  }\n  type: string\n}\n\nexport const usePopupWindow = (props: {\n  eventType?: string\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  onMessage?: (searchParams: PopupMessage['searchParams']) => Promise<void>\n  url: string\n}): {\n  isPopupOpen: boolean\n  openPopupWindow: () => void\n  popupRef?: React.MutableRefObject<Window | null>\n} => {\n  const { eventType, onMessage, url } = props\n  const isReceivingMessage = useRef(false)\n  const [isOpen, setIsOpen] = useState(false)\n  const { serverURL } = useConfig()\n  const popupRef = useRef<Window | null>(null)\n\n  // Optionally broadcast messages back out to the parent component\n  useEffect(() => {\n    const receiveMessage = async (event: MessageEvent): Promise<void> => {\n      if (\n        event.origin !== window.location.origin ||\n        event.origin !== url ||\n        event.origin !== serverURL\n      ) {\n        // console.warn(`Message received by ${event.origin}; IGNORED.`) // eslint-disable-line no-console\n        return\n      }\n\n      if (\n        typeof onMessage === 'function' &&\n        event.data?.type === eventType &&\n        !isReceivingMessage.current\n      ) {\n        isReceivingMessage.current = true\n        await onMessage(event.data?.searchParams)\n        isReceivingMessage.current = false\n      }\n    }\n\n    if (isOpen && popupRef.current) {\n      window.addEventListener('message', receiveMessage, false)\n    }\n\n    return () => {\n      window.removeEventListener('message', receiveMessage)\n    }\n  }, [onMessage, eventType, url, serverURL, isOpen])\n\n  // Customize the size, position, and style of the popup window\n  const openPopupWindow = useCallback(\n    (e?: MouseEvent) => {\n      if (e) {\n        e.preventDefault()\n      }\n\n      const features = {\n        height: 700,\n        left: 'auto',\n        menubar: 'no',\n        popup: 'yes',\n        toolbar: 'no',\n        top: 'auto',\n        width: 800,\n      }\n\n      const popupOptions = Object.entries(features)\n        .reduce((str, [key, value]) => {\n          let strCopy = str\n          if (value === 'auto') {\n            if (key === 'top') {\n              const v = Math.round(window.innerHeight / 2 - features.height / 2)\n              strCopy += `top=${v},`\n            } else if (key === 'left') {\n              const v = Math.round(window.innerWidth / 2 - features.width / 2)\n              strCopy += `left=${v},`\n            }\n            return strCopy\n          }\n\n          strCopy += `${key}=${value},`\n          return strCopy\n        }, '')\n        .slice(0, -1) // remove last ',' (comma)\n\n      const newWindow = window.open(url, '_blank', popupOptions)\n\n      popupRef.current = newWindow\n\n      setIsOpen(true)\n    },\n    [url],\n  )\n\n  // this is the most stable and widely supported way to check if a popup window is no longer open\n  // we poll its ref every x ms and use the popup window's `closed` property\n  useEffect(() => {\n    let timer: NodeJS.Timeout\n\n    if (isOpen) {\n      timer = setInterval(function () {\n        if (popupRef.current.closed) {\n          clearInterval(timer)\n          setIsOpen(false)\n        }\n      }, 1000)\n    } else {\n      clearInterval(timer)\n    }\n\n    return () => {\n      if (timer) {\n        clearInterval(timer)\n      }\n    }\n  }, [isOpen, popupRef])\n\n  return {\n    isPopupOpen: isOpen,\n    openPopupWindow,\n    popupRef,\n  }\n}\n"],"names":["usePopupWindow","props","eventType","onMessage","url","isReceivingMessage","useRef","isOpen","setIsOpen","useState","serverURL","useConfig","popupRef","useEffect","receiveMessage","event","origin","window","location","data","type","current","searchParams","addEventListener","removeEventListener","openPopupWindow","useCallback","e","preventDefault","features","height","left","menubar","popup","toolbar","top","width","popupOptions","Object","entries","reduce","str","key","value","strCopy","v","Math","round","innerHeight","innerWidth","slice","newWindow","open","timer","setInterval","closed","clearInterval","isPopupOpen"],"mappings":"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"}