@getdraft/plugin 1.15.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/dist/index.cjs.js CHANGED
@@ -1,2 +1,2 @@
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.0",_={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;
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/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 * 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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"}
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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"}
package/dist/index.es.js CHANGED
@@ -61,7 +61,7 @@ const x = async (o) => {
61
61
  "mod-g",
62
62
  "mod-p",
63
63
  "mod-s"
64
- ]), L = "1.15.0", M = {
64
+ ]), L = "1.15.1", M = {
65
65
  version: L
66
66
  }, K = () => {
67
67
  var r, s;
@@ -1 +1 @@
1
- {"version":3,"file":"index.es.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 * 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","_a","_b","regex","match","value","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":";;;AAAO,MAAMA,IAAiB,yBACjBC,IAAgB;ACGtB,MAAMC,UAAmB,MAAM;AAAA,EAGpC,YAAYC,GAAcC,GAAiB;AACzC,UAAMA,CAAO;AAHC,IAAAC,EAAA;AAId,SAAK,OAAO,cACZ,KAAK,OAAOF;AAAA,EACd;AACF;AA0BO,IAAKG,sBAAAA,OACVA,EAAA,QAAQ,SACRA,EAAA,QAAQ,SACRA,EAAA,QAAQ,SACRA,EAAA,SAAS,UAJCA,IAAAA,KAAA,CAAA,CAAA,GAOAC,sBAAAA,OACVA,EAAA,UAAU,WACVA,EAAA,SAAS,UAFCA,IAAAA,KAAA,CAAA,CAAA;AC3CZ,MAAMC,IAAmB,OAAOC,MAAqB;AACnD,MAAI;AACF,UAAMC,IAAW,MAAM,MAAMD,GAAU,EAAE,QAAQ,QAAQ;AAEzD,QAAI,CAACC,EAAS;AACZ,aAAO;AAGT,UAAMC,IAAgBD,EAAS,QAAQ,IAAI,gBAAgB;AAE3D,WAAIC,IACK,SAASA,GAAe,EAAE,IAE1B;AAAA,EAEX,QAAgB;AACd,WAAO;AAAA,EACT;AACF,GAEaC,IAA4B,CAACC,MACxC,IAAI,QAAkC,CAACC,GAASC,MAAW;AACzD,QAAMC,IAAM,IAAI,MAAA;AAEhB,EAAAA,EAAI,SAAS,YAAY;AACvB,UAAMC,IAAY,MAAMT,EAAiBK,CAAM,KAAM;AAErD,IAAAC,EAAQ;AAAA,MACN,eAAeE,EAAI;AAAA,MACnB,gBAAgBA,EAAI;AAAA,MACpB,OAAOA,EAAI,QAAQA,EAAI;AAAA,MACvB,UAAAC;AAAA,IAAA,CACD;AAAA,EACH,GAEAD,EAAI,UAAU,CAACE,MAAU;AACvB,IAAAH,EAAOG,CAAK;AAAA,EACd,GAEAF,EAAI,MAAMH;AACZ,CAAC,GAEGM,IAAc,CAACC,GAAajB,MAAiB;AACjD,MAAIiB,EAAI,WAAW;AACjB,WAAOA;AAMT,MAFmBA,EAAI,WAAW,CAAC,KADX,KAGR;AACd,QAAIjB,EAAK,QAAQ,KAAK,MAAM,KAAKA,EAAK,WAAW;AAC/C,aAAOA,EAAK,OAAO,CAAC;AAGtB,QAAIA,EAAK,QAAQ,OAAO,MAAM,KAAKA,EAAK,WAAW;AACjD,aAAOA,EAAK,OAAO,CAAC;AAAA,EAExB;AAEA,SAAOiB;AACT,GAEaC,IAAmB,CAACC,MAAyB;AACxD,MAAIC,IACF,GAAGJ,EAAYG,EAAM,KAAKA,EAAM,IAAI,CAAC,GAAG,YAAA,KAAiBA,EAAM;AACjE,QAAME,IAAQ,UAAU,UAAU,SAAS,QAAQ;AAEnD,EAAID,MAAY,eAAeC,MAC7BD,IAAU;AAGZ,QAAME,IAAS,CAAA;AAEf,UAAKD,KAASF,EAAM,WAAa,CAACE,KAASF,EAAM,YAC/CG,EAAO,KAAK,KAAK,GAGfH,EAAM,YACRG,EAAO,KAAK,OAAO,GAGjBA,EAAO,SAAS,MAClBF,IAAU,GAAGE,EAAO,KAAK,GAAG,CAAC,IAAIN,EAAYI,GAASD,EAAM,IAAI,CAAC,KAG5DC;AACT,GAEaG,wBAAmB,IAAI;AAAA,EAClC;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,CAAC;;GC/EKC,IAAgB,MAAM;AHxBrB,MAAAC,GAAAC;AGyBL,QAAMC,IAAQ,IAAI,OAAO,2BAA2B,GAC9CC,IAAQ,SAAS,OAAO,MAAMD,CAAK,GACnCE,KAAQH,IAAAE,KAAA,iBAAAH,IAAAG,EAAQ,IAAG,gBAAX,gBAAAF,EAAA,KAAAD;AAEd,MAAII,MAAU;AACZ,WAAOhC;AAGT,MAAIgC,KAAA,QAAAA,EAAO,WAAW;AACpB,WAAOA;AAGT,QAAM,CAACC,GAAOC,CAAG,IAAIC,EAAI,QAAQ,MAAM,GAAG;AAE1C,SAAO,GAAGlC,CAAa,KAAKgC,CAAK,IAAIC,CAAG;AAC1C,GAEME,IAAe,CAAC,EAAE,OAAAC,GAAO,UAAAC,GAAU,QAAAC,QAA2B;AAClE,MAAI,CAACF;AACH,UAAM,IAAI,MAAM,yBAAyB;AAG3C,MAAI,CAACC;AACH,UAAM,IAAI,MAAM,4BAA4B;AAG9C,MAAI,CAACC;AACH,UAAM,IAAI,MAAM,0BAA0B;AAE9C,GAEMC,IAAe,CAACC,MAAyB;AAC7C,QAAMC,IAAS,SAAS,cAAc,QAAQ,GACxCC,IAAM,IAAI,IAAIhB,GAAe;AACnC,SAAAgB,EAAI,aAAa,IAAI,gBAAgBF,CAAY,GACjDC,EAAO,MAAMC,EAAI,SAAA,GACjBD,EAAO;AAAA,IACL;AAAA,IACA;AAAA,EAAA,GAEFA,EAAO,aAAa,SAAS,iCAAiC,GAC9DA,EAAO;AAAA,IACL;AAAA,IACA;AAAA,EAAA,GAGKA;AACT,GAEaE,IAAN,MAAMA,EAAe;AAAA,EAI1B,YAAmBC,GAAsB;AAHlC,IAAAxC,EAAA,gBAAmC;AAClC,IAAAA,EAAA,yBAAkB;AAClB,IAAAA,EAAA,sBAA8B;AAO9B,IAAAA,EAAA,sBAAe,CAACiB,MAAyB;AHpF5C,UAAAM;AGqFH,UAAI,CAAC,KAAK,sBAAA,KAA2B,GAACA,IAAA,KAAK,WAAL,QAAAA,EAAa;AACjD;AAGF,YAAML,IAAUF,EAAiBC,CAAK;AAEtC,UAAI,CAACI,EAAa,IAAIH,CAAO;AAC3B;AAGF,YAAMuB,IAASxB,EAAM;AAErB,UACEwB,MACCA,EAAO,qBACN,CAAC,SAAS,YAAY,QAAQ,EAAE,SAASA,EAAO,OAAO;AAEzD;AAGF,MAAAxB,EAAM,eAAA;AAEN,YAAM,EAAE,KAAAF,GAAK,SAAA2B,GAAS,SAAAC,GAAS,UAAAC,GAAU,QAAAC,MAAW5B;AAEpD,WAAK,UAAU,cAAc;AAAA,QAC3B,KAAAF;AAAA,QACA,MAAME,EAAM;AAAA,QACZ,SAAA0B;AAAA,QACA,SAAAD;AAAA,QACA,UAAAE;AAAA,QACA,QAAAC;AAAA,QACA,SAAA3B;AAAA,MAAA,CACD;AAAA,IACH;AAuFQ,IAAAlB,EAAA,uBAAgB,OAAO;AAAA,MAC7B,IAAA8C;AAAA,MACA,GAAGC;AAAA,IAAA,MACqC;AHhNrC,UAAAxB,GAAAC;AGiNH,UAAI;AACF,cAAMc,IAAM,QAAMd,KAAAD,IAAA,KAAK,OAAO,IAAG,gBAAf,gBAAAC,EAAA,KAAAD,GAA6BwB;AAC/C,YAAIC,IAAS,CAAA;AAEb,QAAIV,MACFU,IAAS,MAAMzC,EAA0B+B,CAAG,IAG9C,KAAK,UAAU,iBAAiB,EAAE,IAAAQ,GAAI,KAAK,EAAE,GAAGE,GAAQ,KAAAV,EAAA,GAAO;AAAA,MACjE,QAAY;AACV,aAAK,UAAU,iBAAiB,EAAE,IAAAQ,EAAA,CAAI;AAAA,MACxC;AAAA,IACF;AAEQ,IAAA9C,EAAA,+BAAwB,YAAY;AAC1C,UAAI;AACF,YAAI,CAAC,KAAK,OAAO,GAAG;AAClB,gBAAM,IAAI,MAAM,oCAAoC;AAGtD,cAAMiD,IACJ,MAAM,KAAK,OAAO,GAAG,oBAAA;AAEvB,aAAK,UAAU,uBAAuB;AAAA,UACpC,OAAOA;AAAA,QAAA,CACR;AAAA,MACH,QAAY;AACV,aAAK,UAAU,uBAAuB,EAAE;AAAA,MAC1C;AAAA,IACF;AAEQ,IAAAjD,EAAA,uBAAgB,OAAO,EAAE,IAAA8C,QAAyB;AHhPrD,UAAAvB,GAAAC;AGiPH,YAAM0B,IAAa,OAAOZ,MAAiB;AACzC,cAAMa,IAAY,EAAE,KAAAb,EAAA;AAEpB,YAAIA,GAAK;AACP,gBAAMU,IAAS,MAAMzC,EAA0B+B,CAAG;AAClD,iBAAO,OAAOa,GAAWH,CAAM;AAAA,QACjC;AAEA,aAAK,UAAU,iBAAiB;AAAA,UAC9B,IAAAF;AAAA,UACA,KAAKK;AAAA,QAAA,CACN;AAAA,MACH;AAEA,OAAA3B,KAAAD,IAAA,KAAK,OAAO,IAAG,gBAAf,QAAAC,EAAA,KAAAD,GAA6B2B;AAAA,IAC/B;AAaQ,IAAAlD,EAAA,mBAAY,CAClBoD,MACG;AH/QA,UAAA7B;AGgRH,UAAI6B,EAAG,aAAW7B,IAAA,KAAK,WAAL,gBAAAA,EAAa;AAC7B;AAGF,YAAM;AAAA,QACJ,MAAM,EAAE,QAAA8B,GAAQ,OAAApC,GAAO,GAAG8B,EAAA;AAAA,MAAK,IAC7BK;AAEJ,MAAIC,MAAW,kBACTpC,MAAU,gBACZ,KAAK,cAAc8B,CAAyB,IACnC9B,MAAU,gBACnB,KAAK,cAAc8B,CAA0C,IACpD9B,MAAU,wBACnB,KAAK,sBAAA,IAEL,KAAK;AAAA,QACH,KAAK,OAAO,GAAGA,CAAK;AAAA,QACpB8B;AAAA,MAAA;AAAA,IAIR;AAwEA,IAAA/C,EAAA,iBAAU,MAAM;AH9WX,UAAAuB;AG+WH,OAAAA,IAAA,KAAK,WAAL,QAAAA,EAAa,UACb,KAAK,SAAS,MACd,OAAO,oBAAoB,WAAW,KAAK,SAAS,GAEhD,KAAK,oBACP,OAAO,oBAAoB,WAAW,KAAK,YAAY,GACvD,KAAK,kBAAkB;AAAA,IAE3B;AAzSmB,SAAA,SAAAiB;AAAA,EAAuB;AAAA,EAElC,wBAAwB;AAC9B,WAAO,KAAK,OAAO,0BAA0B;AAAA,EAC/C;AAAA,EAwCQ,qBAAwBc,GAAmB;AACjD,QAAIC,IAA8C,MAC9CC,IAA4D;AAEhE,UAAMC,IAAU,MAAM;AACpB,MAAIF,MACF,aAAaA,CAAK,GAClBA,IAAQ,OAGNC,MACF,OAAO,oBAAoB,WAAWA,GAAU,EAAI,GACpDA,IAAW;AAAA,IAEf,GAEME,IAAM,IAAI,QAAW,CAACjD,GAASC,MAAW;AAC9C,YAAMiD,IAAa,MAAM;AACvB,QAAIJ,kBAAoBA,CAAK,GAC7BA,IAAQ,WAAW,MAAM;AACvB,UAAAE,EAAA,GACA/C;AAAA,YACE,IAAIb,EAAW,iBAAiB,WAAWyD,CAAS,aAAa;AAAA,UAAA;AAAA,QAErE,GAAGf,EAAe,iBAAiB;AAAA,MACrC;AAEA,MAAAiB,IAAW,CAACI,MAAiC;AHrJ5C,YAAArC;AGsJC,YAAIqC,EAAE,aAAWrC,IAAA,KAAK,WAAL,gBAAAA,EAAa,eAAe;AAE7C,cAAM,EAAE,MAAAwB,MAASa;AACjB,YAAIb,EAAK,WAAW,eAEpB;AAAA,cAAIA,EAAK,UAAU,gBAAgBA,EAAK,WAAWO,GAAW;AAC5D,YAAAM,EAAE,gBAAA,GACFD,EAAA;AAEA;AAAA,UACF;AAEA,cAAIZ,EAAK,UAAUO,GAAW;AAC5B,YAAAM,EAAE,gBAAA,GACFH,EAAA;AAEA,kBAAM,EAAE,QAAQI,GAAI,OAAOC,GAAI,IAAAC,GAAI,MAAAjE,GAAM,SAAAC,GAAS,GAAGiE,EAAA,IAASjB;AAE9D,YAAIgB,MAAO,KACTrD;AAAA,cACE,IAAIb;AAAA,gBACFC,KAAQ;AAAA,gBACRC,KAAW;AAAA,cAAA;AAAA,YACb,IAGFU,EAAQuD,CAAS;AAAA,UAErB;AAAA;AAAA,MACF,GAEA,OAAO,iBAAiB,WAAWR,GAAU,EAAI,GACjDG,EAAA;AAAA,IACF,CAAC;AAED,gBAAK,UAAUL,CAAS,GAEjBI;AAAA,EACT;AAAA,EAEQ,UAAUzC,GAAe8B,IAAe,IAAI;AAClD,QAAI,KAAK,UAAU,KAAK,OAAO,iBAAiB,KAAK;AACnD,WAAK,OAAO,cAAc;AAAA,QACxB;AAAA,UACE,GAAGA;AAAA,UACH,OAAA9B;AAAA,UACA,QAAQ;AAAA,QAAA;AAAA,QAEV,KAAK;AAAA,MAAA;AAAA,aAEE,CAAC,KAAK;AACf,YAAM,IAAI,MAAM,kCAAkC;AAAA,EAEtD;AAAA,EAuDQ,aACNgD,GACAjB,GACA;AACA,QAAKiB;AAIL,aAAOA,EAAQjB,CAAM;AAAA,EACvB;AAAA,EA6BA,MAAM;AAAA,IACJ,UAAAkB;AAAA,IACA,cAAAC;AAAA,IACA,KAAAC;AAAA,EAAA,GAKC;AACD,UAAMC,IAAc,SAAS,cAAc,KAAK,OAAO,SAAS;AAEhE,QAAKA,GAEE;AACL,YAAMC,IAAS,CAACC,MAAsB;AHtTrC,YAAAhD;AGuTC,cAAM;AAAA,UACJ,MAAM,EAAE,QAAA8B,GAAQ,OAAApC,EAAA;AAAA,QAAM,IACpBsD;AACJ,YAAIlB,MAAW,iBAAiBpC,MAAU,UAAU;AAClD,cAAIsD,EAAI,aAAWhD,IAAA,KAAK,WAAL,gBAAAA,EAAa,eAAe;AAE/C,eAAK,eAAegD,EAAI;AAExB,gBAAM;AAAA,YACJ,IAAAC;AAAA,YACA,WAAAC;AAAA,YACA,QAAAC;AAAA,YACA,UAAAC;AAAA,YACA,OAAA3C;AAAA,YACA,QAAAE;AAAA,YACA,UAAAD;AAAA,YACA,UAAA2C;AAAA,YACA,UAAAC;AAAA,UAAA,IACE,KAAK;AAET,UAAA9C,EAAa,KAAK,MAAM,GAEpB,KAAK,UAAU,KAAK,OAAO,iBAC7B,KAAK,UAAU,QAAQ;AAAA,YACrB,WAAA0C;AAAA,YACA,QAAAC;AAAA,YACA,KAAAN;AAAA,YACA,UAAAO;AAAA,YACA,OAAA3C;AAAA,YACA,UAAAC;AAAA,YACA,UAAA2C;AAAA,YACA,QAAA1C;AAAA,YACA,UAAA2C;AAAA,YACA,kBAAkB,OAAO,KAAKL,CAAE;AAAA,YAChC,UAAAN;AAAA,YACA,cAAAC;AAAA,UAAA,CACD,GAGH,OAAO,iBAAiB,WAAW,KAAK,SAAS,GACjD,OAAO,oBAAoB,WAAWG,CAAM;AAAA,QAC9C;AAAA,MACF;AAEA,WAAK,SAASnC,EAAa,OAAO,SAAS,MAAM,GACjD,OAAO,iBAAiB,WAAWmC,CAAM,GACzCD,EAAY,YAAY,KAAK,MAAM,GAE/B,KAAK,sBAAA,KAA2B,CAAC,KAAK,oBACxC,OAAO,iBAAiB,WAAW,KAAK,YAAY,GACpD,KAAK,kBAAkB;AAAA,IAE3B;AAvDE,YAAM,IAAI,MAAM,+BAA+B;AAAA,EAwDnD;AAAA,EAaA,eACEH,GACAY,IAGI,IACJ;AACA,SAAK,UAAU,kBAAkB,EAAE,GAAGA,GAAS,UAAAZ,GAAU;AAAA,EAC3D;AAAA,EAEA,WAAWa,GAAiB;AAC1B,SAAK,UAAU,cAAc,EAAE,OAAAA,EAAA,CAAO;AAAA,EACxC;AAAA,EAEA,cAAcA,GAAiB;AAC7B,SAAK,UAAU,iBAAiB,EAAE,OAAAA,EAAA,CAAO;AAAA,EAC3C;AAAA,EAEA,WAAWC,IAA0B,MAAM;AACzC,SAAK,UAAU,cAAc,EAAE,MAAAA,EAAA,CAAM;AAAA,EACvC;AAAA,EAEA,eAAeC,GAAoB;AACjC,SAAK,UAAU,kBAAkB,EAAE,UAAAA,EAAA,CAAU;AAAA,EAC/C;AAAA,EAEA,OAAO;AACL,SAAK,UAAU,MAAM;AAAA,EACvB;AAAA,EAEA,OAAO;AACL,SAAK,UAAU,MAAM;AAAA,EACvB;AAAA,EAEA,OAAO;AACL,WAAO,KAAK,qBAAiC,YAAY;AAAA,EAC3D;AAAA,EAEA,OAAO;AACL,WAAO,KAAK,qBAAiC,YAAY;AAAA,EAC3D;AAAA,EAEA,gBAAgB;AACd,WAAO,KAAK,qBAAiC,eAAe;AAAA,EAC9D;AAAA,EAEA,eAAe;AACb,WAAO,KAAK,qBAAyC,cAAc;AAAA,EACrE;AACF;AAlTEjF,EA9CWuC,GA8CI,qBAAoB;AA9C9B,IAAM2C,IAAN3C;ACtEP,MAAM4C,IAAwC;AAAA,EAC5C,WAAW;AAAA,EACX,uBAAuB;AAAA,EACvB,QAAQ;AAAA,EACR,IAAI,CAAA;AACN;AAEA,MAAMC,EAAc;AAAA,EAApB;AACS,IAAApF,EAAA,iBAAUqF;AAAA;AAAA,EAEjB,OAAO7C,GAAsB;AAC3B,WAAO,IAAI0C,EAAe;AAAA,MACxB,GAAGC;AAAA,MACH,GAAG3C;AAAA,IAAA,CACJ;AAAA,EACH;AACF;AAEA,MAAM8C,IAAc,IAAIF,EAAA;AAQxB,OAAO,cAAcE;"}
1
+ {"version":3,"file":"index.es.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","_a","_b","regex","match","value","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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"}
@@ -1,2 +1,2 @@
1
- var G=Object.defineProperty;var F=(l,d,f)=>d in l?G(l,d,{enumerable:!0,configurable:!0,writable:!0,value:f}):l[d]=f;var a=(l,d,f)=>F(l,typeof d!="symbol"?d+"":d,f);var DraftPlugin=(function(l){"use strict";const d="http://localhost:5174",f="https://seplugin.sendsay.ru";class w extends Error{constructor(t,o){super(o);a(this,"code");this.name="DraftError",this.code=t}}var A=(i=>(i.Audit="audit",i.Guide="guide",i.Style="style",i.Blocks="blocks",i))(A||{}),L=(i=>(i.Desktop="desktop",i.Mobile="mobile",i))(L||{});const C=async i=>{try{const e=await fetch(i,{method:"HEAD"});if(!e.ok)return null;const t=e.headers.get("Content-Length");return t?parseInt(t,10):null}catch{return null}},O=i=>new Promise((e,t)=>{const o=new Image;o.onload=async()=>{const n=await C(i)??0;e({originalWidth:o.width,originalHeight:o.height,ratio:o.width/o.height,fileSize:n})},o.onerror=n=>{t(n)},o.src=i}),I=(i,e)=>{if(i.length!==1)return i;if(i.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 i},M=i=>{let e=`${I(i.key,i.code)}`.toLowerCase()||i.key;const t=navigator.userAgent.includes("Mac OS");e==="backspace"&&t&&(e="delete");const o=[];return(t&&i.metaKey||!t&&i.ctrlKey)&&o.push("mod"),i.shiftKey&&o.push("shift"),o.length>0&&(e=`${o.join("-")}-${I(e,i.code)}`),e},z=new Set(["escape","mod-c","mod-d","mod-z","mod-y","mod-shift-z","mod-v","delete","mod-g","mod-p","mod-s"]),T="1.15.0",U={version:T},_=()=>{var r,s;const i=new RegExp("(^| )seplugin-dev=([^;]+)"),e=document.cookie.match(i),t=(s=e==null?void 0:(r=e[2]).toLowerCase)==null?void 0:s.call(r);if(t==="true")return d;if(t!=null&&t.startsWith("https://"))return t;const[o,n]=U.version.split(".");return`${f}/v${o}.${n}`},K=({token:i,pluginId:e,apiUrl:t})=>{if(!i)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")},S=i=>{const e=document.createElement("iframe"),t=new URL(_());return t.searchParams.set("parentOrigin",i),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},m=class m{constructor(e){a(this,"iframe",null);a(this,"hotkeysAttached",!1);a(this,"iframeOrigin",null);a(this,"onHostHotkey",e=>{var c;if(!this.captureHotkeysEnabled()||!((c=this.iframe)!=null&&c.contentWindow))return;const t=M(e);if(!z.has(t))return;const o=e.target;if(o&&(o.isContentEditable||["INPUT","TEXTAREA","SELECT"].includes(o.tagName)))return;e.preventDefault();const{key:n,ctrlKey:r,metaKey:s,shiftKey:h,altKey:g}=e;this.postEvent("hostHotkey",{key:n,code:e.code,metaKey:s,ctrlKey:r,shiftKey:h,altKey:g,keyCode:t})});a(this,"onImageUpload",async({id:e,...t})=>{var o,n;try{const r=await((n=(o=this.config.on).uploadImage)==null?void 0:n.call(o,t));let s={};r&&(s=await O(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 o,n;const t=async r=>{const s={url:r};if(r){const h=await O(r);Object.assign(s,h)}this.postEvent("imageUploaded",{id:e,img:s})};(n=(o=this.config.on).openGallery)==null||n.call(o,t)});a(this,"onMessage",e=>{var r;if(e.source!==((r=this.iframe)==null?void 0:r.contentWindow))return;const{data:{source:t,event:o,...n}}=e;t==="DraftPlugin"&&(o==="openGallery"?this.onGalleryOpen(n):o==="uploadImage"?this.onImageUpload(n):o==="loadPersonalization"?this.onLoadPersonalization():this.triggerEvent(this.config.on[o],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,o=null;const n=()=>{t&&(clearTimeout(t),t=null),o&&(window.removeEventListener("message",o,!0),o=null)},r=new Promise((s,h)=>{const g=()=>{t&&clearTimeout(t),t=setTimeout(()=>{n(),h(new w("timeout_error",`Action "${e}" timed out`))},m.HEARTBEAT_TIMEOUT)};o=c=>{var p;if(c.source!==((p=this.iframe)==null?void 0:p.contentWindow))return;const{data:u}=c;if(u.source==="DraftPlugin"){if(u.event==="actionPing"&&u.action===e){c.stopPropagation(),g();return}if(u.event===e){c.stopPropagation(),n();const{source:D,event:H,ok:y,code:v,message:k,...P}=u;y===!1?h(new w(v||"unknown_error",k||"Unknown error")):s(P)}}},window.addEventListener("message",o,!0),g()});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:o}){const n=document.querySelector(this.config.container);if(n){const r=s=>{var c;const{data:{source:h,event:g}}=s;if(h==="DraftPlugin"&&g==="loaded"){if(s.source!==((c=this.iframe)==null?void 0:c.contentWindow))return;this.iframeOrigin=s.origin;const{on:u,container:p,locale:D,autosave:H,token:y,apiUrl:v,pluginId:k,authData:P,__config:W}=this.config;K(this.config),this.iframe&&this.iframe.contentWindow&&this.postEvent("init",{container:p,locale:D,uid:o,autosave:H,token:y,pluginId:k,authData:P,apiUrl:v,__config:W,enabledListeners:Object.keys(u),template:e,templateName:t}),window.addEventListener("message",this.onMessage),window.removeEventListener("message",r)}};this.iframe=S(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(m,"HEARTBEAT_TIMEOUT",4e3);let E=m;const x={container:"#draft-plugin-container",disableHotkeysPassing:!1,locale:"ru",on:{}};class N{constructor(){a(this,"version",T)}create(e){return new E({...x,...e})}}const b=new N;return window.DraftPlugin=b,l.DraftError=w,l.EditorMenu=A,l.ViewMode=L,l.default=b,Object.defineProperties(l,{__esModule:{value:!0},[Symbol.toStringTag]:{value:"Module"}}),l})({});
1
+ var G=Object.defineProperty;var F=(l,d,f)=>d in l?G(l,d,{enumerable:!0,configurable:!0,writable:!0,value:f}):l[d]=f;var a=(l,d,f)=>F(l,typeof d!="symbol"?d+"":d,f);var DraftPlugin=(function(l){"use strict";const d="http://localhost:5174",f="https://seplugin.sendsay.ru";class w extends Error{constructor(t,o){super(o);a(this,"code");this.name="DraftError",this.code=t}}var A=(i=>(i.Audit="audit",i.Guide="guide",i.Style="style",i.Blocks="blocks",i))(A||{}),L=(i=>(i.Desktop="desktop",i.Mobile="mobile",i))(L||{});const C=async i=>{try{const e=await fetch(i,{method:"HEAD"});if(!e.ok)return null;const t=e.headers.get("Content-Length");return t?parseInt(t,10):null}catch{return null}},O=i=>new Promise((e,t)=>{const o=new Image;o.onload=async()=>{const n=await C(i)??0;e({originalWidth:o.width,originalHeight:o.height,ratio:o.width/o.height,fileSize:n})},o.onerror=n=>{t(n)},o.src=i}),I=(i,e)=>{if(i.length!==1)return i;if(i.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 i},M=i=>{let e=`${I(i.key,i.code)}`.toLowerCase()||i.key;const t=navigator.userAgent.includes("Mac OS");e==="backspace"&&t&&(e="delete");const o=[];return(t&&i.metaKey||!t&&i.ctrlKey)&&o.push("mod"),i.shiftKey&&o.push("shift"),o.length>0&&(e=`${o.join("-")}-${I(e,i.code)}`),e},z=new Set(["escape","mod-c","mod-d","mod-z","mod-y","mod-shift-z","mod-v","delete","mod-g","mod-p","mod-s"]),T="1.15.1",U={version:T},_=()=>{var r,s;const i=new RegExp("(^| )seplugin-dev=([^;]+)"),e=document.cookie.match(i),t=(s=e==null?void 0:(r=e[2]).toLowerCase)==null?void 0:s.call(r);if(t==="true")return d;if(t!=null&&t.startsWith("https://"))return t;const[o,n]=U.version.split(".");return`${f}/v${o}.${n}`},K=({token:i,pluginId:e,apiUrl:t})=>{if(!i)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")},S=i=>{const e=document.createElement("iframe"),t=new URL(_());return t.searchParams.set("parentOrigin",i),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},m=class m{constructor(e){a(this,"iframe",null);a(this,"hotkeysAttached",!1);a(this,"iframeOrigin",null);a(this,"onHostHotkey",e=>{var c;if(!this.captureHotkeysEnabled()||!((c=this.iframe)!=null&&c.contentWindow))return;const t=M(e);if(!z.has(t))return;const o=e.target;if(o&&(o.isContentEditable||["INPUT","TEXTAREA","SELECT"].includes(o.tagName)))return;e.preventDefault();const{key:n,ctrlKey:r,metaKey:s,shiftKey:h,altKey:g}=e;this.postEvent("hostHotkey",{key:n,code:e.code,metaKey:s,ctrlKey:r,shiftKey:h,altKey:g,keyCode:t})});a(this,"onImageUpload",async({id:e,...t})=>{var o,n;try{const r=await((n=(o=this.config.on).uploadImage)==null?void 0:n.call(o,t));let s={};r&&(s=await O(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 o,n;const t=async r=>{const s={url:r};if(r){const h=await O(r);Object.assign(s,h)}this.postEvent("imageUploaded",{id:e,img:s})};(n=(o=this.config.on).openGallery)==null||n.call(o,t)});a(this,"onMessage",e=>{var r;if(e.source!==((r=this.iframe)==null?void 0:r.contentWindow))return;const{data:{source:t,event:o,...n}}=e;t==="DraftPlugin"&&(o==="openGallery"?this.onGalleryOpen(n):o==="uploadImage"?this.onImageUpload(n):o==="loadPersonalization"?this.onLoadPersonalization():this.triggerEvent(this.config.on[o],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,o=null;const n=()=>{t&&(clearTimeout(t),t=null),o&&(window.removeEventListener("message",o,!0),o=null)},r=new Promise((s,h)=>{const g=()=>{t&&clearTimeout(t),t=setTimeout(()=>{n(),h(new w("timeout_error",`Action "${e}" timed out`))},m.HEARTBEAT_TIMEOUT)};o=c=>{var p;if(c.source!==((p=this.iframe)==null?void 0:p.contentWindow))return;const{data:u}=c;if(u.source==="DraftPlugin"){if(u.event==="actionPing"&&u.action===e){c.stopPropagation(),g();return}if(u.event===e){c.stopPropagation(),n();const{source:D,event:H,ok:y,code:v,message:k,...P}=u;y===!1?h(new w(v||"unknown_error",k||"Unknown error")):s(P)}}},window.addEventListener("message",o,!0),g()});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:o}){const n=document.querySelector(this.config.container);if(n){const r=s=>{var c;const{data:{source:h,event:g}}=s;if(h==="DraftPlugin"&&g==="loaded"){if(s.source!==((c=this.iframe)==null?void 0:c.contentWindow))return;this.iframeOrigin=s.origin;const{on:u,container:p,locale:D,autosave:H,token:y,apiUrl:v,pluginId:k,authData:P,__config:W}=this.config;K(this.config),this.iframe&&this.iframe.contentWindow&&this.postEvent("init",{container:p,locale:D,uid:o,autosave:H,token:y,pluginId:k,authData:P,apiUrl:v,__config:W,enabledListeners:Object.keys(u),template:e,templateName:t}),window.addEventListener("message",this.onMessage),window.removeEventListener("message",r)}};this.iframe=S(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(m,"HEARTBEAT_TIMEOUT",4e3);let E=m;const x={container:"#draft-plugin-container",disableHotkeysPassing:!1,locale:"ru",on:{}};class N{constructor(){a(this,"version",T)}create(e){return new E({...x,...e})}}const b=new N;return window.DraftPlugin=b,l.DraftError=w,l.EditorMenu=A,l.ViewMode=L,l.default=b,Object.defineProperties(l,{__esModule:{value:!0},[Symbol.toStringTag]:{value:"Module"}}),l})({});
2
2
  //# sourceMappingURL=index.iife.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.iife.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 * 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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"}
1
+ {"version":3,"file":"index.iife.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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"}
@@ -11,5 +11,6 @@ declare global {
11
11
  }
12
12
  }
13
13
  export default DraftPlugin;
14
+ export type { PluginInstance } from './plugin';
14
15
  export * from './types';
15
16
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAC1C,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAUvC,cAAM,aAAa;IACV,OAAO,SAAW;IAEzB,MAAM,CAAC,MAAM,EAAE,YAAY;CAM5B;AAED,QAAA,MAAM,WAAW,eAAsB,CAAC;AAExC,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,MAAM;QACd,WAAW,EAAE,aAAa,CAAC;KAC5B;CACF;AAID,eAAe,WAAW,CAAC;AAC3B,cAAc,SAAS,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAC1C,OAAO,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AAUvC,cAAM,aAAa;IACV,OAAO,SAAW;IAEzB,MAAM,CAAC,MAAM,EAAE,YAAY;CAM5B;AAED,QAAA,MAAM,WAAW,eAAsB,CAAC;AAExC,OAAO,CAAC,MAAM,CAAC;IACb,UAAU,MAAM;QACd,WAAW,EAAE,aAAa,CAAC;KAC5B;CACF;AAID,eAAe,WAAW,CAAC;AAC3B,YAAY,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAC/C,cAAc,SAAS,CAAC"}
package/package.json CHANGED
@@ -1,19 +1,22 @@
1
1
  {
2
2
  "name": "@getdraft/plugin",
3
- "version": "1.15.0",
3
+ "version": "1.15.1",
4
4
  "main": "dist/index.cjs.js",
5
5
  "module": "dist/index.es.js",
6
- "browser": "dist/index.iife.js",
7
- "types": "src",
6
+ "types": "dist/types/index.d.ts",
7
+ "exports": {
8
+ ".": {
9
+ "import": "./dist/index.es.js",
10
+ "require": "./dist/index.cjs.js",
11
+ "types": "./dist/types/index.d.ts"
12
+ }
13
+ },
8
14
  "dependencies": {},
9
15
  "devDependencies": {
10
- "@vitejs/plugin-legacy": "5.4.1",
11
- "rollup": "4.22.4",
12
- "terser": "5.31.3",
13
- "rollup-plugin-dts": "6.1.1",
14
- "@getdraft/prettier": "^1.0.0",
16
+ "vite-plugin-dts": "4.0.0",
15
17
  "@types/getdraft": "^1.0.0",
16
- "@getdraft/eslint-config-ts": "^1.0.0"
18
+ "@getdraft/eslint-config-ts": "^1.0.0",
19
+ "@getdraft/prettier": "^1.0.0"
17
20
  },
18
21
  "license": "MIT",
19
22
  "scripts": {
package/src/index.ts CHANGED
@@ -31,4 +31,5 @@ declare global {
31
31
  window.DraftPlugin = DraftPlugin;
32
32
 
33
33
  export default DraftPlugin;
34
+ export type { PluginInstance } from './plugin';
34
35
  export * from './types';
package/vite.config.ts CHANGED
@@ -4,7 +4,13 @@ import dts from 'vite-plugin-dts';
4
4
 
5
5
  export default defineConfig({
6
6
  base: './',
7
- plugins: [dts({ insertTypesEntry: true })],
7
+ plugins: [
8
+ dts({
9
+ insertTypesEntry: true,
10
+ outDir: 'dist/types',
11
+ entryRoot: 'src',
12
+ }),
13
+ ],
8
14
  build: {
9
15
  sourcemap: true,
10
16
  lib: {
@@ -13,5 +19,10 @@ export default defineConfig({
13
19
  formats: ['es', 'cjs', 'iife'],
14
20
  fileName: (format) => `index.${format}.js`,
15
21
  },
22
+ rollupOptions: {
23
+ output: {
24
+ exports: 'named',
25
+ },
26
+ },
16
27
  },
17
28
  });
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes
File without changes