@getdraft/plugin 1.14.0 → 1.15.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/.eslintrc CHANGED
@@ -1,3 +1,3 @@
1
1
  {
2
- "extends": "@letty/eslint-config-ts"
3
- }
2
+ "extends": "@getdraft/eslint-config-ts"
3
+ }
package/README.md CHANGED
@@ -22,8 +22,10 @@ yarn add @draft-email/plugin --exact
22
22
 
23
23
  ### HTML-script
24
24
 
25
- ```
26
- <script src="https://eimage.sendsay.ru/app/se/draft-plugin-v1.12.1.js"></script>
25
+ Подключите плагин напрямую через CDN:
26
+
27
+ ```html
28
+ <script src="https://eimage.sendsay.ru/app/se/draft/plugin-v1.14.js"></script>
27
29
  ```
28
30
 
29
31
  ## Инициализация
@@ -169,6 +171,7 @@ draft.start({
169
171
  | `history` | Добавлено изменение/Произошло перемещение по истории | <code>{<br>&nbsp;hasUndos: boolean;<br>&nbsp;hasRedos: boolean;<br>&nbsp;type: 'change'\|'undo'\|'redo'<br>&nbsp;}</code> |
170
172
  | `uploadImage` | В редактор пришло изображение для загрузки | |
171
173
  | `openGallery` | Открыта галерея | |
174
+ | `loadPersonalization` | Редактор запрашивает данные персонализации. Асинхронный обработчик — должен вернуть словарь персонализации | Обработчик должен вернуть: <code>Promise<{<br>&nbsp;items: Array&lt;<br>&nbsp;&nbsp;{ name: string; value: string }<br>&nbsp;&nbsp;\| { groupName: string; items: Array&lt;{ name: string; value: string }&gt; }<br>&nbsp;&gt;<br>}></code> |
172
175
 
173
176
  ### **container - опциональный**
174
177
 
package/dist/index.cjs.js CHANGED
@@ -1,2 +1,2 @@
1
- "use strict";var P=Object.defineProperty;var L=(t,e,o)=>e in t?P(t,e,{enumerable:!0,configurable:!0,writable:!0,value:o}):t[e]=o;var a=(t,e,o)=>L(t,typeof e!="symbol"?e+"":e,o);Object.defineProperties(exports,{__esModule:{value:!0},[Symbol.toStringTag]:{value:"Module"}});const b="http://localhost:5174",I="https://seplugin.sendsay.ru",A=async t=>{try{const e=await fetch(t,{method:"HEAD"});if(!e.ok)return null;const o=e.headers.get("Content-Length");return o?parseInt(o,10):null}catch{return null}},g=t=>new Promise((e,o)=>{const n=new Image;n.onload=async()=>{const i=await A(t)??0;e({originalWidth:n.width,originalHeight:n.height,ratio:n.width/n.height,fileSize:i})},n.onerror=i=>{o(i)},n.src=t}),u=(t,e)=>{if(t.length!==1)return t;if(t.charCodeAt(0)>=880){if(e.indexOf("Key")===0&&e.length===4)return e.charAt(3);if(e.indexOf("Digit")===0&&e.length===6)return e.charAt(5)}return t},C=t=>{let e=`${u(t.key,t.code)}`.toLowerCase()||t.key;const o=navigator.userAgent.includes("Mac OS");e==="backspace"&&o&&(e="delete");const n=[];return(o&&t.metaKey||!o&&t.ctrlKey)&&n.push("mod"),t.shiftKey&&n.push("shift"),n.length>0&&(e=`${n.join("-")}-${u(e,t.code)}`),e},H=new Set(["escape","mod-c","mod-d","mod-z","mod-y","mod-shift-z","mod-v","delete","mod-g","mod-p","mod-s"]),D="1.14.0",O={version:D},M=()=>{var s,r;const t=new RegExp("(^| )seplugin-dev=([^;]+)"),e=document.cookie.match(t),o=(r=e==null?void 0:(s=e[2]).toLowerCase)==null?void 0:r.call(s);if(o==="true")return b;if(o!=null&&o.startsWith("https://"))return o;const[n,i]=O.version.split(".");return`${I}/v${n}.${i}`},z=({token:t,pluginId:e,apiUrl:o})=>{if(!t)throw new Error("No [token] was provided");if(!e)throw new Error("No [pluginId] was provided");if(!o)throw new Error("No [apiUrl] was provided")},T=()=>{const t=document.createElement("iframe");return t.src=M(),t.setAttribute("style","height:100%;width:100%;min-width:960px;border:0px"),t.setAttribute("allow","clipboard-read; clipboard-write"),t.setAttribute("sandbox","allow-scripts allow-same-origin allow-forms allow-popups"),t};class x{constructor(e){a(this,"iframe",null);a(this,"hotkeysAttached",!1);a(this,"onHostHotkey",e=>{var d;if(!this.captureHotkeysEnabled()||!((d=this.iframe)!=null&&d.contentWindow))return;const o=C(e);if(!H.has(o))return;const n=e.target;if(n&&(n.isContentEditable||["INPUT","TEXTAREA","SELECT"].includes(n.tagName)))return;e.preventDefault();const{key:i,ctrlKey:s,metaKey:r,shiftKey:l,altKey:c}=e;this.postEvent("hostHotkey",{key:i,code:e.code,metaKey:r,ctrlKey:s,shiftKey:l,altKey:c,keyCode:o})});a(this,"onImageUpload",async({id:e,...o})=>{var n,i;try{const s=await((i=(n=this.config.on).uploadImage)==null?void 0:i.call(n,o));let r={};s&&(r=await g(s)),this.postEvent("imageUploaded",{id:e,img:{...r,url:s}})}catch{this.postEvent("imageUploaded",{id:e})}});a(this,"onLoadPersonalization",async()=>{try{if(!this.config.on.loadPersonalization)throw new Error("No personalization loader provided");const e=await this.config.on.loadPersonalization();this.postEvent("loadPersonalization",{items:e})}catch{this.postEvent("loadPersonalization",{})}});a(this,"onGalleryOpen",async({id:e})=>{var n,i;const o=async s=>{const r={url:s};if(s){const l=await g(s);Object.assign(r,l)}this.postEvent("imageUploaded",{id:e,img:r})};(i=(n=this.config.on).openGallery)==null||i.call(n,o)});a(this,"onMessage",e=>{const{data:{source:o,event:n,...i}}=e;o==="DraftPlugin"&&(n==="openGallery"?this.onGalleryOpen(i):n==="uploadImage"?this.onImageUpload(i):n==="loadPersonalization"?this.onLoadPersonalization():this.triggerEvent(this.config.on[n],i))});a(this,"destroy",()=>{var e;(e=this.iframe)==null||e.remove(),this.iframe=null,window.removeEventListener("message",this.onMessage),this.hotkeysAttached&&(window.removeEventListener("keydown",this.onHostHotkey),this.hotkeysAttached=!1)});this.config=e}captureHotkeysEnabled(){return this.config.disableHotkeysPassing!==!0}promisedIframeMethod(e){let o=null;const n=new Promise((i,s)=>{const r=l=>{const{data:{source:c,event:d,...h}}=l;c==="DraftPlugin"&&d===e&&(l.stopPropagation(),o&&(clearTimeout(o),o=null),i(h),window.removeEventListener("message",r,!0))};window.addEventListener("message",r,!0),o=setTimeout(()=>{window.removeEventListener("message",r,!0),s("Request timed out"),o=null},2e3)});return this.postEvent(e),n}postEvent(e,o={}){this.iframe&&this.iframe.contentWindow&&this.iframe.contentWindow.postMessage({...o,event:e,source:"DraftPlugin"},"*")}triggerEvent(e,o){if(e)return e(o)}start({template:e,templateName:o,uid:n}){const i=document.querySelector(this.config.container);if(i){const s=({data:{source:r,event:l}})=>{if(r==="DraftPlugin"&&l==="loaded"){const{on:c,container:d,locale:h,autosave:w,token:y,apiUrl:E,pluginId:v,__config:k}=this.config;z(this.config),this.iframe&&this.iframe.contentWindow&&this.postEvent("init",{container:d,locale:h,uid:n,autosave:w,token:y,pluginId:v,apiUrl:E,__config:k,enabledListeners:Object.keys(c),template:e,templateName:o}),window.addEventListener("message",this.onMessage),window.removeEventListener("message",s)}};this.iframe=T(),window.addEventListener("message",s),i.appendChild(this.iframe),this.captureHotkeysEnabled()&&!this.hotkeysAttached&&(window.addEventListener("keydown",this.onHostHotkey),this.hotkeysAttached=!0)}else throw new Error("Specified container not found")}changeTemplate(e,o={}){this.postEvent("changeTemplate",{...o,template:e})}toggleGrid(e){this.postEvent("toggleGrid",{state:e})}togglePreview(e){this.postEvent("togglePreview",{state:e})}toggleMenu(e=null){this.postEvent("toggleMenu",{menu:e})}toggleViewMode(e){this.postEvent("toggleViewMode",{viewMode:e})}undo(){this.postEvent("undo")}redo(){this.postEvent("redo")}save(){return this.promisedIframeMethod("saveAction")}exit(){return this.promisedIframeMethod("exitAction")}exportContent(){return this.promisedIframeMethod("exportContent")}}var p=(t=>(t.Audit="audit",t.Guide="guide",t.Style="style",t.Blocks="blocks",t))(p||{}),f=(t=>(t.Desktop="desktop",t.Mobile="mobile",t))(f||{});const N={container:"#draft-plugin-container",disableHotkeysPassing:!1,locale:"ru",on:{}};class K{create(e){return new x({...N,...e})}}const m=new K;window.DraftPlugin=m;exports.EditorMenu=p;exports.ViewMode=f;exports.default=m;
1
+ "use strict";var D=Object.defineProperty;var H=(o,e,t)=>e in o?D(o,e,{enumerable:!0,configurable:!0,writable:!0,value:t}):o[e]=t;var a=(o,e,t)=>H(o,typeof e!="symbol"?e+"":e,t);Object.defineProperties(exports,{__esModule:{value:!0},[Symbol.toStringTag]:{value:"Module"}});const C="http://localhost:5174",M="https://seplugin.sendsay.ru";class E extends Error{constructor(t,i){super(i);a(this,"code");this.name="DraftError",this.code=t}}var L=(o=>(o.Audit="audit",o.Guide="guide",o.Style="style",o.Blocks="blocks",o))(L||{}),O=(o=>(o.Desktop="desktop",o.Mobile="mobile",o))(O||{});const x=async o=>{try{const e=await fetch(o,{method:"HEAD"});if(!e.ok)return null;const t=e.headers.get("Content-Length");return t?parseInt(t,10):null}catch{return null}},P=o=>new Promise((e,t)=>{const i=new Image;i.onload=async()=>{const n=await x(o)??0;e({originalWidth:i.width,originalHeight:i.height,ratio:i.width/i.height,fileSize:n})},i.onerror=n=>{t(n)},i.src=o}),A=(o,e)=>{if(o.length!==1)return o;if(o.charCodeAt(0)>=880){if(e.indexOf("Key")===0&&e.length===4)return e.charAt(3);if(e.indexOf("Digit")===0&&e.length===6)return e.charAt(5)}return o},z=o=>{let e=`${A(o.key,o.code)}`.toLowerCase()||o.key;const t=navigator.userAgent.includes("Mac OS");e==="backspace"&&t&&(e="delete");const i=[];return(t&&o.metaKey||!t&&o.ctrlKey)&&i.push("mod"),o.shiftKey&&i.push("shift"),i.length>0&&(e=`${i.join("-")}-${A(e,o.code)}`),e},U=new Set(["escape","mod-c","mod-d","mod-z","mod-y","mod-shift-z","mod-v","delete","mod-g","mod-p","mod-s"]),I="1.15.1",_={version:I},K=()=>{var r,s;const o=new RegExp("(^| )seplugin-dev=([^;]+)"),e=document.cookie.match(o),t=(s=e==null?void 0:(r=e[2]).toLowerCase)==null?void 0:s.call(r);if(t==="true")return C;if(t!=null&&t.startsWith("https://"))return t;const[i,n]=_.version.split(".");return`${M}/v${i}.${n}`},S=({token:o,pluginId:e,apiUrl:t})=>{if(!o)throw new Error("No [token] was provided");if(!e)throw new Error("No [pluginId] was provided");if(!t)throw new Error("No [apiUrl] was provided")},N=o=>{const e=document.createElement("iframe"),t=new URL(K());return t.searchParams.set("parentOrigin",o),e.src=t.toString(),e.setAttribute("style","height:100%;width:100%;min-width:960px;border:0px;"),e.setAttribute("allow","clipboard-read; clipboard-write"),e.setAttribute("sandbox","allow-scripts allow-same-origin allow-forms allow-popups allow-downloads"),e},f=class f{constructor(e){a(this,"iframe",null);a(this,"hotkeysAttached",!1);a(this,"iframeOrigin",null);a(this,"onHostHotkey",e=>{var l;if(!this.captureHotkeysEnabled()||!((l=this.iframe)!=null&&l.contentWindow))return;const t=z(e);if(!U.has(t))return;const i=e.target;if(i&&(i.isContentEditable||["INPUT","TEXTAREA","SELECT"].includes(i.tagName)))return;e.preventDefault();const{key:n,ctrlKey:r,metaKey:s,shiftKey:c,altKey:h}=e;this.postEvent("hostHotkey",{key:n,code:e.code,metaKey:s,ctrlKey:r,shiftKey:c,altKey:h,keyCode:t})});a(this,"onImageUpload",async({id:e,...t})=>{var i,n;try{const r=await((n=(i=this.config.on).uploadImage)==null?void 0:n.call(i,t));let s={};r&&(s=await P(r)),this.postEvent("imageUploaded",{id:e,img:{...s,url:r}})}catch{this.postEvent("imageUploaded",{id:e})}});a(this,"onLoadPersonalization",async()=>{try{if(!this.config.on.loadPersonalization)throw new Error("No personalization loader provided");const e=await this.config.on.loadPersonalization();this.postEvent("loadPersonalization",{items:e})}catch{this.postEvent("loadPersonalization",{})}});a(this,"onGalleryOpen",async({id:e})=>{var i,n;const t=async r=>{const s={url:r};if(r){const c=await P(r);Object.assign(s,c)}this.postEvent("imageUploaded",{id:e,img:s})};(n=(i=this.config.on).openGallery)==null||n.call(i,t)});a(this,"onMessage",e=>{var r;if(e.source!==((r=this.iframe)==null?void 0:r.contentWindow))return;const{data:{source:t,event:i,...n}}=e;t==="DraftPlugin"&&(i==="openGallery"?this.onGalleryOpen(n):i==="uploadImage"?this.onImageUpload(n):i==="loadPersonalization"?this.onLoadPersonalization():this.triggerEvent(this.config.on[i],n))});a(this,"destroy",()=>{var e;(e=this.iframe)==null||e.remove(),this.iframe=null,window.removeEventListener("message",this.onMessage),this.hotkeysAttached&&(window.removeEventListener("keydown",this.onHostHotkey),this.hotkeysAttached=!1)});this.config=e}captureHotkeysEnabled(){return this.config.disableHotkeysPassing!==!0}promisedIframeMethod(e){let t=null,i=null;const n=()=>{t&&(clearTimeout(t),t=null),i&&(window.removeEventListener("message",i,!0),i=null)},r=new Promise((s,c)=>{const h=()=>{t&&clearTimeout(t),t=setTimeout(()=>{n(),c(new E("timeout_error",`Action "${e}" timed out`))},f.HEARTBEAT_TIMEOUT)};i=l=>{var g;if(l.source!==((g=this.iframe)==null?void 0:g.contentWindow))return;const{data:d}=l;if(d.source==="DraftPlugin"){if(d.event==="actionPing"&&d.action===e){l.stopPropagation(),h();return}if(d.event===e){l.stopPropagation(),n();const{source:v,event:k,ok:u,code:p,message:m,...w}=d;u===!1?c(new E(p||"unknown_error",m||"Unknown error")):s(w)}}},window.addEventListener("message",i,!0),h()});return this.postEvent(e),r}postEvent(e,t={}){if(this.iframe&&this.iframe.contentWindow&&this.iframeOrigin)this.iframe.contentWindow.postMessage({...t,event:e,source:"DraftPlugin"},this.iframeOrigin);else if(!this.iframeOrigin)throw new Error("Iframe origin is not initialized")}triggerEvent(e,t){if(e)return e(t)}start({template:e,templateName:t,uid:i}){const n=document.querySelector(this.config.container);if(n){const r=s=>{var l;const{data:{source:c,event:h}}=s;if(c==="DraftPlugin"&&h==="loaded"){if(s.source!==((l=this.iframe)==null?void 0:l.contentWindow))return;this.iframeOrigin=s.origin;const{on:d,container:g,locale:v,autosave:k,token:u,apiUrl:p,pluginId:m,authData:w,__config:b}=this.config;S(this.config),this.iframe&&this.iframe.contentWindow&&this.postEvent("init",{container:g,locale:v,uid:i,autosave:k,token:u,pluginId:m,authData:w,apiUrl:p,__config:b,enabledListeners:Object.keys(d),template:e,templateName:t}),window.addEventListener("message",this.onMessage),window.removeEventListener("message",r)}};this.iframe=N(window.location.origin),window.addEventListener("message",r),n.appendChild(this.iframe),this.captureHotkeysEnabled()&&!this.hotkeysAttached&&(window.addEventListener("keydown",this.onHostHotkey),this.hotkeysAttached=!0)}else throw new Error("Specified container not found")}changeTemplate(e,t={}){this.postEvent("changeTemplate",{...t,template:e})}toggleGrid(e){this.postEvent("toggleGrid",{state:e})}togglePreview(e){this.postEvent("togglePreview",{state:e})}toggleMenu(e=null){this.postEvent("toggleMenu",{menu:e})}toggleViewMode(e){this.postEvent("toggleViewMode",{viewMode:e})}undo(){this.postEvent("undo")}redo(){this.postEvent("redo")}save(){return this.promisedIframeMethod("saveAction")}exit(){return this.promisedIframeMethod("exitAction")}exportContent(){return this.promisedIframeMethod("exportContent")}exportAsFile(){return this.promisedIframeMethod("exportAsFile")}};a(f,"HEARTBEAT_TIMEOUT",4e3);let y=f;const W={container:"#draft-plugin-container",disableHotkeysPassing:!1,locale:"ru",on:{}};class G{constructor(){a(this,"version",I)}create(e){return new y({...W,...e})}}const T=new G;window.DraftPlugin=T;exports.DraftError=E;exports.EditorMenu=L;exports.ViewMode=O;exports.default=T;
2
2
  //# sourceMappingURL=index.cjs.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.cjs.js","sources":["../src/routes.ts","../src/utils.ts","../src/plugin.ts","../src/types.ts","../src/index.ts"],"sourcesContent":["export const LOCAL_ENDPOINT = 'http://localhost:5174';\nexport const PROD_ENDPOINT = 'https://seplugin.sendsay.ru';\n","import { ImageParams } from './types';\n\nconst getImageFileSize = async (imageUrl: string) => {\n try {\n const response = await fetch(imageUrl, { method: 'HEAD' });\n\n if (!response.ok) {\n return null;\n }\n\n const contentLength = response.headers.get('Content-Length');\n\n if (contentLength) {\n return parseInt(contentLength, 10);\n } else {\n return null;\n }\n } catch (error) {\n return null;\n }\n};\n\nexport const getImageParamsFromFileUrl = (imgSrc: string) =>\n new Promise<Omit<ImageParams, 'url'>>((resolve, reject) => {\n const img = new Image();\n\n img.onload = async () => {\n const fileSize = (await getImageFileSize(imgSrc)) ?? 0;\n\n resolve({\n originalWidth: img.width,\n originalHeight: img.height,\n ratio: img.width / img.height,\n fileSize,\n });\n };\n\n img.onerror = (error) => {\n reject(error);\n };\n\n img.src = imgSrc;\n });\n\nconst getLatinKey = (key: string, code: string) => {\n if (key.length !== 1) {\n return key;\n }\n\n const capitalHetaCode = 880;\n const isNonLatin = key.charCodeAt(0) >= capitalHetaCode;\n\n if (isNonLatin) {\n if (code.indexOf('Key') === 0 && code.length === 4) {\n return code.charAt(3);\n }\n\n if (code.indexOf('Digit') === 0 && code.length === 6) {\n return code.charAt(5);\n }\n }\n\n return key;\n};\n\nexport const constructKeyCode = (event: KeyboardEvent) => {\n let keyCode =\n `${getLatinKey(event.key, event.code)}`.toLowerCase() || event.key;\n const macOs = navigator.userAgent.includes('Mac OS');\n\n if (keyCode === 'backspace' && macOs) {\n keyCode = 'delete';\n }\n\n const prefix = [];\n\n if ((macOs && event.metaKey) || (!macOs && event.ctrlKey)) {\n prefix.push('mod');\n }\n\n if (event.shiftKey) {\n prefix.push('shift');\n }\n\n if (prefix.length > 0) {\n keyCode = `${prefix.join('-')}-${getLatinKey(keyCode, event.code)}`;\n }\n\n return keyCode;\n};\n\nexport const CORE_HOTKEYS = new Set([\n 'escape',\n 'mod-c',\n 'mod-d',\n 'mod-z',\n 'mod-y',\n 'mod-shift-z',\n 'mod-v',\n 'delete',\n 'mod-g',\n 'mod-p',\n 'mod-s',\n]);\n","import { LOCAL_ENDPOINT, PROD_ENDPOINT } from './routes';\nimport {\n EditorMenu,\n ExitParams,\n OnInitParams,\n OpenGalleryParams,\n Handlers,\n PluginConfig,\n SaveParams,\n TemplateJSON,\n UploadImageParams,\n ViewMode,\n HandlerParams,\n} from './types';\nimport {\n getImageParamsFromFileUrl,\n constructKeyCode,\n CORE_HOTKEYS,\n} from './utils';\nimport pkg from '../package.json';\n\nconst getDeployLink = () => {\n const regex = new RegExp(`(^| )seplugin-dev=([^;]+)`);\n const match = document.cookie.match(regex);\n const value = match?.[2].toLowerCase?.();\n\n if (value === 'true') {\n return LOCAL_ENDPOINT;\n }\n\n if (value?.startsWith('https://')) {\n return value;\n }\n\n const [major, mid] = pkg.version.split('.');\n\n return `${PROD_ENDPOINT}/v${major}.${mid}`;\n};\n\nconst verifyConfig = ({ token, pluginId, apiUrl }: PluginConfig) => {\n if (!token) {\n throw new Error('No [token] was provided');\n }\n\n if (!pluginId) {\n throw new Error('No [pluginId] was provided');\n }\n\n if (!apiUrl) {\n throw new Error('No [apiUrl] was provided');\n }\n};\n\nconst createIframe = () => {\n const iframe = document.createElement('iframe');\n iframe.src = getDeployLink();\n iframe.setAttribute(\n 'style',\n 'height:100%;width:100%;min-width:960px;border:0px',\n );\n iframe.setAttribute('allow', 'clipboard-read; clipboard-write');\n iframe.setAttribute(\n 'sandbox',\n 'allow-scripts allow-same-origin allow-forms allow-popups',\n );\n\n return iframe;\n};\n\nexport class PluginInstance {\n public iframe: HTMLIFrameElement | null = null;\n private hotkeysAttached = false;\n constructor(public config: PluginConfig) {}\n\n private captureHotkeysEnabled() {\n return this.config.disableHotkeysPassing !== true;\n }\n\n private onHostHotkey = (event: KeyboardEvent) => {\n if (!this.captureHotkeysEnabled() || !this.iframe?.contentWindow) {\n return;\n }\n\n const keyCode = constructKeyCode(event);\n\n if (!CORE_HOTKEYS.has(keyCode)) {\n return;\n }\n\n const target = event.target as HTMLElement | null;\n\n if (\n target &&\n (target.isContentEditable ||\n ['INPUT', 'TEXTAREA', 'SELECT'].includes(target.tagName))\n ) {\n return;\n }\n\n event.preventDefault();\n\n const { key, ctrlKey, metaKey, shiftKey, altKey } = event;\n\n this.postEvent('hostHotkey', {\n key,\n code: event.code,\n metaKey,\n ctrlKey,\n shiftKey,\n altKey,\n keyCode,\n });\n };\n\n private promisedIframeMethod<T>(eventName: string) {\n let timer: ReturnType<typeof setTimeout> | null = null;\n\n const res = new Promise<T>((resolve, reject) => {\n const onEvent = (\n e: MessageEvent<{ source: string; event: keyof PluginConfig['on'] }>,\n ) => {\n const {\n data: { source, event, ...data },\n } = e;\n if (source === 'DraftPlugin' && event === eventName) {\n e.stopPropagation();\n\n if (timer) {\n clearTimeout(timer);\n timer = null;\n }\n resolve(data as T);\n window.removeEventListener('message', onEvent, true);\n }\n };\n\n window.addEventListener('message', onEvent, true);\n\n timer = setTimeout(() => {\n window.removeEventListener('message', onEvent, true);\n reject('Request timed out');\n timer = null;\n }, 2000);\n });\n\n this.postEvent(eventName);\n\n return res;\n }\n\n private postEvent(event: string, data: object = {}) {\n if (this.iframe && this.iframe.contentWindow) {\n this.iframe.contentWindow.postMessage(\n {\n ...data,\n event,\n source: 'DraftPlugin',\n },\n '*',\n );\n }\n }\n\n private onImageUpload = async ({\n id,\n ...data\n }: UploadImageParams & { id: string }) => {\n try {\n const url = await this.config.on.uploadImage?.(data);\n let params = {};\n\n if (url) {\n params = await getImageParamsFromFileUrl(url);\n }\n\n this.postEvent('imageUploaded', { id, img: { ...params, url } });\n } catch (e) {\n this.postEvent('imageUploaded', { id });\n }\n };\n\n private onLoadPersonalization = async () => {\n try {\n if (!this.config.on.loadPersonalization) {\n throw new Error('No personalization loader provided');\n }\n\n const personalizationDictionary =\n await this.config.on.loadPersonalization();\n\n this.postEvent('loadPersonalization', {\n items: personalizationDictionary,\n });\n } catch (e) {\n this.postEvent('loadPersonalization', {});\n }\n };\n\n private onGalleryOpen = async ({ id }: { id: string }) => {\n const applyImage = async (url?: string) => {\n const imgParams = { url };\n\n if (url) {\n const params = await getImageParamsFromFileUrl(url);\n Object.assign(imgParams, params);\n }\n\n this.postEvent('imageUploaded', {\n id,\n img: imgParams,\n });\n };\n\n this.config.on.openGallery?.(applyImage);\n };\n\n private triggerEvent<H extends CallableFunction | undefined>(\n handler: H,\n params: HandlerParams<H>,\n ) {\n if (!handler) {\n return;\n }\n\n return handler(params);\n }\n\n private onMessage = (\n ev: MessageEvent<{ source: string; event: keyof Handlers }>,\n ) => {\n const {\n data: { source, event, ...data },\n } = ev;\n\n if (source === 'DraftPlugin') {\n if (event === 'openGallery') {\n this.onGalleryOpen(data as OpenGalleryParams);\n } else if (event === 'uploadImage') {\n this.onImageUpload(data as UploadImageParams & { id: string });\n } else if (event === 'loadPersonalization') {\n this.onLoadPersonalization();\n } else {\n this.triggerEvent(\n this.config.on[event],\n data as HandlerParams<Handlers[typeof event]>,\n );\n }\n }\n };\n\n start({\n template,\n templateName,\n uid,\n }: {\n template: string | TemplateJSON;\n uid: string;\n templateName?: string;\n }) {\n const containerEl = document.querySelector(this.config.container);\n\n if (!containerEl) {\n throw new Error('Specified container not found');\n } else {\n const onInit = ({ data: { source, event } }: OnInitParams) => {\n if (source === 'DraftPlugin' && event === 'loaded') {\n const {\n on,\n container,\n locale,\n autosave,\n token,\n apiUrl,\n pluginId,\n __config,\n } = this.config;\n\n verifyConfig(this.config);\n\n if (this.iframe && this.iframe.contentWindow) {\n this.postEvent('init', {\n container,\n locale,\n uid,\n autosave,\n token,\n pluginId,\n apiUrl,\n __config,\n enabledListeners: Object.keys(on),\n template,\n templateName,\n });\n }\n\n window.addEventListener('message', this.onMessage);\n window.removeEventListener('message', onInit);\n }\n };\n\n this.iframe = createIframe();\n window.addEventListener('message', onInit);\n containerEl.appendChild(this.iframe);\n\n if (this.captureHotkeysEnabled() && !this.hotkeysAttached) {\n window.addEventListener('keydown', this.onHostHotkey);\n this.hotkeysAttached = true;\n }\n }\n }\n\n destroy = () => {\n this.iframe?.remove();\n this.iframe = null;\n window.removeEventListener('message', this.onMessage);\n\n if (this.hotkeysAttached) {\n window.removeEventListener('keydown', this.onHostHotkey);\n this.hotkeysAttached = false;\n }\n };\n\n changeTemplate(\n template: string | TemplateJSON,\n options: {\n templateName?: string;\n uid?: string;\n } = {},\n ) {\n this.postEvent('changeTemplate', { ...options, template });\n }\n\n toggleGrid(state?: boolean) {\n this.postEvent('toggleGrid', { state });\n }\n\n togglePreview(state?: boolean) {\n this.postEvent('togglePreview', { state });\n }\n\n toggleMenu(menu: EditorMenu | null = null) {\n this.postEvent('toggleMenu', { menu });\n }\n\n toggleViewMode(viewMode: ViewMode) {\n this.postEvent('toggleViewMode', { viewMode });\n }\n\n undo() {\n this.postEvent('undo');\n }\n\n redo() {\n this.postEvent('redo');\n }\n\n save() {\n return this.promisedIframeMethod<SaveParams>('saveAction');\n }\n\n exit() {\n return this.promisedIframeMethod<ExitParams>('exitAction');\n }\n\n exportContent() {\n return this.promisedIframeMethod<SaveParams>('exportContent');\n }\n}\n","export type EventHandler<T> = (data: T) => void;\n\nexport type AsyncEventHandler<P, R> = (params: P) => Promise<R>;\n\nexport interface SaveParams {\n json: TemplateJSON;\n html: string;\n}\n\nexport type OnInitParams = MessageEvent<{\n source: string;\n event: 'loaded' | unknown;\n}>;\n\nexport enum EditorMenu {\n Audit = 'audit',\n Guide = 'guide',\n Style = 'style',\n Blocks = 'blocks',\n}\n\nexport enum ViewMode {\n Desktop = 'desktop',\n Mobile = 'mobile',\n}\n\nexport interface ReadyParams extends SaveParams {\n lastSavedTemplate?: {\n template: TemplateJSON;\n date: Date;\n };\n}\n\nexport interface ErrorParams {\n message: string;\n code?: string;\n}\n\nexport interface ExitParams extends SaveParams {}\n\nexport interface NotificationParams {\n intent: 'error' | 'info' | 'success' | 'warning';\n errorDetails?: object;\n message?: string;\n title: string;\n}\n\nexport interface OpenGalleryParams {\n id: string;\n}\n\nexport interface UploadImageParams {\n file: string;\n name: string;\n}\n\nexport interface ImageParams {\n url: string;\n originalWidth: number;\n originalHeight: number;\n ratio: number;\n fileSize: number;\n}\n\nexport interface AutosaveParams {\n json: TemplateJSON;\n}\n\ninterface ToggleParams {\n state: boolean;\n}\n\nexport interface ToggleGridParams extends ToggleParams {}\n\nexport interface TogglePreviewParams extends ToggleParams {\n json: TemplateJSON;\n html: string;\n darkHTML: string;\n}\n\nexport interface ToggleViewModeParams {\n mode: ViewMode;\n}\n\nexport interface ToggleMenuParams {\n menu: EditorMenu | null;\n}\n\nexport interface HistoryParams {\n hasUndos: boolean;\n hasRedos: boolean;\n currentStep: number;\n}\n\nexport type TemplateJSON = Record<string, any>;\n\nexport type PersonalizationDictionary = Array<\n | { name: string; value: string }\n | { groupName: string; items: Array<{ name: string; value: string }> }\n>;\n\nexport type ToolbarActionName =\n | 'duplicate'\n | 'destroy'\n | 'edit'\n | 'up'\n | 'down';\nexport type HotkeyContext = 'hotkey';\nexport type OverlayContext = 'overlay-click' | 'overlay-double-click';\nexport type ToolbarContext = 'toolbar';\nexport type UnitSettingsContext = 'unit-settings';\nexport type BlockType = 'system' | 'user';\nexport type DndDragContext = 'canvas' | 'list' | 'palette';\nexport type DndDropContext = 'dropline' | 'placeholder' | 'empty-placeholder';\nexport type DndDropTarget = 'root' | 'section' | 'column' | 'element';\n\nexport interface DuplicateEvent {\n name: 'duplicate';\n context: HotkeyContext | ToolbarContext | UnitSettingsContext;\n unitType: string;\n}\n\nexport interface DestroyEvent {\n name: 'destroy';\n context: HotkeyContext | ToolbarContext | UnitSettingsContext;\n unitType: string;\n}\n\nexport interface EditEvent {\n name: 'edit';\n context: OverlayContext | ToolbarContext;\n unitType: string;\n}\n\nexport interface HistoryEvent {\n name: 'history';\n type: 'undo' | 'redo';\n context: HotkeyContext;\n}\n\nexport interface PreviewEvent {\n name: 'preview';\n context: HotkeyContext;\n}\n\nexport interface ToolbarMoveEvent {\n name: 'up' | 'down';\n context: ToolbarContext;\n unitType: string;\n}\n\nexport interface BlockInsertEvent {\n name: 'block-insert';\n context: 'double-click';\n blockType: BlockType;\n blockName: string;\n blockCategory: string | number;\n blockId: string | number;\n}\n\nexport interface BlockSaveEvent {\n name: 'block-save';\n blockType: 'user';\n blockName: string;\n blockCategory: string | number;\n blockId: string | number;\n elements: Record<string, number> & { sum: number };\n}\n\nexport interface BlockDestroyEvent {\n name: 'block-destroy';\n}\n\nexport interface DropEventDragMeta {\n context?: DndDragContext;\n layer?: 'element' | 'section' | 'column' | 'root';\n type?: string | string[];\n blockCategory?: string | number;\n blockId?: string | number;\n blockName?: string;\n blockType?: BlockType;\n alt?: boolean;\n dropKind?: 'create' | 'move';\n}\n\nexport interface DropEventDropzoneMeta {\n context?: DndDropContext;\n dropTarget?: DndDropTarget;\n}\n\nexport interface DropEvent {\n name: 'drop';\n drag?: DropEventDragMeta;\n dropzone?: DropEventDropzoneMeta;\n}\n\nexport type AnalyticsParams =\n | DuplicateEvent\n | DestroyEvent\n | EditEvent\n | HistoryEvent\n | PreviewEvent\n | ToolbarMoveEvent\n | BlockInsertEvent\n | BlockSaveEvent\n | BlockDestroyEvent\n | DropEvent;\n\nexport interface Handlers {\n ready: EventHandler<ReadyParams>;\n error: EventHandler<ErrorParams>;\n save: EventHandler<SaveParams>;\n exit: EventHandler<ExitParams>;\n notification: EventHandler<NotificationParams>;\n autosave: EventHandler<AutosaveParams>;\n toggleGrid: EventHandler<ToggleGridParams>;\n togglePreview: EventHandler<TogglePreviewParams>;\n toggleViewMode: EventHandler<ToggleViewModeParams>;\n toggleMenu: EventHandler<ToggleMenuParams>;\n history: EventHandler<HistoryParams>;\n analytics: EventHandler<AnalyticsParams>;\n openGallery: EventHandler<(url?: string) => Promise<void>>;\n uploadImage: AsyncEventHandler<UploadImageParams, string>;\n loadPersonalization: AsyncEventHandler<void, PersonalizationDictionary>;\n}\n\nexport type HandlerParams<H> = H extends\n | EventHandler<infer P>\n | AsyncEventHandler<infer P, unknown>\n ? P\n : never;\n\nexport interface PluginConfig {\n container: string;\n locale?: string;\n autosave?:\n | boolean\n | {\n interval: number;\n };\n token: string;\n pluginId: string;\n apiUrl: string;\n disableHotkeysPassing?: boolean;\n __config?: object;\n on: Partial<Handlers>;\n}\n","import { PluginInstance } from './plugin';\nimport { PluginConfig } from './types';\n\nconst DEFAULT_CONFIG: Partial<PluginConfig> = {\n container: '#draft-plugin-container',\n disableHotkeysPassing: false,\n locale: 'ru',\n on: {},\n};\n\nclass PluginWrapper {\n create(config: PluginConfig) {\n return new PluginInstance({\n ...DEFAULT_CONFIG,\n ...config,\n });\n }\n}\n\nconst DraftPlugin = new PluginWrapper();\n\ndeclare global {\n interface Window {\n DraftPlugin: PluginWrapper;\n }\n}\n\nwindow.DraftPlugin = DraftPlugin;\n\nexport default DraftPlugin;\nexport * from './types';\n"],"names":["LOCAL_ENDPOINT","PROD_ENDPOINT","getImageFileSize","imageUrl","response","contentLength","getImageParamsFromFileUrl","imgSrc","resolve","reject","img","fileSize","error","getLatinKey","key","code","constructKeyCode","event","keyCode","macOs","prefix","CORE_HOTKEYS","getDeployLink","regex","match","value","_b","_a","major","mid","pkg","verifyConfig","token","pluginId","apiUrl","createIframe","iframe","PluginInstance","config","__publicField","target","ctrlKey","metaKey","shiftKey","altKey","id","data","url","params","personalizationDictionary","applyImage","imgParams","ev","source","eventName","timer","res","onEvent","e","handler","template","templateName","uid","containerEl","onInit","on","container","locale","autosave","__config","options","state","menu","viewMode","EditorMenu","ViewMode","DEFAULT_CONFIG","PluginWrapper","DraftPlugin"],"mappings":"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"}
1
+ {"version":3,"file":"index.cjs.js","sources":["../src/routes.ts","../src/types.ts","../src/utils.ts","../src/plugin.ts","../src/index.ts"],"sourcesContent":["export const LOCAL_ENDPOINT = 'http://localhost:5174';\nexport const PROD_ENDPOINT = 'https://seplugin.sendsay.ru';\n","export type EventHandler<T> = (data: T) => void;\n\nexport type AsyncEventHandler<P, R> = (params: P) => Promise<R>;\n\nexport class DraftError extends Error {\n public readonly code: string;\n\n constructor(code: string, message: string) {\n super(message);\n this.name = 'DraftError';\n this.code = code;\n }\n}\n\nexport type MessageData = {\n source: string;\n event: string;\n ok?: boolean;\n code?: string;\n message?: string;\n action?: string;\n};\n\nexport interface ExportAsFileParams {\n blob: Blob;\n fileName: string;\n}\n\nexport interface SaveParams {\n json: TemplateJSON;\n html: string;\n}\n\nexport type OnInitParams = MessageEvent<{\n source: string;\n event: 'loaded' | unknown;\n}>;\n\nexport enum EditorMenu {\n Audit = 'audit',\n Guide = 'guide',\n Style = 'style',\n Blocks = 'blocks',\n}\n\nexport enum ViewMode {\n Desktop = 'desktop',\n Mobile = 'mobile',\n}\n\nexport interface ReadyParams extends SaveParams {\n lastSavedTemplate?: {\n template: TemplateJSON;\n date: Date;\n };\n}\n\nexport interface ErrorParams {\n message: string;\n code?: string;\n details?: unknown;\n}\n\nexport interface ExitParams extends SaveParams {}\n\nexport interface NotificationParams {\n intent: 'error' | 'info' | 'success' | 'warning';\n errorDetails?: object;\n message?: string;\n title: string;\n}\n\nexport interface OpenGalleryParams {\n id: string;\n}\n\nexport interface UploadImageParams {\n file: string;\n name: string;\n}\n\nexport interface ImageParams {\n url: string;\n originalWidth: number;\n originalHeight: number;\n ratio: number;\n fileSize: number;\n}\n\nexport interface AutosaveParams {\n json: TemplateJSON;\n}\n\ninterface ToggleParams {\n state: boolean;\n}\n\nexport interface ToggleGridParams extends ToggleParams {}\n\nexport interface TogglePreviewParams extends ToggleParams {\n json: TemplateJSON;\n html: string;\n darkHTML: string;\n}\n\nexport interface ToggleViewModeParams {\n mode: ViewMode;\n}\n\nexport interface ToggleMenuParams {\n menu: EditorMenu | null;\n}\n\nexport interface HistoryParams {\n hasUndos: boolean;\n hasRedos: boolean;\n currentStep: number;\n}\n\nexport type TemplateJSON = Record<string, any>;\n\nexport type PersonalizationDictionary = Array<\n | { name: string; value: string }\n | { groupName: string; items: Array<{ name: string; value: string }> }\n>;\n\nexport type ToolbarActionName =\n | 'duplicate'\n | 'destroy'\n | 'edit'\n | 'up'\n | 'down';\nexport type HotkeyContext = 'hotkey';\nexport type OverlayContext = 'overlay-click' | 'overlay-double-click';\nexport type ToolbarContext = 'toolbar';\nexport type UnitSettingsContext = 'unit-settings';\nexport type BlockType = 'system' | 'user';\nexport type DndDragContext = 'canvas' | 'list' | 'palette';\nexport type DndDropContext = 'dropline' | 'placeholder' | 'empty-placeholder';\nexport type DndDropTarget = 'root' | 'section' | 'column' | 'element';\n\nexport interface DuplicateEvent {\n name: 'duplicate';\n context: HotkeyContext | ToolbarContext | UnitSettingsContext;\n unitType: string;\n}\n\nexport interface DestroyEvent {\n name: 'destroy';\n context: HotkeyContext | ToolbarContext | UnitSettingsContext;\n unitType: string;\n}\n\nexport interface EditEvent {\n name: 'edit';\n context: OverlayContext | ToolbarContext;\n unitType: string;\n}\n\nexport interface HistoryEvent {\n name: 'history';\n type: 'undo' | 'redo';\n context: HotkeyContext;\n}\n\nexport interface PreviewEvent {\n name: 'preview';\n context: HotkeyContext;\n}\n\nexport interface ToolbarMoveEvent {\n name: 'up' | 'down';\n context: ToolbarContext;\n unitType: string;\n}\n\nexport interface BlockInsertEvent {\n name: 'block-insert';\n context: 'double-click';\n blockType: BlockType;\n blockName: string;\n blockCategory: string | number;\n blockId: string | number;\n}\n\nexport interface BlockSaveEvent {\n name: 'block-save';\n blockType: 'user';\n blockName: string;\n blockCategory: string | number;\n blockId: string | number;\n elements: Record<string, number> & { sum: number };\n}\n\nexport interface BlockDestroyEvent {\n name: 'block-destroy';\n}\n\nexport interface DropEventDragMeta {\n context?: DndDragContext;\n layer?: 'element' | 'section' | 'column' | 'root';\n type?: string | string[];\n blockCategory?: string | number;\n blockId?: string | number;\n blockName?: string;\n blockType?: BlockType;\n alt?: boolean;\n dropKind?: 'create' | 'move';\n}\n\nexport interface DropEventDropzoneMeta {\n context?: DndDropContext;\n dropTarget?: DndDropTarget;\n}\n\nexport interface DropEvent {\n name: 'drop';\n drag?: DropEventDragMeta;\n dropzone?: DropEventDropzoneMeta;\n}\n\nexport type AnalyticsParams =\n | DuplicateEvent\n | DestroyEvent\n | EditEvent\n | HistoryEvent\n | PreviewEvent\n | ToolbarMoveEvent\n | BlockInsertEvent\n | BlockSaveEvent\n | BlockDestroyEvent\n | DropEvent;\n\nexport interface Handlers {\n ready: EventHandler<ReadyParams>;\n error: EventHandler<ErrorParams>;\n save: EventHandler<SaveParams>;\n exit: EventHandler<ExitParams>;\n notification: EventHandler<NotificationParams>;\n autosave: EventHandler<AutosaveParams>;\n toggleGrid: EventHandler<ToggleGridParams>;\n togglePreview: EventHandler<TogglePreviewParams>;\n toggleViewMode: EventHandler<ToggleViewModeParams>;\n toggleMenu: EventHandler<ToggleMenuParams>;\n history: EventHandler<HistoryParams>;\n analytics: EventHandler<AnalyticsParams>;\n openGallery: EventHandler<(url?: string) => Promise<void>>;\n uploadImage: AsyncEventHandler<UploadImageParams, string>;\n loadPersonalization: AsyncEventHandler<void, PersonalizationDictionary>;\n}\n\nexport type HandlerParams<H> = H extends\n | EventHandler<infer P>\n | AsyncEventHandler<infer P, unknown>\n ? P\n : never;\n\nexport interface PluginConfig {\n container: string;\n locale?: string;\n autosave?:\n | boolean\n | {\n interval: number;\n };\n token: string;\n authData?: {\n orgId: number;\n projectId?: number;\n };\n pluginId: string;\n apiUrl: string;\n disableHotkeysPassing?: boolean;\n __config?: object;\n on: Partial<Handlers>;\n}\n","import { ImageParams } from './types';\n\nconst getImageFileSize = async (imageUrl: string) => {\n try {\n const response = await fetch(imageUrl, { method: 'HEAD' });\n\n if (!response.ok) {\n return null;\n }\n\n const contentLength = response.headers.get('Content-Length');\n\n if (contentLength) {\n return parseInt(contentLength, 10);\n } else {\n return null;\n }\n } catch (error) {\n return null;\n }\n};\n\nexport const getImageParamsFromFileUrl = (imgSrc: string) =>\n new Promise<Omit<ImageParams, 'url'>>((resolve, reject) => {\n const img = new Image();\n\n img.onload = async () => {\n const fileSize = (await getImageFileSize(imgSrc)) ?? 0;\n\n resolve({\n originalWidth: img.width,\n originalHeight: img.height,\n ratio: img.width / img.height,\n fileSize,\n });\n };\n\n img.onerror = (error) => {\n reject(error);\n };\n\n img.src = imgSrc;\n });\n\nconst getLatinKey = (key: string, code: string) => {\n if (key.length !== 1) {\n return key;\n }\n\n const capitalHetaCode = 880;\n const isNonLatin = key.charCodeAt(0) >= capitalHetaCode;\n\n if (isNonLatin) {\n if (code.indexOf('Key') === 0 && code.length === 4) {\n return code.charAt(3);\n }\n\n if (code.indexOf('Digit') === 0 && code.length === 6) {\n return code.charAt(5);\n }\n }\n\n return key;\n};\n\nexport const constructKeyCode = (event: KeyboardEvent) => {\n let keyCode =\n `${getLatinKey(event.key, event.code)}`.toLowerCase() || event.key;\n const macOs = navigator.userAgent.includes('Mac OS');\n\n if (keyCode === 'backspace' && macOs) {\n keyCode = 'delete';\n }\n\n const prefix = [];\n\n if ((macOs && event.metaKey) || (!macOs && event.ctrlKey)) {\n prefix.push('mod');\n }\n\n if (event.shiftKey) {\n prefix.push('shift');\n }\n\n if (prefix.length > 0) {\n keyCode = `${prefix.join('-')}-${getLatinKey(keyCode, event.code)}`;\n }\n\n return keyCode;\n};\n\nexport const CORE_HOTKEYS = new Set([\n 'escape',\n 'mod-c',\n 'mod-d',\n 'mod-z',\n 'mod-y',\n 'mod-shift-z',\n 'mod-v',\n 'delete',\n 'mod-g',\n 'mod-p',\n 'mod-s',\n]);\n","import { LOCAL_ENDPOINT, PROD_ENDPOINT } from './routes';\nimport {\n DraftError,\n EditorMenu,\n ExitParams,\n OnInitParams,\n OpenGalleryParams,\n Handlers,\n PluginConfig,\n SaveParams,\n TemplateJSON,\n UploadImageParams,\n ViewMode,\n HandlerParams,\n MessageData,\n ExportAsFileParams,\n} from './types';\nimport {\n getImageParamsFromFileUrl,\n constructKeyCode,\n CORE_HOTKEYS,\n} from './utils';\nimport pkg from '../package.json';\n\nconst getDeployLink = () => {\n const regex = new RegExp(`(^| )seplugin-dev=([^;]+)`);\n const match = document.cookie.match(regex);\n const value = match?.[2].toLowerCase?.();\n\n if (value === 'true') {\n return LOCAL_ENDPOINT;\n }\n\n if (value?.startsWith('https://')) {\n return value;\n }\n\n const [major, mid] = pkg.version.split('.');\n\n return `${PROD_ENDPOINT}/v${major}.${mid}`;\n};\n\nconst verifyConfig = ({ token, pluginId, apiUrl }: PluginConfig) => {\n if (!token) {\n throw new Error('No [token] was provided');\n }\n\n if (!pluginId) {\n throw new Error('No [pluginId] was provided');\n }\n\n if (!apiUrl) {\n throw new Error('No [apiUrl] was provided');\n }\n};\n\nconst createIframe = (parentOrigin: string) => {\n const iframe = document.createElement('iframe');\n const url = new URL(getDeployLink());\n url.searchParams.set('parentOrigin', parentOrigin);\n iframe.src = url.toString();\n iframe.setAttribute(\n 'style',\n 'height:100%;width:100%;min-width:960px;border:0px;',\n );\n iframe.setAttribute('allow', 'clipboard-read; clipboard-write');\n iframe.setAttribute(\n 'sandbox',\n 'allow-scripts allow-same-origin allow-forms allow-popups allow-downloads',\n );\n\n return iframe;\n};\n\nexport class PluginInstance {\n public iframe: HTMLIFrameElement | null = null;\n private hotkeysAttached = false;\n private iframeOrigin: string | null = null;\n constructor(public config: PluginConfig) {}\n\n private captureHotkeysEnabled() {\n return this.config.disableHotkeysPassing !== true;\n }\n\n private onHostHotkey = (event: KeyboardEvent) => {\n if (!this.captureHotkeysEnabled() || !this.iframe?.contentWindow) {\n return;\n }\n\n const keyCode = constructKeyCode(event);\n\n if (!CORE_HOTKEYS.has(keyCode)) {\n return;\n }\n\n const target = event.target as HTMLElement | null;\n\n if (\n target &&\n (target.isContentEditable ||\n ['INPUT', 'TEXTAREA', 'SELECT'].includes(target.tagName))\n ) {\n return;\n }\n\n event.preventDefault();\n\n const { key, ctrlKey, metaKey, shiftKey, altKey } = event;\n\n this.postEvent('hostHotkey', {\n key,\n code: event.code,\n metaKey,\n ctrlKey,\n shiftKey,\n altKey,\n keyCode,\n });\n };\n\n private static HEARTBEAT_TIMEOUT = 4000;\n\n private promisedIframeMethod<T>(eventName: string) {\n let timer: ReturnType<typeof setTimeout> | null = null;\n let listener: ((e: MessageEvent<MessageData>) => void) | null = null;\n\n const cleanup = () => {\n if (timer) {\n clearTimeout(timer);\n timer = null;\n }\n\n if (listener) {\n window.removeEventListener('message', listener, true);\n listener = null;\n }\n };\n\n const res = new Promise<T>((resolve, reject) => {\n const resetTimer = () => {\n if (timer) clearTimeout(timer);\n timer = setTimeout(() => {\n cleanup();\n reject(\n new DraftError('timeout_error', `Action \"${eventName}\" timed out`),\n );\n }, PluginInstance.HEARTBEAT_TIMEOUT);\n };\n\n listener = (e: MessageEvent<MessageData>) => {\n if (e.source !== this.iframe?.contentWindow) return;\n\n const { data } = e;\n if (data.source !== 'DraftPlugin') return;\n\n if (data.event === 'actionPing' && data.action === eventName) {\n e.stopPropagation();\n resetTimer();\n\n return;\n }\n\n if (data.event === eventName) {\n e.stopPropagation();\n cleanup();\n\n const { source: _1, event: _2, ok, code, message, ...rest } = data;\n\n if (ok === false) {\n reject(\n new DraftError(\n code || 'unknown_error',\n message || 'Unknown error',\n ),\n );\n } else {\n resolve(rest as T);\n }\n }\n };\n\n window.addEventListener('message', listener, true);\n resetTimer();\n });\n\n this.postEvent(eventName);\n\n return res;\n }\n\n private postEvent(event: string, data: object = {}) {\n if (this.iframe && this.iframe.contentWindow && this.iframeOrigin) {\n this.iframe.contentWindow.postMessage(\n {\n ...data,\n event,\n source: 'DraftPlugin',\n },\n this.iframeOrigin,\n );\n } else if (!this.iframeOrigin) {\n throw new Error('Iframe origin is not initialized');\n }\n }\n\n private onImageUpload = async ({\n id,\n ...data\n }: UploadImageParams & { id: string }) => {\n try {\n const url = await this.config.on.uploadImage?.(data);\n let params = {};\n\n if (url) {\n params = await getImageParamsFromFileUrl(url);\n }\n\n this.postEvent('imageUploaded', { id, img: { ...params, url } });\n } catch (e) {\n this.postEvent('imageUploaded', { id });\n }\n };\n\n private onLoadPersonalization = async () => {\n try {\n if (!this.config.on.loadPersonalization) {\n throw new Error('No personalization loader provided');\n }\n\n const personalizationDictionary =\n await this.config.on.loadPersonalization();\n\n this.postEvent('loadPersonalization', {\n items: personalizationDictionary,\n });\n } catch (e) {\n this.postEvent('loadPersonalization', {});\n }\n };\n\n private onGalleryOpen = async ({ id }: { id: string }) => {\n const applyImage = async (url?: string) => {\n const imgParams = { url };\n\n if (url) {\n const params = await getImageParamsFromFileUrl(url);\n Object.assign(imgParams, params);\n }\n\n this.postEvent('imageUploaded', {\n id,\n img: imgParams,\n });\n };\n\n this.config.on.openGallery?.(applyImage);\n };\n\n private triggerEvent<H extends CallableFunction | undefined>(\n handler: H,\n params: HandlerParams<H>,\n ) {\n if (!handler) {\n return;\n }\n\n return handler(params);\n }\n\n private onMessage = (\n ev: MessageEvent<{ source: string; event: keyof Handlers }>,\n ) => {\n if (ev.source !== this.iframe?.contentWindow) {\n return;\n }\n\n const {\n data: { source, event, ...data },\n } = ev;\n\n if (source === 'DraftPlugin') {\n if (event === 'openGallery') {\n this.onGalleryOpen(data as OpenGalleryParams);\n } else if (event === 'uploadImage') {\n this.onImageUpload(data as UploadImageParams & { id: string });\n } else if (event === 'loadPersonalization') {\n this.onLoadPersonalization();\n } else {\n this.triggerEvent(\n this.config.on[event],\n data as HandlerParams<Handlers[typeof event]>,\n );\n }\n }\n };\n\n start({\n template,\n templateName,\n uid,\n }: {\n template: string | TemplateJSON;\n uid: string | number;\n templateName?: string;\n }) {\n const containerEl = document.querySelector(this.config.container);\n\n if (!containerEl) {\n throw new Error('Specified container not found');\n } else {\n const onInit = (msg: OnInitParams) => {\n const {\n data: { source, event },\n } = msg;\n if (source === 'DraftPlugin' && event === 'loaded') {\n if (msg.source !== this.iframe?.contentWindow) return;\n\n this.iframeOrigin = msg.origin;\n\n const {\n on,\n container,\n locale,\n autosave,\n token,\n apiUrl,\n pluginId,\n authData,\n __config,\n } = this.config;\n\n verifyConfig(this.config);\n\n if (this.iframe && this.iframe.contentWindow) {\n this.postEvent('init', {\n container,\n locale,\n uid,\n autosave,\n token,\n pluginId,\n authData,\n apiUrl,\n __config,\n enabledListeners: Object.keys(on),\n template,\n templateName,\n });\n }\n\n window.addEventListener('message', this.onMessage);\n window.removeEventListener('message', onInit);\n }\n };\n\n this.iframe = createIframe(window.location.origin);\n window.addEventListener('message', onInit);\n containerEl.appendChild(this.iframe);\n\n if (this.captureHotkeysEnabled() && !this.hotkeysAttached) {\n window.addEventListener('keydown', this.onHostHotkey);\n this.hotkeysAttached = true;\n }\n }\n }\n\n destroy = () => {\n this.iframe?.remove();\n this.iframe = null;\n window.removeEventListener('message', this.onMessage);\n\n if (this.hotkeysAttached) {\n window.removeEventListener('keydown', this.onHostHotkey);\n this.hotkeysAttached = false;\n }\n };\n\n changeTemplate(\n template: string | TemplateJSON,\n options: {\n templateName?: string;\n uid?: string;\n } = {},\n ) {\n this.postEvent('changeTemplate', { ...options, template });\n }\n\n toggleGrid(state?: boolean) {\n this.postEvent('toggleGrid', { state });\n }\n\n togglePreview(state?: boolean) {\n this.postEvent('togglePreview', { state });\n }\n\n toggleMenu(menu: EditorMenu | null = null) {\n this.postEvent('toggleMenu', { menu });\n }\n\n toggleViewMode(viewMode: ViewMode) {\n this.postEvent('toggleViewMode', { viewMode });\n }\n\n undo() {\n this.postEvent('undo');\n }\n\n redo() {\n this.postEvent('redo');\n }\n\n save() {\n return this.promisedIframeMethod<SaveParams>('saveAction');\n }\n\n exit() {\n return this.promisedIframeMethod<ExitParams>('exitAction');\n }\n\n exportContent() {\n return this.promisedIframeMethod<SaveParams>('exportContent');\n }\n\n exportAsFile() {\n return this.promisedIframeMethod<ExportAsFileParams>('exportAsFile');\n }\n}\n","import { PluginInstance } from './plugin';\nimport { PluginConfig } from './types';\nimport { version } from '../package.json';\n\nconst DEFAULT_CONFIG: Partial<PluginConfig> = {\n container: '#draft-plugin-container',\n disableHotkeysPassing: false,\n locale: 'ru',\n on: {},\n};\n\nclass PluginWrapper {\n public version = version;\n\n create(config: PluginConfig) {\n return new PluginInstance({\n ...DEFAULT_CONFIG,\n ...config,\n });\n }\n}\n\nconst DraftPlugin = new PluginWrapper();\n\ndeclare global {\n interface Window {\n DraftPlugin: PluginWrapper;\n }\n}\n\nwindow.DraftPlugin = DraftPlugin;\n\nexport default DraftPlugin;\nexport type { PluginInstance } from './plugin';\nexport * from './types';\n"],"names":["LOCAL_ENDPOINT","PROD_ENDPOINT","DraftError","code","message","__publicField","EditorMenu","ViewMode","getImageFileSize","imageUrl","response","contentLength","getImageParamsFromFileUrl","imgSrc","resolve","reject","img","fileSize","error","getLatinKey","key","constructKeyCode","event","keyCode","macOs","prefix","CORE_HOTKEYS","getDeployLink","regex","match","value","_b","_a","major","mid","pkg","verifyConfig","token","pluginId","apiUrl","createIframe","parentOrigin","iframe","url","_PluginInstance","config","target","ctrlKey","metaKey","shiftKey","altKey","id","data","params","personalizationDictionary","applyImage","imgParams","ev","source","eventName","timer","listener","cleanup","res","resetTimer","e","_1","_2","ok","rest","handler","template","templateName","uid","containerEl","onInit","msg","on","container","locale","autosave","authData","__config","options","state","menu","viewMode","PluginInstance","DEFAULT_CONFIG","PluginWrapper","version","DraftPlugin"],"mappings":"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"}