diff --git a/bower.json b/bower.json index 722018bad..83a3da704 100644 --- a/bower.json +++ b/bower.json @@ -1,6 +1,6 @@ { "name": "suneditor", - "version": "2.9.1", + "version": "2.9.2", "description": "Pure JavaScript based WYSIWYG web editor", "main": "src/suneditor.js", "keywords": [ diff --git a/dist/suneditor.min.js b/dist/suneditor.min.js index 8391a6d5b..0f19f958c 100644 --- a/dist/suneditor.min.js +++ b/dist/suneditor.min.js @@ -1 +1 @@ -!function(e){var t={};function i(n){if(t[n])return t[n].exports;var o=t[n]={i:n,l:!1,exports:{}};return e[n].call(o.exports,o,o.exports,i),o.l=!0,o.exports}i.m=e,i.c=t,i.d=function(e,t,n){i.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:n})},i.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},i.t=function(e,t){if(1&t&&(e=i(e)),8&t)return e;if(4&t&&"object"==typeof e&&e&&e.__esModule)return e;var n=Object.create(null);if(i.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:e}),2&t&&"string"!=typeof e)for(var o in e)i.d(n,o,function(t){return e[t]}.bind(null,o));return n},i.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return i.d(t,"a",t),t},i.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},i.p="",i(i.s="XJR1")}({"0A7J":function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d"),_modules_colorPicker__WEBPACK_IMPORTED_MODULE_1__=__webpack_require__("EjF6");__webpack_exports__.a={name:"fontColor",add:function(core,targetElement){core.addModule([_modules_colorPicker__WEBPACK_IMPORTED_MODULE_1__.a]);const context=core.context;context.fontColor={previewEl:null,colorInput:null};let listDiv=eval(this.setSubmenu(context.colorPicker.colorListHTML));context.fontColor.colorInput=listDiv.getElementsByClassName("sun-editor-id-submenu-color-input")[0],context.fontColor.colorInput.addEventListener("keyup",this.onChangeInput.bind(core)),listDiv.getElementsByClassName("sun-editor-id-submenu-color-submit")[0].addEventListener("click",this.submit.bind(core)),listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickup.bind(core)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");return t.className="layer_editor",t.style.display="none",t.innerHTML=e,t},on:function(){const e=this.context.colorPicker;e._colorInput=this.context.fontColor.colorInput,e._defaultColor="#000000",e._styleProperty="color",this.plugins.colorPicker.init.call(this,this.getSelectionNode(),null)},onChangeInput:function(e){this.plugins.colorPicker.setCurrentColor.call(this,"#"+e.target.value)},submit:function(){this.plugins.fontColor.applyColor.call(this,this.context.colorPicker._currentColor)},pickup:function(e){if(e.preventDefault(),e.stopPropagation(),!/^BUTTON$/i.test(e.target.tagName))return!1;this.plugins.fontColor.applyColor.call(this,e.target.getAttribute("data-value"))},applyColor:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("SPAN");t.style.color=e,this.nodeChange(t,["color"]),this.submenuOff(),this.focus()}}},"1kvd":function(e,t,i){"use strict";var n=i("8e1d");t.a={name:"dialog",add:function(e){const t=e.context;t.dialog={};let i=n.a.createElement("DIV");i.className="sun-editor-id-dialogBox sun-editor-common";let o=n.a.createElement("DIV");o.className="modal-dialog-background sun-editor-id-dialog-back",o.style.display="none";let l=n.a.createElement("DIV");l.className="modal-dialog sun-editor-id-dialog-modal",l.style.display="none",i.appendChild(o),i.appendChild(l),t.dialog.modalArea=i,t.dialog.back=o,t.dialog.modal=l,t.dialog.modal.addEventListener("click",this.onClick_dialog.bind(e)),t.element.relative.appendChild(i),i=null,o=null,l=null},onClick_dialog:function(e){e.stopPropagation(),(/modal-dialog/.test(e.target.className)||/close/.test(e.target.getAttribute("data-command")))&&this.plugins.dialog.close.call(this)},open:function(e,t){if(this.modalForm)return!1;this.context.dialog.updateModal=t,"full"===this.context.option.popupDisplay?this.context.dialog.modalArea.style.position="fixed":this.context.dialog.modalArea.style.position="absolute",this.context.dialog.kind=e,this.modalForm=this.context[e].modal;const i=this.context[e].focusElement;this.context.dialog.modalArea.style.display="block",this.context.dialog.back.style.display="block",this.context.dialog.modal.style.display="block",this.modalForm.style.display="block",i&&i.focus()},close:function(){this.modalForm.style.display="none",this.context.dialog.back.style.display="none",this.context.dialog.modalArea.style.display="none",this.modalForm=null,this.context.dialog.updateModal=!1,this.plugins[this.context.dialog.kind].init.call(this)}}},"3FqI":function(e,t,i){},"50IV":function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d");__webpack_exports__.a={name:"font",add:function(core,targetElement){let listDiv=eval(this.setSubmenu(core.context.option));listDiv.getElementsByClassName("list_family")[0].addEventListener("click",this.pickup.bind(core)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");let i,n,o,l;t.className="layer_editor",t.style.display="none";let a=e.font?e.font:["Arial","Comic Sans MS","Courier New,Courier","Impact,Charcoal,sans-serif","Georgia","tahoma","Trebuchet MS,Helvetica","Verdana"],s='
",s+="
",t.innerHTML=s,t},pickup:function(e){if(!/^BUTTON$/i.test(e.target.tagName))return!1;e.preventDefault(),e.stopPropagation();const t=e.target;_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.changeTxt(this.context.tool.font,t.getAttribute("data-txt"));const i=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("SPAN");i.style.fontFamily=t.getAttribute("data-value"),this.nodeChange(i,["font-family"]),this.submenuOff(),this.focus()}}},"8e1d":function(e,t,i){"use strict";const n={_d:document,_w:window,zeroWidthSpace:"​",getXMLHttpRequest:function(){if(!this._w.ActiveXObject)return this._w.XMLHttpRequest?new XMLHttpRequest:null;try{return new ActiveXObject("Msxml2.XMLHTTP")}catch(e){try{return new ActiveXObject("Microsoft.XMLHTTP")}catch(e){return null}}},copyObj:function(e){const t={};for(let i in e)t[i]=e[i];return t},createElement:function(e){return this._d.createElement(e)},createTextNode:function(e){return this._d.createTextNode(e||"")},getIncludePath:function(e,t){let i="";const n=[],o="js"===t?"script":"link",l="js"===t?"src":"href";let a="(?:";for(let t=0,i=e.length;t0?n[0][l]:""),-1===i.indexOf(":/")&&"//"!==i.slice(0,2)&&(i=0===i.indexOf("/")?location.href.match(/^.*?:\/\/[^\/]*/)[0]+i:location.href.match(/^[^\?]*\/(?:)/)[0]+i),!i)throw"[SUNEDITOR.util.getIncludePath.fail] The SUNEDITOR installation path could not be automatically detected. (name: +"+name+", extension: "+t+")";return i},convertContentsForEditor:function(e){let t,i,n="";e=e.trim();for(let o=0,l=(t=this._d.createRange().createContextualFragment(e).childNodes).length;o0&&(n+="

"+t+"

")}else n+=i;const o={"&":"&"," ":" ","'":""","<":"<",">":">"};return e=e.replace(/&|\u00A0|'|<|>/g,function(e){return"string"==typeof o[e]?o[e]:e}),0===n.length&&(n="

"+(e.length>0?e:this.zeroWidthSpace)+"

"),n},convertHTMLForCodeView:function(e){return e.replace(/\s*<(?:li|td)\s*(?:[a-z\-]+)?\s*(?:="?[^>]*"?)?\s*>/gi,this._insertIndent).replace(/<\/?(?:blockquote|hr|ol|ul|table|tbody|thead|th|tr)\s*(?:[a-z\-]+)?\s*(?:="?[^>]*"?)?\s*>(?=[^\n])/gi,this._insertLineBreak).replace(/<\/(?:p|div|h[1-6]|li|td|pre)>(?=[^\n])/gi,this._insertLineBreak)},_insertIndent:function(e){return" "+e.trim()},_insertLineBreak:function(e){return e+"\n"},isWysiwygDiv:function(e){return!(!e||1!==e.nodeType||!this.hasClass(e,"sun-editor-id-wysiwyg"))},isFormatElement:function(e){return!(!e||1!==e.nodeType||!/^(?:P|DIV|H[1-6]|LI|CODE)$/i.test(e.nodeName)||/sun-editor-id-comp/.test(e.className))},isRangeFormatElement:function(e){return!(!e||1!==e.nodeType||!/^(?:BLOCKQUOTE|TABLE|THEAD|TBODY|TH|TR|TD|OL|UL|PRE|FIGCAPTION)$/i.test(e.nodeName))},getFormatElement:function(e){if(!e)return null;for(;!this.isFormatElement(e)&&!this.isWysiwygDiv(e.parentNode);)e=e.parentNode;if(this.isWysiwygDiv(e)||this.isRangeFormatElement(e)){return this.getListChildren(e,function(e){return this.isFormatElement(e)}.bind(this))[0]}return e},getRangeFormatElement:function(e){if(!e)return null;for(;!this.isRangeFormatElement(e)&&!this.isWysiwygDiv(e);)e=e.parentNode;return this.isWysiwygDiv(e)?null:e},getArrayIndex:function(e,t){let i=-1;for(let n=0,o=e.length;n]*"?)?\s*/gi,"").replace(/<\/?(?:span|font)\s*(?:[a-z\-]+)?\s*(?:="?[^>]*"?)?\s*>/gi,"").replace(/<\/?[a-z]+:[a-z]+\s*(?:[a-z\-]+)?\s*(?:="?[^>]*"?)?\s*>/gi,""):n+=i[e].textContent;return n||e}};t.a=n},EjF6:function(e,t,i){"use strict";var n=i("8e1d");t.a={name:"colorPicker",add:function(e){const t=e.context;t.colorPicker={colorListHTML:"",_colorInput:"",_defaultColor:"#000",_styleProperty:"color",_currentColor:""};let i=this.createColorList(e.context.option,e.lang);t.colorPicker.colorListHTML=i,i=null},createColorList:function(e,t){const i=e.colorList?e.colorList:["#ff0000","#ff5e00","#ffe400","#abf200","#00d8ff","#0055ff","#6600ff","#ff00dd","#000000","#ffd8d8","#fae0d4","#faf4c0","#e4f7ba","#d4f4fa","#d9e5ff","#e8d9ff","#ffd9fa","#ffffff","#ffa7a7","#ffc19e","#faed7d","#cef279","#b2ebf4","#b2ccff","#d1b2ff","#ffb2f5","#bdbdbd","#f15f5f","#f29661","#e5d85c","#bce55c","#5cd1e5","#6699ff","#a366ff","#f261df","#8c8c8c","#980000","#993800","#998a00","#6b9900","#008299","#003399","#3d0099","#990085","#353535","#670000","#662500","#665c00","#476600","#005766","#002266","#290066","#660058","#222222"];let n='
    ';for(let e=0,t=i.length;e '}return n+='
"},init:function(e,t){const i=this.plugins.colorPicker;let n=t||(i.getColorInNode.call(this,e)||this.context.colorPicker._defaultColor);n=i.isHexColor(n)?n:i.rgb2hex(n),i.setInputText.call(this,n)},setCurrentColor:function(e){this.context.colorPicker._currentColor=e,this.context.colorPicker._colorInput.style.borderColor=e},setInputText:function(e){this.context.colorPicker._colorInput.value=e.replace("#",""),this.plugins.colorPicker.setCurrentColor.call(this,e)},getColorInNode:function(e){let t="";const i=this.context.colorPicker._styleProperty;for(;!n.a.isWysiwygDiv(e)&&0===t.length;)1===e.nodeType&&e.style[i]&&(t=e.style[i]),e=e.parentNode;return t},isHexColor:function(e){return/^#[0-9a-f]{3}(?:[0-9a-f]{3})?$/i.test(e)},rgb2hex:function(e){return(e=e.match(/^rgba?[\s+]?\([\s+]?(\d+)[\s+]?,[\s+]?(\d+)[\s+]?,[\s+]?(\d+)[\s+]?/i))&&4===e.length?"#"+("0"+parseInt(e[1],10).toString(16)).slice(-2)+("0"+parseInt(e[2],10).toString(16)).slice(-2)+("0"+parseInt(e[3],10).toString(16)).slice(-2):""}}},KKur:function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d"),_modules_dialog__WEBPACK_IMPORTED_MODULE_1__=__webpack_require__("1kvd"),_modules_resizing__WEBPACK_IMPORTED_MODULE_2__=__webpack_require__("ee5k"),_modules_notice__WEBPACK_IMPORTED_MODULE_3__=__webpack_require__("PAX9");__webpack_exports__.a={name:"image",add:function(core){core.addModule([_modules_dialog__WEBPACK_IMPORTED_MODULE_1__.a,_modules_resizing__WEBPACK_IMPORTED_MODULE_2__.a,_modules_notice__WEBPACK_IMPORTED_MODULE_3__.a]);const context=core.context;context.image={_linkElement:null,_container:null,_cover:null,_element:null,_element_w:1,_element_h:1,_element_l:0,_element_t:0,_origin_w:"auto"===context.option.imageWidth?"":context.option.imageWidth,_origin_h:"",_altText:"",_caption:null,captionCheckEl:null,_linkValue:"",_align:"none",_captionChecked:!1,_proportionChecked:!0,_floatClassRegExp:"float\\-[a-z]+",_xmlHttp:null,_resizing:context.option.imageResizing,_defaultAuto:"auto"===context.option.imageWidth};let image_dialog=eval(this.setDialog(core.context.option,core.lang));context.image.modal=image_dialog,context.image.imgUrlFile=image_dialog.getElementsByClassName("sun-editor-id-image-url")[0],context.image.imgInputFile=context.image.focusElement=image_dialog.getElementsByClassName("sun-editor-id-image-file")[0],context.image.altText=image_dialog.getElementsByClassName("sun-editor-id-image-alt")[0],context.image.imgLink=image_dialog.getElementsByClassName("sun-editor-id-image-link")[0],context.image.imgLinkNewWindowCheck=image_dialog.getElementsByClassName("sun-editor-id-linkCheck")[0],context.image.captionCheckEl=image_dialog.getElementsByClassName("suneditor-id-image-check-caption")[0],context.image.modal.getElementsByClassName("sun-editor-tab-button")[0].addEventListener("click",this.openTab.bind(core)),context.image.modal.getElementsByClassName("btn-primary")[0].addEventListener("click",this.submit.bind(core)),context.image.imageX={},context.image.imageY={},context.option.imageResizing&&(context.image.proportion=image_dialog.getElementsByClassName("suneditor-id-image-check-proportion")[0],context.image.imageX=image_dialog.getElementsByClassName("sun-editor-id-image-x")[0],context.image.imageY=image_dialog.getElementsByClassName("sun-editor-id-image-y")[0],context.image.imageX.value=context.option.imageWidth,context.image.imageX.addEventListener("change",this.setInputSize.bind(core,"x")),context.image.imageY.addEventListener("change",this.setInputSize.bind(core,"y")),image_dialog.getElementsByClassName("sun-editor-id-image-revert-button")[0].addEventListener("click",this.sizeRevert.bind(core))),context.dialog.modal.appendChild(image_dialog),image_dialog=null},setDialog:function(e,t){const i=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");i.className="modal-content sun-editor-id-dialog-image",i.style.display="none";let n='
",i.innerHTML=n,i},openTab:function(e){const t=this.context.image.modal,i="init"===e?t.getElementsByClassName("sun-editor-id-tab-link")[0]:e.target;if(!/^BUTTON$/i.test(i.tagName))return!1;const n=i.getAttribute("data-tab-link");let o,l,a;for(l=t.getElementsByClassName("sun-editor-id-tab-content"),o=0;o0){const t=this.context.option.imageUploadUrl,i=this.context.dialog.updateModal?1:e.length;if(null!==t&&t.length>0){const n=new FormData;for(let t=0;t0?1:s.length;a0){const n=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("A");return n.href=/^https?:\/\//.test(t)?t:"http://"+t,n.target=i?"_blank":"",n.setAttribute("data-image-link","image"),e.setAttribute("data-image-link",t),n.appendChild(e),n}return e},setInputSize:function(e){this.context.dialog.updateModal&&this.context.image.proportion.checked&&("x"===e?this.context.image.imageY.value=Math.round(this.context.image._element_h/this.context.image._element_w*this.context.image.imageX.value):this.context.image.imageX.value=Math.round(this.context.image._element_w/this.context.image._element_h*this.context.image.imageY.value))},submit:function(e){this.showLoading(),e.preventDefault(),e.stopPropagation(),this.context.image._linkValue=this.context.image.imgLink.value,this.context.image._altText=this.context.image.altText.value,this.context.image._align=this.context.image.modal.querySelector('input[name="suneditor_image_radio"]:checked').value,this.context.image._captionChecked=this.context.image.captionCheckEl.checked,this.context.image._resizing&&(this.context.image._proportionChecked=this.context.image.proportion.checked);try{this.context.dialog.updateModal&&this.plugins.image.update_image.call(this),this.context.image.imgInputFile&&this.context.image.imgInputFile.files.length>0?this.plugins.image.onRender_imgInput.call(this):this.context.image.imgUrlFile&&this.context.image.imgUrlFile.value.trim().length>0?this.plugins.image.onRender_imgUrl.call(this):this.closeLoading()}catch(e){throw this.closeLoading(),_modules_notice__WEBPACK_IMPORTED_MODULE_3__.a.open.call(this,'[SUNEDITOR.image.submit.fail] cause : "'+e.message+'"'),Error('[SUNEDITOR.image.submit.fail] cause : "'+e.message+'"')}finally{this.plugins.dialog.close.call(this)}return!1},_onload_image:function(e,t){if(/\d+/.test(e.style.width)&&(e.style.height=e.offsetHeight+"px"),e.setAttribute("origin-size",e.naturalWidth+","+e.naturalHeight),e.setAttribute("data-origin",e.offsetWidth+","+e.offsetHeight),!t)return;let i=e.getAttribute("data-index");if(i){const t=this._variable._imagesInfo[i];t.name=e.getAttribute("data-file-name"),t.size=1*e.getAttribute("data-file-size")}else i=this._variable._imageIndex,e.setAttribute("data-index",i),this._variable._imagesInfo[i]={index:i,name:t.name,size:t.size,select:function(){const t=this.plugins.resizing.call_controller_resize.call(this,e,"image");this.plugins.image.onModifyMode.call(this,e,t),e.scrollIntoView()}.bind(this),delete:this.plugins.image.destroy.bind(this,e)},this._variable._imageIndex++,e.setAttribute("data-file-name",t.name),e.setAttribute("data-file-size",t.size);this._imageUpload(e,i,!1)},create_image:function(e,t,i,n,o,l,a,s){if(l)return void(a.src=e);const r=this.context.image;let d=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("IMG");d.addEventListener("load",this.plugins.image._onload_image.bind(this,d,s)),d.src=e,d.setAttribute("data-align",o),d.alt=r._altText,(d=this.plugins.image.onRender_link(d,t,i)).setAttribute("data-rotate","0"),r._resizing&&(/\d+/.test(n)&&(d.style.width=n),d.setAttribute("data-proportion",r._proportionChecked));const c=this.plugins.resizing.set_cover.call(this,d),_=this.plugins.resizing.set_container.call(this,c,"sun-editor-id-image-container");r._captionChecked&&(r._caption=this.plugins.resizing.create_caption.call(this),r._caption.setAttribute("contenteditable",!1),c.appendChild(r._caption)),c.style.margin="none"!==o?"auto":"0",_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(_,r._floatClassRegExp),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.addClass(_,"float-"+o),r._resizing&&/\d+/.test(n)||(this.context.resizing._resize_plugin="image",r._element=d,r._cover=c,r._container=_,this.plugins.image.setAutoSize.call(this)),this.insertNode(_,_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getFormatElement(this.getSelectionNode())),this.appendFormatTag(_)},update_image:function(){const e=this.context.image,t=e._linkValue;let i=e._element,n=e._cover,o=e._container,l=!1;const a=1*e.imageX.value!==i.offsetWidth||1*e.imageY.value!==i.offsetHeight;if(null===n&&(l=!0,i=e._element.cloneNode(!0),n=this.plugins.resizing.set_cover.call(this,i)),null===o&&(n=n.cloneNode(!0),l=!0,o=this.plugins.resizing.set_container.call(this,n,"sun-editor-id-image-container")),l&&(o.innerHTML="",o.appendChild(n)),i.alt=e._altText,e._resizing&&(i.setAttribute("data-proportion",e._proportionChecked),a&&this.plugins.image.setSize.call(this,e.imageX.value,e.imageY.value)),e._captionChecked?e._caption||(e._caption=this.plugins.resizing.create_caption.call(this),n.appendChild(e._caption)):e._caption&&(_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeItem(e._caption),e._caption=null),e._align&&"none"!==e._align?n.style.margin="auto":n.style.margin="0",_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(o,this.context.image._floatClassRegExp),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.addClass(o,"float-"+e._align),i.setAttribute("data-align",e._align),t.trim().length>0)if(null!==e._linkElement)e._linkElement.href=t,e._linkElement.target=e.imgLinkNewWindowCheck.checked?"_blank":"",i.setAttribute("data-image-link",t);else{let o=this.plugins.image.onRender_link(i,t,this.context.image.imgLinkNewWindowCheck.checked);n.insertBefore(o,e._caption)}else if(null!==e._linkElement){const t=i;t.setAttribute("data-image-link","");let o=t.cloneNode(!0);n.removeChild(e._linkElement),n.insertBefore(o,e._caption),i=o}if(l){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getFormatElement(e._element);t.parentNode.insertBefore(o,t),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeItem(t)}(e._resizing&&a||this.context.resizing._rotateVertical&&e._captionChecked)&&this.plugins.resizing.setTransformSize.call(this,i)},sizeRevert:function(){const e=this.context.image;e._origin_w&&(e.imageX.value=e._element_w=e._origin_w,e.imageY.value=e._element_h=e._origin_h)},onModifyMode:function(e,t){const i=this.context.image;i._linkElement=/^A$/i.test(e.parentNode.nodeName)?e.parentNode:null,i._element=e,i._cover=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getParentElement(e,".sun-editor-figure-cover"),i._container=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getParentElement(e,".sun-editor-id-image-container"),i._caption=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getChildElement(i._cover,"FIGCAPTION"),i._align=e.getAttribute("data-align")||"none",i._element_w=t.w,i._element_h=t.h,i._element_t=t.t,i._element_l=t.l;let n=i._element.getAttribute("data-origin");n?(n=n.split(","),i._origin_w=1*n[0],i._origin_h=1*n[1]):(i._origin_w=t.w,i._origin_h=t.h,i._element.setAttribute("data-origin",t.w+","+t.h))},openModify:function(e){const t=this.context.image;t.imgUrlFile.value=t._element.src,t.altText.value=t._element.alt,t.imgLink.value=null===t._linkElement?"":t._linkElement.href,t.imgLinkNewWindowCheck.checked=t._linkElement&&"_blank"===t._linkElement.target,t.modal.querySelector('input[name="suneditor_image_radio"][value="'+t._align+'"]').checked=!0,t._captionChecked=t.captionCheckEl.checked=!!t._caption,t._resizing&&(t.proportion.checked=t._proportionChecked="true"===t._element.getAttribute("data-proportion"),t.imageX.value=t._element.offsetWidth,t.imageY.value=t._element.offsetHeight,t.imageX.disabled=!1,t.imageY.disabled=!1,t.proportion.disabled=!1),e||this.plugins.dialog.open.call(this,"image",!0)},setSize:function(e,t){const i=this.context.image;i._element.style.width=e+"px",i._element.style.height=t+"px"},setAutoSize:function(){const e=this.context.image;this.plugins.resizing.resetTransform.call(this,e._element),this.plugins.image.cancelPercentAttr.call(this),e._element.style.maxWidth="100%",e._element.style.width="",e._element.style.height="",e._cover.style.width="",e._cover.style.height=""},setPercentSize:function(e){const t=this.context.image;t._container.style.width=e,t._container.style.height="",t._cover.style.width="100%",t._cover.style.height="",t._element.style.width="100%",t._element.style.height="",/100/.test(e)&&(_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(t._container,this.context.image._floatClassRegExp),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.addClass(t._container,"float-center"))},cancelPercentAttr:function(){const e=this.context.image;e._element.style.maxWidth="none",e._cover.style.width="",e._cover.style.height="",e._container.style.width="",e._container.style.height="",_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(e._container,this.context.image._floatClassRegExp),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.addClass(e._container,"float-"+e._align)},resetAlign:function(){const e=this.context.image;e._element.setAttribute("data-align",""),e._align="none",e._cover.style.margin="0",_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(e._container,e._floatClassRegExp)},destroy:function(e){const t=e||this.context.image._element,i=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getParentElement(t,".sun-editor-id-image-container")||t,n=t.getAttribute("data-index");_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeItem(i),this.plugins.image.init.call(this),this.controllersOff(),n&&(delete this._variable._imagesInfo[n],this._imageUpload(t,n,!0))},init:function(){const e=this.context.image;if(e.imgInputFile&&(e.imgInputFile.value=""),e.imgUrlFile&&(e.imgUrlFile.value=""),e.altText.value="",e.imgLink.value="",e.imgLinkNewWindowCheck.checked=!1,e.modal.querySelector('input[name="suneditor_image_radio"][value="none"]').checked=!0,e.captionCheckEl.checked=!1,e._element=null,this.plugins.image.openTab.call(this,"init"),e._resizing){const t="auto"===this.context.option.imageWidth;e.proportion.checked=!1,e.imageX.value=t?"":this.context.option.imageWidth,e.imageY.value="",e.imageX.disabled=t,e.imageY.disabled=!0,e.proportion.disabled=!0}}}},MIhV:function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d");__webpack_exports__.a={name:"fontSize",add:function(core,targetElement){let listDiv=eval(this.setSubmenu(core.context.option));listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickup.bind(core)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");t.className="layer_editor layer_size",t.style.display="none";const i=e.fontSize?e.fontSize:[8,9,10,11,12,14,16,18,20,22,24,26,28,36,48,72];let n='
    ';for(let e=0,t=i.length;e"}return n+="
",t.innerHTML=n,t},pickup:function(e){if(e.preventDefault(),e.stopPropagation(),!/^BUTTON$/i.test(e.target.tagName))return!1;_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.changeTxt(this.context.tool.fontSize,e.target.getAttribute("data-value"));const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("SPAN");t.style.fontSize=e.target.getAttribute("data-value")+"px",this.nodeChange(t,["font-size"]),this.submenuOff(),this.focus()}}},PAX9:function(e,t,i){"use strict";var n=i("8e1d");t.a={name:"notice",add:function(e){const t=e.context;t.notice={};let i=n.a.createElement("DIV"),o=n.a.createElement("SPAN"),l=n.a.createElement("BUTTON");i.className="sun-editor-id-notice",l.className="close",l.setAttribute("aria-label","Close"),l.setAttribute("title",e.lang.dialogBox.close),l.innerHTML='',i.appendChild(o),i.appendChild(l),t.notice.modal=i,t.notice.message=o,l.addEventListener("click",this.onClick_cancel.bind(e)),t.element.editorArea.insertBefore(i,t.element.wysiwyg),i=null},onClick_cancel:function(e){e.preventDefault(),e.stopPropagation(),this.plugins.notice.close.call(this)},open:function(e){this.context.notice.message.textContent=e,this.context.notice.modal.style.display="block"},close:function(){this.context.notice.modal.style.display="none"}}},Rp48:function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d"),_modules_dialog__WEBPACK_IMPORTED_MODULE_1__=__webpack_require__("1kvd");__webpack_exports__.a={name:"link",add:function(core){core.addModule([_modules_dialog__WEBPACK_IMPORTED_MODULE_1__.a]);const context=core.context;context.link={};let link_dialog=eval(this.setDialog(core.lang));context.link.modal=link_dialog,context.link.focusElement=link_dialog.getElementsByClassName("sun-editor-id-link-url")[0],context.link.linkAnchorText=link_dialog.getElementsByClassName("sun-editor-id-link-text")[0],context.link.linkNewWindowCheck=link_dialog.getElementsByClassName("sun-editor-id-link-check")[0];let link_button=eval(this.setController_LinkButton(core.lang));context.link.linkBtn=link_button,context.link._linkAnchor=null,link_dialog.getElementsByClassName("btn-primary")[0].addEventListener("click",this.submit.bind(core)),link_button.addEventListener("click",this.onClick_linkBtn.bind(core)),context.dialog.modal.appendChild(link_dialog),context.element.relative.appendChild(link_button),link_dialog=null,link_button=null},setDialog:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");return t.className="modal-content sun-editor-id-dialog-link",t.style.display="none",t.innerHTML='",t},setController_LinkButton:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");return t.className="sun-editor-id-link-btn",t.style.display="none",t.innerHTML='
',t},submit:function(e){this.showLoading(),e.preventDefault(),e.stopPropagation();const t=function(){if(0===this.context.link.focusElement.value.trim().length)return!1;const e=this.context.link.focusElement.value,t=this.context.link.linkAnchorText,i=0===t.value.length?e:t.value;if(this.context.dialog.updateModal)this.context.link._linkAnchor.href=e,this.context.link._linkAnchor.textContent=i,this.context.link._linkAnchor.target=this.context.link.linkNewWindowCheck.checked?"_blank":"",this.setRange(this.context.link._linkAnchor.childNodes[0],0,this.context.link._linkAnchor.childNodes[0],this.context.link._linkAnchor.textContent.length);else{const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("A");t.href=e,t.textContent=i,t.target=this.context.link.linkNewWindowCheck.checked?"_blank":"",this.insertNode(t),this.setRange(t.childNodes[0],0,t.childNodes[0],t.textContent.length)}this.context.link.focusElement.value="",this.context.link.linkAnchorText.value=""}.bind(this);try{t()}finally{this.plugins.dialog.close.call(this),this.closeLoading(),this.focus()}return!1},call_controller_linkButton:function(e){this.editLink=this.context.link._linkAnchor=e;const t=this.context.link.linkBtn;t.getElementsByTagName("A")[0].href=e.href,t.getElementsByTagName("A")[0].textContent=e.textContent;const i=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getOffset(e);t.style.left=i.left-this.context.element.wysiwyg.scrollLeft+"px",t.style.top=i.top+e.offsetHeight+10+"px",t.style.display="block",this.controllerArray=[t]},onClick_linkBtn:function(e){e.stopPropagation();const t=e.target.getAttribute("data-command")||e.target.parentNode.getAttribute("data-command");t&&(e.preventDefault(),/update/.test(t)?(this.context.link.focusElement.value=this.context.link._linkAnchor.href,this.context.link.linkAnchorText.value=this.context.link._linkAnchor.textContent,this.context.link.linkNewWindowCheck.checked=!!/_blank/i.test(this.context.link._linkAnchor.target),this.plugins.dialog.open.call(this,"link",!0)):(_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeItem(this.context.link._linkAnchor),this.context.link._linkAnchor=null,this.focus()),this.controllersOff())},init:function(){const e=this.context.link;e.linkBtn.style.display="none",e._linkAnchor=null,e.focusElement.value="",e.linkAnchorText.value="",e.linkNewWindowCheck.checked=!1}}},VquE:function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d");__webpack_exports__.a={name:"table",add:function(core,targetElement){const context=core.context;context.table={_element:null,_tdElement:null,_trElement:null,_trElements:null,_tdIndex:0,_trIndex:0,_tdCnt:0,_trCnt:0,_tableXY:[]};let listDiv=eval(this.setSubmenu()),tablePicker=listDiv.getElementsByClassName("sun-editor-id-table-picker")[0];context.table.tableHighlight=listDiv.getElementsByClassName("sun-editor-id-table-highlighted")[0],context.table.tableUnHighlight=listDiv.getElementsByClassName("sun-editor-id-table-unhighlighted")[0],context.table.tableDisplay=listDiv.getElementsByClassName("sun-editor-table-display")[0];let resizeDiv=eval(this.setController_tableEditor(core.lang));context.table.resizeDiv=resizeDiv,tablePicker.addEventListener("mousemove",this.onMouseMove_tablePicker.bind(core)),tablePicker.addEventListener("click",this.appendTable.bind(core)),resizeDiv.addEventListener("click",this.onClick_resizeDiv.bind(core)),targetElement.parentNode.appendChild(listDiv),context.element.relative.appendChild(resizeDiv),listDiv=null,tablePicker=null,resizeDiv=null},setSubmenu:function(){const e=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");return e.className="table-content",e.style.display="none",e.innerHTML='
1 x 1
',e},setController_tableEditor:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");return t.className="sun-editor-id-table-edit",t.style.display="none",t.innerHTML='
',t},appendTable:function(){const e=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("TABLE");let t=this.context.table._tableXY[0],i=this.context.table._tableXY[1],n="";for(;i>0;){n+="";let e=t;for(;e>0;)n+="
"+_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.zeroWidthSpace+"
",--e;n+="",--i}n+="",e.innerHTML=n;const o=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getFormatElement(this.getSelectionNode());this.insertNode(e,/^LI$/i.test(o.nodeName)?_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getRangeFormatElement(o):o),this.appendFormatTag(e),this.focus(),this.plugins.table.reset_table_picker.call(this)},onMouseMove_tablePicker:function(e){e.stopPropagation();let t=Math.ceil(e.offsetX/18),i=Math.ceil(e.offsetY/18);t=t<1?1:t,i=i<1?1:i,this.context.table.tableHighlight.style.width=t+"em",this.context.table.tableHighlight.style.height=i+"em";let n=t<5?5:t>9?10:t+1,o=i<5?5:i>9?10:i+1;this.context.table.tableUnHighlight.style.width=n+"em",this.context.table.tableUnHighlight.style.height=o+"em",_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.changeTxt(this.context.table.tableDisplay,t+" x "+i),this.context.table._tableXY=[t,i]},reset_table_picker:function(){if(!this.context.table.tableHighlight)return;const e=this.context.table.tableHighlight.style,t=this.context.table.tableUnHighlight.style;e.width="1em",e.height="1em",t.width="5em",t.height="5em",_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.changeTxt(this.context.table.tableDisplay,"1 x 1"),this.submenuOff()},init:function(){const e=this.context.table;e._tdElement&&_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(e._tdElement,"sun-editor-selected-cell"),e._element=null,e._tdElement=null,e._trElement=null,e._trElements=0,e._tdIndex=0,e._trIndex=0,e._trCnt=0,e._tdCnt=0,e._tableXY=[]},call_controller_tableEdit:function(e){const t=this.context.table.resizeDiv;this.plugins.table.setPositionControllerDiv.call(this,e,!1),t.style.display="block",this.controllerArray=[t],this.controllerFunction=[this.plugins.table.init.bind(this)]},setPositionControllerDiv:function(e,t){const i=this.context.table,n=i.resizeDiv;let o=i._element;if(!o){for(o=e;!/^TABLE$/i.test(o.nodeName);)o=o.parentNode;i._element=o}i._tdElement!==e&&(i._tdElement&&_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.removeClass(i._tdElement,"sun-editor-selected-cell"),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.addClass(e,"sun-editor-selected-cell"),i._tdElement=e,i._trElement=e.parentNode),(t||0===i._trCnt)&&(i._trElements=o.rows,i._tdIndex=e.cellIndex,i._trIndex=i._trElement.rowIndex,i._trCnt=o.rows.length,i._tdCnt=i._trElement.cells.length);const l=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.getOffset(e);n.style.left=l.left-this.context.element.wysiwyg.scrollLeft+"px",n.style.top=l.top+e.offsetHeight+12+"px"},insertRowCell:function(e,t){const i=this.context.table;if("row"===e){const e="up"===t?i._trIndex:i._trIndex+1;let n="";for(let e=0,t=i._tdCnt;e
"+_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.zeroWidthSpace+"
";i._element.insertRow(e).innerHTML=n}else{const e=i._trElements,n="left"===t?i._tdIndex:i._tdIndex+1;let o=null;for(let t=0,l=i._trCnt;t"+_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.zeroWidthSpace+""}this.plugins.table.setPositionControllerDiv.call(this,i._tdElement,!0)},deleteRowCell:function(e){const t=this.context.table;if("row"===e)t._element.deleteRow(t._trIndex);else{const e=t._trElements,i=t._tdIndex;for(let n=0,o=t._trCnt;n'+e.toolbar.alignLeft+'
  • ",t},pickup:function(e){e.preventDefault(),e.stopPropagation();let t=e.target,i=null;for(;!i&&!/UL/i.test(t.tagName);)i=t.getAttribute("data-command"),t=t.parentNode;this.focus(),this.execCommand(i,!1,null),this.submenuOff()}}},WUQj:function(e,t,i){},WzUB:function(module,__webpack_exports__,__webpack_require__){"use strict";var _lib_util__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("8e1d");__webpack_exports__.a={name:"formatBlock",add:function(core,targetElement){let listDiv=eval(this.setSubmenu(core.lang));listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickUp.bind(core)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement("DIV");return t.className="layer_editor layer_block",t.style.display="none",t.innerHTML='
    ",t},pickUp:function(e){e.preventDefault(),e.stopPropagation();let t=e.target,i=null,n=null;for(;!i&&!/UL/i.test(t.tagName);)i=t.getAttribute("data-command"),n=t.getAttribute("data-value"),t=t.parentNode;if("range"===i){const e=_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.createElement(n);this.wrapToTags(e),this.setRange(e.firstChild,0,e.firstChild,0),this.appendFormatTag(e,_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.isCell(this.getSelectionNode())?"DIV":"")}else this.execCommand("formatBlock",!1,n),_lib_util__WEBPACK_IMPORTED_MODULE_0__.a.changeTxt(this.commandMap.FORMAT,n);this.submenuOff(),this.focus()}}},XJR1:function(e,t,i){"use strict";i.r(t);i("3FqI"),i("WUQj");var n=i("WRt5"),o=i("50IV"),l=i("MIhV"),a=i("0A7J"),s=i("s0fJ"),r=i("g4XY"),d=i("gMuy"),c=i("VquE"),_=i("WzUB"),u=i("Rp48"),g=i("KKur"),m=i("hlhS"),h={align:n.a,font:o.a,fontSize:l.a,fontColor:a.a,hiliteColor:s.a,horizontalRule:r.a,list:d.a,table:c.a,formatBlock:_.a,link:u.a,image:g.a,video:m.a},p=i("8e1d"),f=i("PAX9");var b=function(e,t,i){const n=document,o=window,l={context:e,plugins:{},initPlugins:{},lang:i,dialogForm:null,submenu:null,submenuActiveButton:null,controllerArray:[],controllerFunction:[],codeViewDisabledButtons:e.element.toolbar.querySelectorAll('.sun-editor-id-toolbar button:not([class~="code-view-enabled"])'),_isInline:/inline/i.test(e.option.mode),_isBalloon:/balloon/i.test(e.option.mode),_inlineToolbarAttr:{width:0,height:0,isShow:!1},_imageUpload:function(e,t,i){s.onImageUpload&&s.onImageUpload(e,t,i)},commandMap:{FORMAT:e.tool.format,FONT:e.tool.font,SIZE:e.tool.fontSize,B:e.tool.bold,U:e.tool.underline,I:e.tool.italic,STRIKE:e.tool.strike,SUB:e.tool.subscript,SUP:e.tool.superscript},_variable:{selectionNode:null,range:null,wysiwygActive:!0,isFullScreen:!1,innerHeight_fullScreen:0,resizeClientY:0,tabSize:4,minResizingSize:65,currentNodes:[],_originCssText:e.element.topArea.style.cssText,_bodyOverflow:"",_editorAreaOriginCssText:"",_wysiwygOriginCssText:"",_codeOriginCssText:"",_sticky:!1,_imagesInfo:[],_imageIndex:0},callPlugin:function(e,t){if(!this.plugins[e])throw Error('[SUNEDITOR.core.callPlugin.fail] The called plugin does not exist or is in an invalid format. (pluginName:"'+e+'")');this.initPlugins[e]||(this.plugins[e].add(this,this.plugins[e].buttonElement),this.initPlugins[e]=!0),t()},addModule:function(e){let t="";for(let i=0,n=e.length;i0){for(let t=0;t0){for(let e=0;e":i)},focus:function(){if("none"===e.element.wysiwyg.style.display)return;const t=p.a.getParentElement(this.getSelectionNode(),"figcaption");t?t.focus():e.element.wysiwyg.focus(),this._setEditorRange(),a._findButtonEffectTag()},_setEditorRange:function(){const e=o.getSelection();let t=null;t=e.rangeCount>0?e.getRangeAt(0):this._createDefaultRange(),this._variable.range=t,t.collapsed?this.setSelectionNode(t.commonAncestorContainer):this.setSelectionNode(e.extentNode||e.anchorNode)},_createDefaultRange:function(){const t=n.createRange();return t.setStart(e.element.wysiwyg.firstChild,0),t.setEnd(e.element.wysiwyg.firstChild,0),t},setRange:function(e,t,i,l){const a=n.createRange();a.setStart(e,t),a.setEnd(i,l);const s=o.getSelection();s.removeAllRanges&&s.removeAllRanges(),s.addRange(a),this._variable.range=a,this._setEditorRange()},getRange:function(){return this._variable.range||this._createDefaultRange()},setSelectionNode:function(e){this._variable.selectionNode=e},getSelectionNode:function(){return this._variable.selectionNode?this._variable.selectionNode:e.element.wysiwyg.firstChild},getSelectedFormatElements:function(){let t=this.getRange();if(p.a.isWysiwygDiv(t.startContainer)){const i=e.element.wysiwyg.children;this.setRange(i[0],0,i[i.length-1],i[i.length-1].textContent.length),t=this.getRange()}const i=t.startContainer,n=t.endContainer,o=t.commonAncestorContainer,l=[];if(!p.a.isWysiwygDiv(o)&&!p.a.isRangeFormatElement(o))return[p.a.getFormatElement(o)];const a=p.a.getListChildren(o,function(e){return p.a.isFormatElement(e)});if(i===n)return a[0];let s=p.a.getFormatElement(i),r=p.a.getFormatElement(n),d=0,c=0;for(let e=0,t=a.length;e=0;e--)r[e]===n.parentNode&&r[e].firstChild===n&&0===i&&(d=e,n=n.parentNode);for(let e=c-1,t=n;e>d;e--)r[e]===t.parentNode&&1===r[e].nodeType&&(r.splice(e,1),t=t.parentNode,--c);for(let e=d;e<=c;e++){const l=r[e];0===l.length||3===l.nodeType&&void 0===l.data?p.a.removeItem(l):l!==t?l!==n?p.a.removeItem(l):(s=1===n.nodeType?p.a.createTextNode(n.textContent):p.a.createTextNode(n.substringData(o,n.length-o))).length>0?n.data=s.data:p.a.removeItem(n):(a=1===t.nodeType?p.a.createTextNode(t.textContent):p.a.createTextNode(t.substringData(0,i))).length>0?t.data=a.data:p.a.removeItem(t)}},wrapToTags:function(e){const t=this.getRange(),i=this.getSelectedFormatElements();if(!i){const t=p.a.createElement(p.a.isCell(this.getSelectionNode())?"DIV":"P");return t.innerHTML=p.a.zeroWidthSpace,e.appendChild(t),void this.getSelectionNode().appendChild(e)}let n,o,l,a=i[i.length-1];n=p.a.isRangeFormatElement(a)||p.a.isFormatElement(a)?a:p.a.getRangeFormatElement(a)||p.a.getFormatElement(a),p.a.isCell(n)?(o=null,l=n):(o=n.nextSibling,l=n.parentNode);let s=null,r=null,d="";for(let t=0,n=i.length;t=t.length)return}}if(o=i.startContainer,l=i.startOffset,1===o.nodeType&&o.childNodes.length>0)for(;o&&!p.a.isBreak(o)&&1===o.nodeType;){s=[];for(let e=0,t=(a=o.childNodes).length;e0){for(;o&&!p.a.isBreak(o)&&1===o.nodeType;){s=[];for(let e=0,t=(a=o.childNodes).length;e0&&(t=e.style.cssText.replace(h,"").trim()),(t.length>0||e.nodeName!==g)&&(e.style.cssText.length>0&&(e.style.cssText=t),!0)};if(p.a.isWysiwygDiv(u)||p.a.isRangeFormatElement(u)){const t=this.getSelectedFormatElements(),i=t.length-1;m=e.cloneNode(!1),f=this._nodeChange_startLine(t[0],m,v,r,d,n);for(let o=1;o0?(m=e.cloneNode(!1),b=this._nodeChange_endLine(t[i],m,v,c,_,n)):b=f}else{m=e.cloneNode(!1);const t=this._nodeChange_oneLine(p.a.getFormatElement(u),m,v,r,d,c,_,n,i.collapsed);f.container=t.startContainer,f.offset=t.startOffset,b.container=t.endContainer,b.offset=t.endOffset}this.setRange(f.container,f.offset,b.container,b.offset)},_nodeChange_oneLine:function(e,t,i,n,o,l,a,s,r){const d=e,c=e.cloneNode(!1),_=n===l;let u,g,m,h,f=n,b=o,v=l,y=a,E=!1,x=!1;function C(e){const t=new RegExp("(?:;|^|\\s)(?:"+h+"null)\\s*:[^;]*\\s*(?:;|$)","ig");let i="";return t&&e.style.cssText.length>0&&(i=t.test(e.style.cssText)),!i}return function e(n,o){const l=n.childNodes;for(let n=0,a=l.length;nb?y-b:b-y));for(e.data.length>0&&o.appendChild(e),g=k,u=[],h="";g!==c&&g!==d&&null!==g;)i(g)&&1===g.nodeType&&C(g)&&(u.push(g.cloneNode(!1)),h+=g.style.cssText.substr(0,g.style.cssText.indexOf(":"))+"|"),g=g.parentNode;const l=u.pop()||n;for(m=g=l;u.length>0;)g=u.pop(),m.appendChild(g),m=g;if(t.appendChild(l),c.appendChild(t),f=n,b=0,E=!0,g!==n&&g.appendChild(f),!_)continue}if(x||k!==v){if(E){if(1===k.nodeType&&!p.a.isBreak(k)){e(k,k);continue}for(g=k,u=[],h="";null!==g.parentNode&&g!==d&&g!==t;)1===g.nodeType&&!p.a.isBreak(k)&&(x||i(g))&&C(g)&&(u.push(g.cloneNode(!1)),h+=g.style.cssText.substr(0,g.style.cssText.indexOf(":"))+"|"),g=g.parentNode;const n=u.pop()||k;for(m=g=n;u.length>0;)g=u.pop(),m.appendChild(g),m=g;n===k?o=x?c:t:x?(c.appendChild(n),o=g):(t.appendChild(n),o=g)}a=k.cloneNode(!1),o.appendChild(a),1!==k.nodeType||p.a.isBreak(k)||(w=a),e(k,w)}else{const e=p.a.createTextNode(1===v.nodeType?"":v.substringData(y,v.length-y)),n=p.a.createTextNode(_||1===v.nodeType?"":v.substringData(0,y));if(e.data.length>0){for(g=k,h="",u=[];g!==c&&g!==d&&null!==g;)1===g.nodeType&&C(g)&&(u.push(g.cloneNode(!1)),h+=g.style.cssText.substr(0,g.style.cssText.indexOf(":"))+"|"),g=g.parentNode;for(a=m=g=u.pop()||e;u.length>0;)g=u.pop(),m.appendChild(g),m=g;c.appendChild(a),g.textContent=e.data}for(g=k,u=[],h="";g!==c&&g!==d&&null!==g;)i(g)&&1===g.nodeType&&C(g)&&(u.push(g.cloneNode(!1)),h+=g.style.cssText.substr(0,g.style.cssText.indexOf(":"))+"|"),g=g.parentNode;const o=u.pop()||n;for(m=g=o;u.length>0;)g=u.pop(),m.appendChild(g),m=g;t.appendChild(o),v=n,y=n.data.length,x=!0,!s&&r&&(t=n,n.textContent=p.a.zeroWidthSpace),g!==n&&g.appendChild(v)}}}(e,c),s?(f=p.a.createTextNode(r?p.a.zeroWidthSpace:t.textContent),c.insertBefore(f,t),c.removeChild(t),r&&(b=1)):r&&(f=v=t,b=1,y=1),p.a.removeEmptyNode(c),e.parentNode.insertBefore(c,e),p.a.removeItem(e),{startContainer:f,startOffset:b,endContainer:s||!v.textContent?f:v,endOffset:s||!v.textContent?f.textContent.length:y}},_nodeChange_middleLine:function(e,t,i,n){n?t=p.a.createTextNode(e.textContent?e.textContent:p.a.zeroWidthSpace):function e(t,n){const o=t.childNodes;for(let t=0,l=o.length;t0){const e=r.pop();for(c=d=e;r.length>0;)d=r.pop(),c.appendChild(d),c=d;t.appendChild(e),o=d}else o=t}if(g||m!==_){if(!g||i(m)){const e=m.cloneNode(!1);o.appendChild(e),1!==m.nodeType||p.a.isBreak(m)||(h=e)}e(m,h)}else{const e=p.a.createTextNode(1===_.nodeType?"":_.substringData(0,u)),n=p.a.createTextNode(1===_.nodeType?"":_.substringData(u,_.length-u));for(e.data.length>0&&o.appendChild(e),d=o,r=[];d!==s&&null!==d;)1===d.nodeType&&i(d)&&r.push(d.cloneNode(!1)),d=d.parentNode;const l=r.pop()||o;for(c=d=l;r.length>0;)d=r.pop(),c.appendChild(d),c=d;l!==o?(t.appendChild(l),o=d):o=t,p.a.isBreak(m)&&t.appendChild(m.cloneNode(!1)),s.appendChild(t),_=n,u=0,g=!0,o.appendChild(_)}}}(e,s),l&&(_=p.a.createTextNode(t.textContent),s.insertBefore(_,t),s.removeChild(t)),l||0!==s.children.length?(p.a.removeEmptyNode(s),e.parentNode.insertBefore(s,e),p.a.removeItem(e)):e.childNodes?_=e.childNodes[0]:(_=p.a.createTextNode(p.a.zeroWidthSpace),e.appendChild(_)),{container:_,offset:u}},_nodeChange_endLine:function(e,t,i,n,o,l){const a=e,s=e.cloneNode(!1);let r,d,c,_=n,u=o,g=!1;return function e(n,o){const l=n.childNodes;for(let n=l.length-1;0<=n;n--){const m=l[n];let h=o;if(g&&!p.a.isBreak(m)){if(1===m.nodeType){e(m,m);continue}for(d=m,r=[];null!==d.parentNode&&d!==a&&d!==t;)i(d)&&1===d.nodeType&&r.push(d.cloneNode(!1)),d=d.parentNode;if(r.length>0){const e=r.pop();for(c=d=e;r.length>0;)d=r.pop(),c.appendChild(d),c=d;t.insertBefore(e,t.firstChild),o=d}else o=t}if(g||m!==_){if(!g||i(m)){const e=m.cloneNode(!1);o.insertBefore(e,o.firstChild),1!==m.nodeType||p.a.isBreak(m)||(h=e)}e(m,h)}else{const e=p.a.createTextNode(1===_.nodeType?"":_.substringData(u,_.length-u)),n=p.a.createTextNode(1===_.nodeType?"":_.substringData(0,u));for(e.data.length>0&&o.insertBefore(e,o.firstChild),d=o,r=[];d!==s&&null!==d;)i(d)&&1===d.nodeType&&r.push(d.cloneNode(!1)),d=d.parentNode;const l=r.pop()||o;for(c=d=l;r.length>0;)d=r.pop(),c.appendChild(d),c=d;l!==o?(t.insertBefore(l,t.firstChild),o=d):o=t,p.a.isBreak(m)&&t.appendChild(m.cloneNode(!1)),s.insertBefore(t,s.firstChild),_=n,u=n.data.length,g=!0,o.insertBefore(_,o.firstChild)}}}(e,s),l&&(_=p.a.createTextNode(t.textContent),u=_.textContent.length,s.insertBefore(_,t),s.removeChild(t)),l||0!==s.childNodes.length?(p.a.removeEmptyNode(s),e.parentNode.insertBefore(s,e),p.a.removeItem(e)):e.childNodes?_=e.childNodes[0]:(_=p.a.createTextNode(p.a.zeroWidthSpace),e.appendChild(_)),{container:_,offset:u}},commandHandler:function(t,i){switch(i){case"codeView":this.controllersOff(),this.toggleCodeView(),p.a.toggleClass(t,"on");break;case"fullScreen":this.controllersOff(),this.toggleFullScreen(t),p.a.toggleClass(t,"on");break;case"indent":case"outdent":this.indent(i);break;case"redo":case"undo":this.execCommand(i,!1,null);break;case"removeFormat":this.removeFormat();break;case"preview":case"print":this.openWindowContents(i);break;case"showBlocks":this.toggleDisplayBlocks(),p.a.toggleClass(t,"on");break;case"subscript":p.a.hasClass(e.tool.superscript,"on")&&(this.execCommand("superscript",!1,null),p.a.removeClass(e.tool.superscript,"on")),this.execCommand(i,!1,null),p.a.toggleClass(t,"on");break;case"superscript":p.a.hasClass(e.tool.subscript,"on")&&(this.execCommand("subscript",!1,null),p.a.removeClass(e.tool.subscript,"on")),this.execCommand(i,!1,null),p.a.toggleClass(t,"on");break;default:this.execCommand(i,!1,t.getAttribute("data-value")),p.a.toggleClass(t,"on")}this.focus()},removeFormat:function(){this.nodeChange(p.a.createElement("REMOVENODE"))},indent:function(e){const t=this.getSelectedFormatElements();let i,n;for(let o=0,l=t.length;o0?e.element.code.scrollHeight+"px":"auto"),this._variable.wysiwygActive=!1,e.element.code.focus();else{const t=e.element.code.value.trim();e.element.wysiwyg.innerHTML=t.length>0?p.a.convertContentsForEditor(t):"

    "+p.a.zeroWidthSpace+"

    ",e.element.wysiwyg.scrollTop=0,e.element.code.style.display="none",e.element.wysiwyg.style.display="block","auto"===e.option.height&&(e.element.code.style.height="0px"),this._variable.wysiwygActive=!0,this.focus()}},toggleFullScreen:function(t){this._variable.isFullScreen?(this._variable.isFullScreen=!1,e.element.code.style.cssText=this._variable._codeOriginCssText,e.element.wysiwyg.style.cssText=this._variable._wysiwygOriginCssText,e.element.toolbar.style.cssText="",e.element.editorArea.style.cssText=this._variable._editorAreaOriginCssText,e.element.topArea.style.cssText=this._variable._originCssText,n.body.style.overflow=this._variable._bodyOverflow,e.option.stickyToolbar>-1&&(p.a.removeClass(e.element.toolbar,"sun-editor-sticky"),a.onScroll_window()),p.a.removeClass(t.firstElementChild,"icon-reduction"),p.a.addClass(t.firstElementChild,"icon-expansion")):(this._variable.isFullScreen=!0,e.element.topArea.style.position="fixed",e.element.topArea.style.top="0",e.element.topArea.style.left="0",e.element.topArea.style.width="100%",e.element.topArea.style.height="100%",e.element.topArea.style.zIndex="2147483647",this._variable._bodyOverflow=n.body.style.overflow,n.body.style.overflow="hidden",this._variable._editorAreaOriginCssText=e.element.editorArea.style.cssText,this._variable._wysiwygOriginCssText=e.element.wysiwyg.style.cssText,this._variable._codeOriginCssText=e.element.code.style.cssText,e.element.editorArea.style.cssText=e.element.toolbar.style.cssText=e.element.wysiwyg.style.cssText=e.element.code.style.cssText="",e.element.toolbar.style.width=e.element.wysiwyg.style.height=e.element.code.style.height="100%",e.element.toolbar.style.position="relative",this._variable.innerHeight_fullScreen=o.innerHeight-e.element.toolbar.offsetHeight,e.element.editorArea.style.height=this._variable.innerHeight_fullScreen+"px",p.a.removeClass(t.firstElementChild,"icon-expansion"),p.a.addClass(t.firstElementChild,"icon-reduction"))},openWindowContents:function(t){const n="print"===t,l=o.open("","_blank");l.mimeType="text/html",l.document.write(''+(n?i.toolbar.print:i.toolbar.preview)+'
    '+this.getContents()+"
    "+(n?"' : '') + '' + '');\n },\n\n /**\r\n * @description Gets the current contents\r\n * @returns {Object}\r\n */\n getContents: function () {\n let contents = '';\n if (context.element.wysiwyg.innerText.trim().length === 0) return contents;\n\n if (editor._variable.wysiwygActive) {\n contents = context.element.wysiwyg.innerHTML;\n } else {\n contents = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(context.element.code.value);\n }\n\n const renderHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement('DIV');\n renderHTML.innerHTML = contents;\n const figcaptions = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildren(renderHTML, function (current) {\n return /FIGCAPTION/i.test(current.nodeName);\n });\n\n for (let i = 0, len = figcaptions.length; i < len; i++) {\n figcaptions[i].outerHTML = figcaptions[i].outerHTML.replace(/(?!^]*>)/i, '');\n }\n\n return renderHTML.innerHTML;\n }\n };\n /**\r\n * @description event function\r\n */\n\n const event = {\n _shortcutKeyCode: {\n 66: ['bold', 'B'],\n 83: ['strikethrough', 'STRIKE'],\n 85: ['underline', 'U'],\n 73: ['italic', 'I'],\n 89: ['redo'],\n 90: ['undo'],\n 219: ['outdent'],\n 221: ['indent']\n },\n _directionKeyKeyCode: new RegExp('^(?:8|13|32|46|33|34|35|36|37|38|39|40|98|100|102|104)$'),\n _changeButtonClassTagCheck: new RegExp('^(?:B|U|I|STRIKE|SUB|SUP)$'),\n _findButtonEffectTag: function () {\n const commandMap = editor.commandMap;\n const classOnCheck = this._changeButtonClassTagCheck;\n const commandMapNodes = [];\n const currentNodes = [];\n let findFormat = true,\n findFont = true,\n findSize = true,\n findA = true;\n let findB = true,\n findI = true,\n findU = true,\n findS = true;\n let cssText = '',\n nodeName = '';\n\n for (let selectionParent = editor.getSelectionNode(); !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\n if (!selectionParent) break;\n if (selectionParent.nodeType !== 1) continue;\n nodeName = selectionParent.nodeName.toUpperCase();\n currentNodes.push(nodeName);\n /** Format */\n\n if (findFormat && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(selectionParent)) {\n commandMapNodes.push('FORMAT');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap.FORMAT, nodeName);\n findFormat = false;\n continue;\n }\n /** Font */\n\n\n if (findFont && (selectionParent.style.fontFamily.length > 0 || selectionParent.face && selectionParent.face.length > 0)) {\n commandMapNodes.push('FONT');\n const selectFont = (selectionParent.style.fontFamily || selectionParent.face || lang.toolbar.font).replace(/[\"']/g, '');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap.FONT, selectFont);\n findFont = false;\n }\n /** A */\n\n\n if (findA && /^A$/.test(nodeName) && selectionParent.getAttribute('data-image-link') === null) {\n if (!context.link || editor.controllerArray[0] !== context.link.linkBtn) {\n editor.callPlugin('link', function () {\n editor.plugins.link.call_controller_linkButton.call(editor, selectionParent);\n });\n }\n\n findA = false;\n } else if (findA && context.link && editor.controllerArray[0] === context.link.linkBtn) {\n editor.controllersOff();\n }\n /** SPAN */\n\n\n if (findSize && /^SPAN$/.test(nodeName)) {\n /** font size */\n if (selectionParent.style.fontSize.length > 0) {\n commandMapNodes.push('SIZE');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap.SIZE, selectionParent.style.fontSize.match(/\\d+/)[0]);\n findSize = false;\n }\n }\n /** command map */\n\n\n cssText = selectionParent.style.cssText;\n\n if (findB && /font\\-weight\\s*:\\s*(?:\\d+|bold|bolder)(?:;|\\s|)/.test(cssText)) {\n commandMapNodes.push('B');\n findB = false;\n }\n\n if (findI && /font\\-style\\s*:\\s*(?:italic|oblique)(?:;|\\s)/.test(cssText)) {\n commandMapNodes.push('I');\n findI = false;\n }\n\n if (findU && /text\\-decoration(?:\\-line)?\\s*:\\s*underline(?:;|\\s|)/.test(cssText)) {\n commandMapNodes.push('U');\n findU = false;\n }\n\n if (findS && /text\\-decoration(?:\\-line)?\\s*:\\s*line-through(?:;|\\s|)/.test(cssText)) {\n commandMapNodes.push('STRIKE');\n findS = false;\n }\n\n commandMapNodes.push(/^STRONG$/.test(nodeName) ? 'B' : /^EM$/.test(nodeName) ? 'I' : nodeName);\n }\n /** A Tag edit controller off */\n\n\n if (findA) editor.controllersOff();\n /** toggle class on */\n\n for (let i = 0; i < commandMapNodes.length; i++) {\n nodeName = commandMapNodes[i];\n\n if (classOnCheck.test(nodeName)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(commandMap[nodeName], 'on');\n }\n }\n /** remove class, display text */\n\n\n for (let key in commandMap) {\n if (commandMapNodes.indexOf(key) > -1) continue;\n\n if (/^FONT/i.test(key)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap[key], lang.toolbar.font);\n } else if (/^SIZE$/i.test(key)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap[key], lang.toolbar.fontSize);\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(commandMap[key], 'on');\n }\n }\n /** save current nodes */\n\n\n editor._variable.currentNodes = currentNodes.reverse();\n /** Displays the current node structure to resizingBar */\n\n if (context.option.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\n },\n _cancelCaptionEdit: function () {\n this.setAttribute('contenteditable', false);\n this.removeEventListener('blur', event._cancelCaptionEdit);\n },\n onClick_toolbar: function (e) {\n e.preventDefault();\n e.stopPropagation();\n let target = e.target;\n let display = target.getAttribute('data-display');\n let command = target.getAttribute('data-command');\n let className = target.className;\n\n while (!command && !/editor_tool/.test(className) && !/sun-editor-id-toolbar/.test(className)) {\n target = target.parentNode;\n command = target.getAttribute('data-command');\n display = target.getAttribute('data-display');\n className = target.className;\n }\n\n if (!command && !display) return;\n if (target.disabled) return;\n /** Dialog, Submenu */\n\n if (display) {\n if (/submenu/.test(display) && (target.nextElementSibling === null || target !== editor.submenuActiveButton)) {\n editor.submenuOff();\n editor.callPlugin(command, function () {\n editor.submenuOn(target);\n });\n return;\n } else if (/dialog/.test(display)) {\n editor.callPlugin(command, function () {\n editor.plugins.dialog.open.call(editor, command, false);\n });\n }\n\n editor.submenuOff();\n return;\n }\n\n editor.submenuOff();\n /** default command */\n\n if (command) {\n editor.focus();\n editor.commandHandler(target, command);\n }\n },\n onClick_wysiwyg: function (e) {\n e.stopPropagation();\n const targetElement = e.target;\n editor.submenuOff();\n\n if (/^IMG$/i.test(targetElement.nodeName)) {\n e.preventDefault();\n editor.callPlugin('image', function () {\n const size = editor.plugins.resizing.call_controller_resize.call(editor, targetElement, 'image');\n editor.plugins.image.onModifyMode.call(editor, targetElement, size);\n\n if (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(targetElement, '.sun-editor-id-image-container')) {\n editor.plugins.image.openModify.call(editor, true);\n editor.plugins.image.update_image.call(editor);\n editor.controllersOff();\n }\n });\n return;\n }\n\n if (/sun-editor-id-iframe-inner-resizing-cover/i.test(targetElement.className)) {\n e.preventDefault();\n editor.callPlugin('video', function () {\n const iframe = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getChildElement(targetElement.parentNode, 'iframe');\n const size = editor.plugins.resizing.call_controller_resize.call(editor, iframe, 'video');\n editor.plugins.video.onModifyMode.call(editor, iframe, size);\n });\n return;\n }\n\n editor._setEditorRange();\n\n event._findButtonEffectTag();\n\n if (editor._isBalloon) {\n const range = editor.getRange();\n\n if (range.collapsed) {\n event._hideToolbar();\n } else {\n event._showToolbarBalloon(range);\n\n return;\n }\n }\n\n const figcaption = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(targetElement, 'FIGCAPTION');\n\n if (figcaption && figcaption.getAttribute('contenteditable') !== 'ture') {\n e.preventDefault();\n figcaption.setAttribute('contenteditable', true);\n figcaption.focus();\n\n if (editor._isInline && !editor._inlineToolbarAttr.isShow) {\n event._showToolbarInline();\n\n const hideToolbar = function () {\n event._hideToolbar();\n\n _d.removeEventListener('click', hideToolbar);\n };\n\n _d.addEventListener('click', hideToolbar);\n }\n } else {\n const td = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(targetElement, _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell);\n\n if (td) {\n if (editor.controllerArray.length === 0) {\n editor.callPlugin('table', editor.plugins.table.call_controller_tableEdit.bind(editor, td));\n }\n }\n }\n\n if (userFunction.onClick) userFunction.onClick(e);\n },\n _showToolbarBalloon: function (range) {\n const toolbar = context.element.toolbar;\n const childNodes = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildNodes(range.commonAncestorContainer);\n\n const selection = _w.getSelection();\n\n const isDirTop = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getArrayIndex(childNodes, selection.focusNode) < _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getArrayIndex(childNodes, selection.anchorNode);\n let rects = range.getClientRects();\n rects = rects[isDirTop ? 0 : rects.length - 1];\n toolbar.style.display = 'block';\n let l = (isDirTop ? rects.left : rects.right) - context.element.topArea.offsetLeft + _w.scrollX - toolbar.offsetWidth / 2;\n let t = (isDirTop ? rects.top - toolbar.offsetHeight - 11 : rects.bottom + 11) - context.element.topArea.offsetTop + _w.scrollY;\n toolbar.style.left = (l < 0 ? 20 : l) + 'px';\n toolbar.style.top = t + 'px';\n\n if (isDirTop) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.element._arrow, 'arrow-up');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(context.element._arrow, 'arrow-down');\n context.element._arrow.style.top = toolbar.offsetHeight + 'px';\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.element._arrow, 'arrow-down');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(context.element._arrow, 'arrow-up');\n context.element._arrow.style.top = '-11px';\n }\n\n const arrow_width = context.element._arrow.offsetWidth;\n const arrow_left = toolbar.offsetWidth / 2 + (l < 0 ? l - arrow_width : 0);\n context.element._arrow.style.left = (arrow_left < arrow_width / 2 ? arrow_width / 2 - 1 : arrow_left) + 'px';\n },\n _showToolbarInline: function () {\n const toolbar = context.element.toolbar;\n toolbar.style.display = 'block';\n editor._inlineToolbarAttr.width = toolbar.style.width = context.option.toolbarWidth;\n editor._inlineToolbarAttr.top = toolbar.style.top = -1 - toolbar.offsetHeight + 'px';\n event.onScroll_window();\n editor._inlineToolbarAttr.isShow = true;\n },\n _hideToolbar: function () {\n context.element.toolbar.style.display = 'none';\n editor._inlineToolbarAttr.isShow = false;\n },\n onKeyDown_wysiwyg: function (e) {\n const keyCode = e.keyCode;\n const shift = e.shiftKey;\n const ctrl = e.ctrlKey || e.metaKey;\n const alt = e.altKey;\n e.stopPropagation();\n\n if (editor._isBalloon) {\n event._hideToolbar();\n }\n\n function shortcutCommand(keyCode) {\n const key = event._shortcutKeyCode[keyCode];\n if (!key) return false;\n editor.commandHandler(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(editor.getSelectionNode()), key[0]);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(editor.commandMap[key[1]], 'on');\n return true;\n }\n /** Shortcuts */\n\n\n if (ctrl && !/^(?:16|17|18)$/.test(keyCode)) {\n if (!(shift && keyCode !== 83) && shortcutCommand(keyCode)) {\n e.preventDefault();\n return;\n }\n }\n /** default key action */\n\n\n const selectionNode = editor.getSelectionNode();\n\n switch (keyCode) {\n case 8:\n /**backspace key*/\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(selectionNode) && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode.parentNode) && selectionNode.previousSibling === null) {\n e.preventDefault();\n e.stopPropagation();\n selectionNode.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n return false;\n }\n\n break;\n\n case 9:\n /**tab key*/\n e.preventDefault();\n if (ctrl || alt) break;\n editor.controllersOff();\n let currentNode = selectionNode || editor.getSelectionNode();\n\n while (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(currentNode) && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(currentNode)) {\n currentNode = currentNode.parentNode;\n }\n\n if (currentNode && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(currentNode)) {\n const table = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(currentNode, 'table');\n const cells = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildren(table, _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell);\n let idx = shift ? _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].prevIdx(cells, currentNode) : _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].nextIdx(cells, currentNode);\n if (idx === cells.length && !shift) idx = 0;\n if (idx === -1 && shift) idx = cells.length - 1;\n const moveCell = cells[idx];\n if (!moveCell) return false;\n editor.setRange(moveCell, 0, moveCell, 0);\n break;\n }\n /** format Tag */\n\n\n const lines = editor.getSelectedFormatElements();\n\n if (!shift) {\n const tabText = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\n\n if (lines.length === 1) {\n editor.insertNode(tabText);\n editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize);\n } else {\n for (let i = 0, len = lines.length; i < len; i++) {\n lines[i].insertBefore(tabText.cloneNode(false), lines[i].firstChild);\n }\n }\n } else {\n for (let i = 0, len = lines.length, child; i < len; i++) {\n child = lines[i].firstChild;\n\n if (/^\\s{1,4}$/.test(child.textContent)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(child);\n } else if (/^\\s{1,4}/.test(child.textContent)) {\n child.textContent = child.textContent.replace(/^\\s{1,4}/, '');\n }\n }\n }\n\n break;\n }\n\n if (userFunction.onKeyDown) userFunction.onKeyDown(e);\n },\n onKeyUp_wysiwyg: function (e) {\n editor._setEditorRange();\n\n editor.controllersOff();\n const selectionNode = editor.getSelectionNode();\n /** when format tag deleted */\n\n if (e.keyCode === 8 && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode) && context.element.wysiwyg.textContent.length === 0) {\n e.preventDefault();\n e.stopPropagation();\n const oFormatTag = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(editor._variable.currentNodes[0]) ? editor._variable.currentNodes[0] : 'P');\n oFormatTag.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n selectionNode.appendChild(oFormatTag);\n editor.setSelectionNode(oFormatTag);\n editor.setRange(oFormatTag, 0, oFormatTag, 0);\n return;\n }\n\n if ((_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode.parentElement) || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(selectionNode.parentElement)) && selectionNode.nodeType === 3) {\n editor.execCommand('formatBlock', false, _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode.parentElement) ? 'P' : 'DIV');\n\n editor._setEditorRange();\n\n event._findButtonEffectTag();\n\n return;\n }\n\n if (event._directionKeyKeyCode.test(e.keyCode)) {\n event._findButtonEffectTag();\n }\n\n if (userFunction.onKeyUp) userFunction.onKeyUp(e);\n },\n onScroll_wysiwyg: function (e) {\n editor.controllersOff();\n if (userFunction.onScroll) userFunction.onScroll(e);\n },\n onDrop_wysiwyg: function (e) {\n const files = e.dataTransfer.files;\n\n if (files.length > 0) {\n e.stopPropagation();\n e.preventDefault();\n editor.focus();\n editor.callPlugin('image', function () {\n context.image.imgInputFile.files = files;\n editor.plugins.image.onRender_imgInput.call(editor);\n context.image.imgInputFile.files = null;\n });\n }\n\n if (userFunction.onDrop) userFunction.onDrop(e);\n },\n onMouseDown_resizingBar: function (e) {\n e.stopPropagation();\n editor._variable.resizeClientY = e.clientY;\n context.element.resizeBackground.style.display = 'block';\n\n function closureFunc() {\n context.element.resizeBackground.style.display = 'none';\n\n _d.removeEventListener('mousemove', event._resize_editor);\n\n _d.removeEventListener('mouseup', closureFunc);\n }\n\n _d.addEventListener('mousemove', event._resize_editor);\n\n _d.addEventListener('mouseup', closureFunc);\n },\n _resize_editor: function (e) {\n const resizeInterval = context.element.editorArea.offsetHeight + (e.clientY - editor._variable.resizeClientY);\n context.element.wysiwyg.style.height = context.element.code.style.height = (resizeInterval < editor._variable.minResizingSize ? editor._variable.minResizingSize : resizeInterval) + 'px';\n editor._variable.resizeClientY = e.clientY;\n },\n onResize_window: function () {\n if (context.element.topArea.offsetWidth === 0) return;\n\n if (editor._variable.isFullScreen) {\n editor._variable.innerHeight_fullScreen += _w.innerHeight - context.element.toolbar.offsetHeight - editor._variable.innerHeight_fullScreen;\n context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\n } else if (editor._variable._sticky) {\n context.element.toolbar.style.width = context.element.topArea.offsetWidth - 2 + 'px';\n event.onScroll_window();\n }\n\n editor.controllersOff();\n },\n onScroll_window: function () {\n if (editor._variable.isFullScreen || context.element.topArea.offsetWidth === 0) return;\n const element = context.element;\n const editorHeight = element.editorArea.offsetHeight;\n const editorTop = element.topArea.offsetTop - (editor._isInline ? element.toolbar.offsetHeight : 0);\n const y = (this.scrollY || _d.documentElement.scrollTop) + context.option.stickyToolbar;\n\n if (y < editorTop) {\n event._offStickyToolbar(element);\n } else if (y + editor._variable.minResizingSize >= editorHeight + editorTop) {\n if (!editor._variable._sticky) event._onStickyToolbar(element);\n element.toolbar.style.top = editorHeight + editorTop + context.option.stickyToolbar - y - editor._variable.minResizingSize + 'px';\n } else if (y >= editorTop) {\n event._onStickyToolbar(element);\n }\n },\n _onStickyToolbar: function (element) {\n if (!editor._isInline) {\n element._stickyDummy.style.height = element.toolbar.offsetHeight + 'px';\n element._stickyDummy.style.display = 'block';\n }\n\n element.toolbar.style.top = context.option.stickyToolbar + 'px';\n element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : element.toolbar.offsetWidth + 'px';\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(element.toolbar, 'sun-editor-sticky');\n editor._variable._sticky = true;\n },\n _offStickyToolbar: function (element) {\n element._stickyDummy.style.display = 'none';\n element.toolbar.style.top = editor._isInline ? editor._inlineToolbarAttr.top : '';\n element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : '';\n element.editorArea.style.marginTop = '';\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(element.toolbar, 'sun-editor-sticky');\n editor._variable._sticky = false;\n },\n _codeViewAutoScroll: function () {\n context.element.code.style.height = context.element.code.scrollHeight + 'px';\n },\n onPaste_wysiwyg: function (e) {\n if (!e.clipboardData.getData) return true;\n const cleanData = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].cleanHTML(e.clipboardData.getData('text/html'));\n\n if (cleanData) {\n editor.execCommand('insertHTML', false, cleanData);\n e.stopPropagation();\n e.preventDefault();\n }\n }\n };\n /** add event listeners */\n\n /** toolbar event */\n\n context.element.toolbar.addEventListener('click', event.onClick_toolbar, false);\n context.element.toolbar.addEventListener('mousedown', function (e) {\n e.preventDefault();\n }, false);\n /** editor area */\n\n context.element.relative.addEventListener('click', editor.focus.bind(editor), false);\n context.element.wysiwyg.addEventListener('click', event.onClick_wysiwyg, false);\n context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\n context.element.wysiwyg.addEventListener('keydown', event.onKeyDown_wysiwyg, false);\n context.element.wysiwyg.addEventListener('keyup', event.onKeyUp_wysiwyg, false);\n context.element.wysiwyg.addEventListener('drop', event.onDrop_wysiwyg, false);\n context.element.wysiwyg.addEventListener('paste', event.onPaste_wysiwyg, false);\n /** code view area auto line */\n\n if (context.option.height === 'auto') context.element.code.addEventListener('keyup', event._codeViewAutoScroll, false);\n /** resizingBar */\n\n if (context.element.resizingBar) {\n if (/\\d+/.test(context.option.height)) {\n context.element.resizingBar.addEventListener('mousedown', event.onMouseDown_resizingBar, false);\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(context.element.resizingBar, 'none-resize');\n }\n }\n /** inlineToolbar */\n\n\n if (editor._isInline) {\n context.element.wysiwyg.addEventListener('focus', event._showToolbarInline, false);\n }\n\n if (editor._isInline || editor._isBalloon) {\n context.element.wysiwyg.addEventListener('blur', event._hideToolbar, false);\n }\n /** window event */\n\n\n _w.addEventListener('resize', event.onResize_window, false);\n\n if (context.option.stickyToolbar > -1) _w.addEventListener('scroll', event.onScroll_window, false);\n /** add plugin to plugins object */\n\n if (plugins) {\n Object.keys(plugins).map(function (key) {\n let plugin = plugins[key];\n editor.plugins[plugin.name] = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].copyObj(plugin);\n });\n }\n /** User function */\n\n\n const userFunction = {\n /**\r\n * @description Event functions\r\n * @param {Object} event - Event Object\r\n */\n onScroll: null,\n onClick: null,\n onKeyDown: null,\n onKeyUp: null,\n onDrop: null,\n\n /**\r\n * @description Called when the image is uploaded or the uploaded image is deleted\r\n * @param {Element} targetImgElement - Current img element\r\n * @param {Number} index - Uploaded index\r\n * @param {Boolean} isDelete - Whether or not it was called after the delete operation\r\n */\n onImageUpload: null,\n\n /**\r\n * @description Open a notice area\r\n * @param {String} message - Notice message\r\n */\n noticeOpen: function (message) {\n editor.addModule([_plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"]]);\n _plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"].open.call(editor, message);\n },\n\n /**\r\n * @description Close a notice area\r\n */\n noticeClose: function () {\n editor.addModule([_plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"]]);\n _plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"].close.call(editor);\n },\n\n /**\r\n * @description Copying the contents of the editor to the original textarea\r\n */\n save: function () {\n context.element.originElement.value = editor.getContents();\n },\n\n /**\r\n * @description Gets the suneditor's context object. Contains settings, plugins, and cached element objects\r\n * @returns {Object}\r\n */\n getContext: function () {\n return context;\n },\n\n /**\r\n * @description Gets the contents of the suneditor\r\n * @returns {String}\r\n */\n getContents: function () {\n return editor.getContents();\n },\n\n /**\r\n * @description Gets uploaded images informations\r\n * @returns {Array}\r\n */\n getImagesInfo: function () {\n return editor._variable._imagesInfo;\n },\n\n /**\r\n * @description Inserts an HTML element or HTML string or plain string at the current cursor position\r\n * @param {Element|String} html - HTML Element or HTML string or plain string\r\n */\n insertHTML: function (html) {\n if (!html.nodeType || html.nodeType !== 1) {\n const template = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement('template');\n template.innerHTML = html;\n html = template.firstChild || template.content.firstChild;\n }\n\n editor.insertNode(html);\n editor.focus();\n },\n\n /**\r\n * @description Change the contents of the suneditor\r\n * @param {String} contents - Contents to Input\r\n */\n setContents: function (contents) {\n if (editor._variable.wysiwygActive) {\n context.element.wysiwyg.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(contents);\n } else {\n context.element.code.value = contents;\n }\n },\n\n /**\r\n * @description Add contents to the suneditor\r\n * @param {String} contents - Contents to Input\r\n */\n appendContents: function (contents) {\n if (editor._variable.wysiwygActive) {\n context.element.wysiwyg.innerHTML += _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(contents);\n } else {\n context.element.code.value += contents;\n }\n },\n\n /**\r\n * @description Disable the suneditor\r\n */\n disabled: function () {\n context.tool.cover.style.display = 'block';\n context.element.wysiwyg.setAttribute('contenteditable', false);\n context.element.code.setAttribute('disabled', 'disabled');\n },\n\n /**\r\n * @description Enabled the suneditor\r\n */\n enabled: function () {\n context.tool.cover.style.display = 'none';\n context.element.wysiwyg.setAttribute('contenteditable', true);\n context.element.code.removeAttribute('disabled');\n },\n\n /**\r\n * @description Show the suneditor\r\n */\n show: function () {\n const topAreaStyle = context.element.topArea.style;\n if (topAreaStyle.display === 'none') topAreaStyle.display = context.option.display;\n },\n\n /**\r\n * @description Hide the suneditor\r\n */\n hide: function () {\n context.element.topArea.style.display = 'none';\n },\n\n /**\r\n * @description Destroy the suneditor\r\n */\n destroy: function () {\n /** remove window event listeners */\n _w.removeEventListener('resize', event.onResize_window);\n\n _w.removeEventListener('scroll', event.onScroll_window);\n /** remove element */\n\n\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(context.element.topArea);\n this.onScroll = null;\n this.onClick = null;\n this.onKeyDown = null;\n this.onKeyUp = null;\n this.onDrop = null;\n this.save = null;\n this.onImageUpload = null;\n this.noticeOpen = null;\n this.noticeClose = null;\n this.getContext = null;\n this.getContents = null;\n this.getImagesInfo = null;\n this.insertHTML = null;\n this.setContents = null;\n this.appendContents = null;\n this.disabled = null;\n this.enabled = null;\n this.show = null;\n this.hide = null;\n this.destroy = null;\n context = null;\n plugins = null;\n lang = null;\n }\n };\n return userFunction;\n};\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (core);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/lib/core.js.js","sources":["webpack:///./src/lib/core.js?8a8d"],"sourcesContent":["/*\r\n * wysiwyg web editor\r\n *\r\n * suneditor.js\r\n * Copyright 2017 JiHong Lee.\r\n * MIT license.\r\n */\r\n'use strict';\r\n\r\nimport util from './util';\r\nimport notice from '../plugins/modules/notice';\r\n\r\n/**\r\n * @description SunEditor core closure\r\n * @param context\r\n * @param plugins\r\n * @param lang\r\n * @returns {{save: save, getContext: getContext, getContent: getContent, setContent: setContent, appendContent: appendContent, disabled: disabled, enabled: enabled, show: show, hide: hide, destroy: destroy}}\r\n */\r\nconst core = function (context, plugins, lang) {\r\n    const _d = document;\r\n    const _w = window;\r\n    /**\r\n     * @description editor core object\r\n     * should always bind this object when registering an event in the plug-in.\r\n     */\r\n    const editor = {\r\n        /**\r\n         * @description Elements and user options parameters of the suneditor\r\n         */\r\n        context: context,\r\n\r\n        /**\r\n         * @description loaded plugins\r\n         */\r\n        plugins: {},\r\n\r\n        /**\r\n         * @description Whether the plugin is initialized\r\n         */\r\n        initPlugins: {},\r\n\r\n        /**\r\n         * @description loaded language\r\n         */\r\n        lang: lang,\r\n\r\n        /**\r\n         * @description dialog element\r\n         */\r\n        dialogForm: null,\r\n\r\n        /**\r\n         * @description submenu element\r\n         */\r\n        submenu: null,\r\n\r\n        /**\r\n         * @description active button element in submenu\r\n         */\r\n        submenuActiveButton: null,\r\n\r\n        /**\r\n         * @description The elements array to be processed unvisible when the controllersOff function is executed (resizing, link modified button, table controller)\r\n         */\r\n        controllerArray: [],\r\n\r\n        /**\r\n         * @description The functions array to be executed when the controllersOff function is executed ex) init function of table plugin\r\n         */\r\n        controllerFunction: [],\r\n\r\n        /**\r\n         * @description An array of buttons whose class name is not \"code-view-enabled\"\r\n         */\r\n        codeViewDisabledButtons: context.element.toolbar.querySelectorAll('.sun-editor-id-toolbar button:not([class~=\"code-view-enabled\"])'),\r\n\r\n        /**\r\n         * @description Is inline mode?\r\n         * @private\r\n         */\r\n        _isInline: /inline/i.test(context.option.mode),\r\n\r\n        /**\r\n         * @description Is balloon mode?\r\n         * @private\r\n         */\r\n        _isBalloon: /balloon/i.test(context.option.mode),\r\n\r\n        /**\r\n         * @description Required value when using inline mode to sticky toolbar\r\n         * @private\r\n         */\r\n        _inlineToolbarAttr: {width: 0, height: 0, isShow: false},\r\n\r\n        /**\r\n         * @description An user event function when image uploaded success or remove image\r\n         * @private\r\n         */\r\n        _imageUpload: function (targetImgElement, index, isDelete) {\r\n            if (userFunction.onImageUpload) userFunction.onImageUpload(targetImgElement, index, isDelete);\r\n        },\r\n\r\n        /**\r\n         * @description Elements that need to change text or className for each selection change\r\n         * @property {Element} FORMAT - format button\r\n         * @property {Element} FONT - font family button\r\n         * @property {Element} SIZE - font size button\r\n         * @property {Element} B - bold button\r\n         * @property {Element} U - underline button\r\n         * @property {Element} I - italic button\r\n         * @property {Element} STRIKE - strike button\r\n         * @property {Element} SUB - subscript button\r\n         * @property {Element} SUP - superscript button\r\n         */\r\n        commandMap: {\r\n            FORMAT: context.tool.format,\r\n            FONT: context.tool.font,\r\n            SIZE: context.tool.fontSize,\r\n            B: context.tool.bold,\r\n            U: context.tool.underline,\r\n            I: context.tool.italic,\r\n            STRIKE: context.tool.strike,\r\n            SUB: context.tool.subscript,\r\n            SUP: context.tool.superscript\r\n        },\r\n\r\n        /**\r\n         * @description Variables used internally in editor operation\r\n         * @property {(Element|null)} selectionNode - Contains selection node\r\n         * @property {(Object|null)} range - The current range object\r\n         * @property {Boolean} wysiwygActive - The wysiwyg frame or code view state\r\n         * @property {Boolean} isFullScreen - State of full screen\r\n         * @property {Number} innerHeight_fullScreen - InnerHeight in editor when in full screen\r\n         * @property {Number} resizeClientY - Remember the vertical size of the editor before resizing the editor (Used when calculating during resize operation)\r\n         * @property {Number} tabSize - Indented size when tab button clicked (4)\r\n         * @property {Number} minResizingSize - Minimum size of editing area when resized (65)\r\n         * @property {Array} currentNodes -  An array of the current cursor's node structure\r\n         * @private\r\n         */\r\n        _variable: {\r\n            selectionNode: null,\r\n            range: null,\r\n            wysiwygActive: true,\r\n            isFullScreen: false,\r\n            innerHeight_fullScreen: 0,\r\n            resizeClientY: 0,\r\n            tabSize: 4,\r\n            minResizingSize: 65,\r\n            currentNodes: [],\r\n            _originCssText: context.element.topArea.style.cssText,\r\n            _bodyOverflow: '',\r\n            _editorAreaOriginCssText: '',\r\n            _wysiwygOriginCssText: '',\r\n            _codeOriginCssText: '',\r\n            _sticky: false,\r\n            _imagesInfo: [],\r\n            _imageIndex: 0\r\n        },\r\n\r\n        /**\r\n         * @description If the plugin is not added, add the plugin and call the 'add' function.\r\n         * If the plugin is added call callBack function.\r\n         * @param {String} pluginName - The name of the plugin to call\r\n         * @param {function} callBackFunction - Function to be executed immediately after module call\r\n         */\r\n        callPlugin: function (pluginName, callBackFunction) {\r\n            if (!this.plugins[pluginName]) {\r\n                throw Error('[SUNEDITOR.core.callPlugin.fail] The called plugin does not exist or is in an invalid format. (pluginName:\"' + pluginName + '\")');\r\n            } else if (!this.initPlugins[pluginName]){\r\n                this.plugins[pluginName].add(this, this.plugins[pluginName].buttonElement);\r\n                this.initPlugins[pluginName] = true;\r\n            }\r\n                \r\n            callBackFunction();\r\n        },\r\n\r\n        /**\r\n         * @description If the module is not added, add the module and call the 'add' function\r\n         * @param {Array} moduleArray - module object's Array [dialog, resizing]\r\n         */\r\n        addModule: function (moduleArray) {\r\n            let moduleName = '';\r\n            for (let i = 0, len = moduleArray.length; i < len; i++) {\r\n                moduleName = moduleArray[i].name;\r\n                if (!this.plugins[moduleName]) {\r\n                    this.plugins[moduleName] = util.copyObj(moduleArray[i]);\r\n                    this.plugins[moduleName].add(this);\r\n                }\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Enabled submenu\r\n         * @param {Element} element - Submenu element to call\r\n         */\r\n        submenuOn: function (element) {\r\n            const submenuName = element.getAttribute('data-command');\r\n            if (this.plugins[submenuName].on) this.plugins[submenuName].on.call(this);\r\n\r\n            this.submenu = element.nextElementSibling;\r\n            this.submenu.style.display = 'block';\r\n            util.addClass(element, 'on');\r\n            this.submenuActiveButton = element;\r\n\r\n            const overLeft = this.context.element.toolbar.offsetWidth - (element.parentElement.offsetLeft + this.submenu.offsetWidth);\r\n            if (overLeft < 0) this.submenu.style.left = overLeft + 'px';\r\n            else this.submenu.style.left = '1px';\r\n        },\r\n\r\n        /**\r\n         * @description Disable submenu\r\n         */\r\n        submenuOff: function () {\r\n            if (this.submenu) {\r\n                this.submenu.style.display = 'none';\r\n                this.submenu = null;\r\n                util.removeClass(this.submenuActiveButton, 'on');\r\n                this.submenuActiveButton = null;\r\n            }\r\n\r\n            this.controllersOff();\r\n        },\r\n\r\n        /**\r\n         * @description Disable controller in editor area (link button, image resize button)\r\n         */\r\n        controllersOff: function () {\r\n            const len = this.controllerArray.length;\r\n            const fLen = this.controllerFunction.length;\r\n\r\n            if (len > 0) {\r\n                for (let i = 0; i < len; i++) {\r\n                    this.controllerArray[i].style.display = 'none';\r\n                }\r\n                this.controllerArray = [];\r\n            }\r\n\r\n            if (fLen > 0) {\r\n                for (let i = 0; i < fLen; i++) {\r\n                    this.controllerFunction[i]();\r\n                }\r\n                this.controllerFunction = [];\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description javascript execCommand\r\n         * @param {String} command - javascript execCommand function property\r\n         * @param {Boolean} showDefaultUI - javascript execCommand function property\r\n         * @param {String} value - javascript execCommand function property\r\n         */\r\n        execCommand: function (command, showDefaultUI, value, styleWithCss) {\r\n            _d.execCommand(command, showDefaultUI, (command === 'formatBlock' ? '<' + value + '>' : value));\r\n        },\r\n\r\n        /**\r\n         * @description Focus to wysiwyg area\r\n         */\r\n        focus: function () {\r\n            if (context.element.wysiwyg.style.display === 'none') return;\r\n\r\n            const caption = util.getParentElement(this.getSelectionNode(), 'figcaption');\r\n            if (caption) {\r\n                caption.focus();\r\n            } else {\r\n                context.element.wysiwyg.focus();\r\n            }\r\n\r\n            this._setEditorRange();\r\n            event._findButtonEffectTag();\r\n        },\r\n\r\n        /**\r\n         * @description Saving the range object and the currently selected node of editor\r\n         * @private\r\n         */\r\n        _setEditorRange: function () {\r\n            const selection = _w.getSelection();\r\n            let range = null;\r\n\r\n            if (selection.rangeCount > 0) {\r\n                range = selection.getRangeAt(0);\r\n            }\r\n            else {\r\n                range = this._createDefaultRange();\r\n            }\r\n\r\n            this._variable.range = range;\r\n\r\n            if (range.collapsed) {\r\n                this.setSelectionNode(range.commonAncestorContainer);\r\n            } else {\r\n                this.setSelectionNode(selection.extentNode || selection.anchorNode);\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Return the range object of editor's first child node\r\n         * @returns {Object}\r\n         * @private\r\n         */\r\n        _createDefaultRange: function () {\r\n            const range = _d.createRange();\r\n            range.setStart(context.element.wysiwyg.firstChild, 0);\r\n            range.setEnd(context.element.wysiwyg.firstChild, 0);\r\n            return range;\r\n        },\r\n\r\n        /**\r\n         * @description Set current editor's range object\r\n         * @param {Element} startCon - The startContainer property of the selection object.\r\n         * @param {Number} startOff - The startOffset property of the selection object.\r\n         * @param {Element} endCon - The endContainer property of the selection object.\r\n         * @param {Element} endOff - The endOffset property of the selection object.\r\n         */\r\n        setRange: function (startCon, startOff, endCon, endOff) {\r\n            const range = _d.createRange();\r\n            range.setStart(startCon, startOff);\r\n            range.setEnd(endCon, endOff);\r\n\r\n            const selection = _w.getSelection();\r\n\r\n            if (selection.removeAllRanges) {\r\n                selection.removeAllRanges();\r\n            }\r\n\r\n            selection.addRange(range);\r\n            this._variable.range = range;\r\n            this._setEditorRange();\r\n        },\r\n\r\n        /**\r\n         * @description Get current editor's range object\r\n         * @returns {Object}\r\n         */\r\n        getRange: function () {\r\n            return this._variable.range || this._createDefaultRange();\r\n        },\r\n\r\n        /**\r\n         * @description Set the selected node. (Used by getSelectionNode function)\r\n         * @param {Node} node - node object\r\n         */\r\n        setSelectionNode: function (node) {\r\n            this._variable.selectionNode = node;\r\n        },\r\n\r\n        /**\r\n         * @description Get current select node\r\n         * @returns {Node}\r\n         */\r\n        getSelectionNode: function () {\r\n            if (this._variable.selectionNode) {\r\n                return this._variable.selectionNode;\r\n            }\r\n\r\n            return context.element.wysiwyg.firstChild;\r\n        },\r\n\r\n        /**\r\n         * @description Returns a \"formatElement\"(P, DIV, H[1-6], LI) array from the currently selected range.\r\n         * @returns {Array}\r\n         */\r\n        getSelectedFormatElements: function () {\r\n            let range = this.getRange();\r\n\r\n            if (util.isWysiwygDiv(range.startContainer)) {\r\n                const children = context.element.wysiwyg.children;\r\n                this.setRange(children[0], 0, children[children.length - 1], children[children.length - 1].textContent.length);\r\n                range = this.getRange();\r\n            }\r\n\r\n            const startCon = range.startContainer;\r\n            const endCon = range.endContainer;\r\n            const commonCon = range.commonAncestorContainer;\r\n            const rangeFormatElements = [];\r\n\r\n            if (!util.isWysiwygDiv(commonCon) && !util.isRangeFormatElement(commonCon)) return [util.getFormatElement(commonCon)];\r\n\r\n            // get line nodes\r\n            const lineNodes = util.getListChildren(commonCon, function (current) {\r\n                return util.isFormatElement(current);\r\n            });\r\n\r\n            if (startCon === endCon) return lineNodes[0];\r\n\r\n            let startLine = util.getFormatElement(startCon);\r\n            let endLine = util.getFormatElement(endCon);\r\n            let startIdx = 0;\r\n            let endIdx = 0;\r\n\r\n            for (let i = 0, len = lineNodes.length; i < len; i++) {\r\n                if (startLine === lineNodes[i]) {\r\n                    startIdx = i;\r\n                    continue;\r\n                }\r\n                if (endLine === lineNodes[i]) {\r\n                    endIdx = i;\r\n                    break;\r\n                }\r\n            }\r\n\r\n            for (let i = startIdx; i <= endIdx; i++) {\r\n                rangeFormatElements.push(lineNodes[i]);\r\n            }\r\n\r\n            return rangeFormatElements;\r\n        },\r\n\r\n        /**\r\n         * @description Returns a \"rangeFormatElement\"(blockquote, TABLE, TR, TD, OL, UL, PRE) array from the currently selected range.\r\n         * @returns {Array}\r\n         */\r\n        getSelectedRangeFormatElements: function () {\r\n            let range = this.getRange();\r\n\r\n            if (util.isWysiwygDiv(range.startContainer)) {\r\n                const children = context.element.wysiwyg.children;\r\n                this.setRange(children[0], 0, children[children.length - 1], children[children.length - 1].textContent.length);\r\n                range = this.getRange();\r\n            }\r\n\r\n            const startCon = range.startContainer;\r\n            const endCon = range.endContainer;\r\n            const commonCon = range.commonAncestorContainer;\r\n            const rangeFormatElements = [];\r\n\r\n            if (util.isRangeFormatElement(commonCon)) return [commonCon];\r\n            if (!util.isWysiwygDiv(commonCon)) {\r\n                const el = util.getRangeFormatElement(commonCon);\r\n                return el ? [el] : [];\r\n            }\r\n\r\n            // get range Elements\r\n            const rangeElements = util.getListChildren(commonCon, function (current) {\r\n                return util.isRangeFormatElement(current);\r\n            });\r\n\r\n            if (startCon === endCon) return rangeElements[0];\r\n\r\n            let startLine = util.getRangeFormatElement(startCon);\r\n            let endLine = util.getRangeFormatElement(endCon);\r\n            let startIdx = 0;\r\n            let endIdx = 0;\r\n\r\n            for (let i = 0, len = rangeElements.length; i < len; i++) {\r\n                if (startLine === rangeElements[i]) {\r\n                    startIdx = i;\r\n                    continue;\r\n                }\r\n                if (endLine === rangeElements[i]) {\r\n                    endIdx = i;\r\n                    break;\r\n                }\r\n            }\r\n\r\n            for (let i = startIdx; i <= endIdx; i++) {\r\n                if (rangeElements[i]) rangeFormatElements.push(rangeElements[i]);\r\n            }\r\n\r\n            return rangeFormatElements;\r\n        },\r\n\r\n        /**\r\n         * @description Determine if this offset is the edge offset of container\r\n         * @param {Object} container - The container property of the selection object.\r\n         * @param {Number} offset - The offset property of the selection object.\r\n         * @returns {Boolean}\r\n         */\r\n        isEdgePoint: function (container, offset) {\r\n            return (offset === 0) || (offset === container.nodeValue.length);\r\n        },\r\n\r\n        /**\r\n         * @description Show loading box\r\n         */\r\n        showLoading: function () {\r\n            context.element.loading.style.display = 'block';\r\n        },\r\n\r\n        /**\r\n         * @description Close loading box\r\n         */\r\n        closeLoading: function () {\r\n            context.element.loading.style.display = 'none';\r\n        },\r\n\r\n        /**\r\n         * @description Append format element to sibling node of argument element.\r\n         * If the \"formatNodeName\" argument value is present, the tag of that argument value is inserted,\r\n         * If not, the currently selected format tag is inserted.\r\n         * @param {Element} element - Insert as siblings of that element\r\n         * @param {String|null} formatNodeName - Node name to be inserted\r\n         * @returns {Element}\r\n         */\r\n        appendFormatTag: function (element, formatNodeName) {\r\n            const formatEl = util.getRangeFormatElement(element) || util.getFormatElement(element);\r\n            const currentFormatEl = util.getFormatElement(this.getSelectionNode());\r\n            const oFormatName = formatNodeName ? formatNodeName : util.isFormatElement(currentFormatEl) ? currentFormatEl.nodeName : 'P';\r\n            const oFormat = util.createElement(oFormatName);\r\n            oFormat.innerHTML = util.zeroWidthSpace;\r\n\r\n            if (util.isCell(formatEl)) formatEl.insertBefore(oFormat, element.nextElementSibling);\r\n            else formatEl.parentNode.insertBefore(oFormat, formatEl.nextElementSibling);\r\n\r\n            return oFormat;\r\n        },\r\n\r\n        /**\r\n         * @description Delete selected node and insert argument value node\r\n         * @param {Element} oNode - Node to be inserted\r\n         * @param {(Element|null)} rightNode - If the node exists, it is inserted after the node\r\n         */\r\n        insertNode: function (oNode, rightNode) {\r\n            const range = this.getRange();\r\n            let parentNode = null;\r\n\r\n            if (!rightNode) {\r\n                const startCon = range.startContainer;\r\n                const startOff = range.startOffset;\r\n                const endCon = range.endContainer;\r\n                const endOff = range.endOffset;\r\n                const commonCon = range.commonAncestorContainer;\r\n\r\n                parentNode = startCon;\r\n                if (startCon.nodeType === 3) {\r\n                    parentNode = startCon.parentNode;\r\n                }\r\n\r\n                /** No Select range node */\r\n                if (range.collapsed) {\r\n                    if (commonCon.nodeType === 3) {\r\n                        rightNode = commonCon.splitText(endOff);\r\n                    }\r\n                    else {\r\n                        if (parentNode.lastChild !== null && util.isBreak(parentNode.lastChild)) {\r\n                            parentNode.removeChild(parentNode.lastChild);\r\n                        }\r\n                        rightNode = null;\r\n                    }\r\n                }\r\n                /** Select range nodes */\r\n                else {\r\n                    const isSameContainer = startCon === endCon;\r\n\r\n                    if (isSameContainer) {\r\n                        if (this.isEdgePoint(endCon, endOff)) rightNode = endCon.nextSibling;\r\n                        else rightNode = endCon.splitText(endOff);\r\n\r\n                        let removeNode = startCon;\r\n                        if (!this.isEdgePoint(startCon, startOff)) removeNode = startCon.splitText(startOff);\r\n\r\n                        parentNode.removeChild(removeNode);\r\n                    }\r\n                    else {\r\n                        this.removeNode();\r\n                        parentNode = commonCon;\r\n                        rightNode = endCon;\r\n\r\n                        while (rightNode.parentNode !== commonCon) {\r\n                            rightNode = rightNode.parentNode;\r\n                        }\r\n                    }\r\n                }\r\n            }\r\n            else {\r\n                parentNode = rightNode.parentNode;\r\n                rightNode = rightNode.nextSibling;\r\n            }\r\n\r\n            try {\r\n                parentNode.insertBefore(oNode, rightNode);\r\n            } catch (e) {\r\n                parentNode.appendChild(oNode);\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Delete the currently selected node\r\n         */\r\n        removeNode: function () {\r\n            const range = this.getRange();\r\n\r\n            if (range.deleteContents) {\r\n                range.deleteContents();\r\n                return;\r\n            }\r\n\r\n            const startCon = range.startContainer;\r\n            const startOff = range.startOffset;\r\n            const endCon = range.endContainer;\r\n            const endOff = range.endOffset;\r\n            const commonCon = range.commonAncestorContainer;\r\n\r\n            let beforeNode = null;\r\n            let afterNode = null;\r\n\r\n            const childNodes = util.getListChildNodes(commonCon);\r\n            let startIndex = util.getArrayIndex(childNodes, startCon);\r\n            let endIndex = util.getArrayIndex(childNodes, endCon);\r\n\r\n            for (let i = startIndex + 1, startNode = startCon; i >= 0; i--) {\r\n                if (childNodes[i] === startNode.parentNode && childNodes[i].firstChild === startNode && startOff === 0) {\r\n                    startIndex = i;\r\n                    startNode = startNode.parentNode;\r\n                }\r\n            }\r\n\r\n            for (let i = endIndex - 1, endNode = endCon; i > startIndex; i--) {\r\n                if (childNodes[i] === endNode.parentNode && childNodes[i].nodeType === 1) {\r\n                    childNodes.splice(i, 1);\r\n                    endNode = endNode.parentNode;\r\n                    --endIndex;\r\n                }\r\n            }\r\n\r\n            for (let i = startIndex; i <= endIndex; i++) {\r\n                const item = childNodes[i];\r\n\r\n                if (item.length === 0 || (item.nodeType === 3 && item.data === undefined)) {\r\n                    util.removeItem(item);\r\n                    continue;\r\n                }\r\n\r\n                if (item === startCon) {\r\n                    if (startCon.nodeType === 1) {\r\n                        beforeNode = util.createTextNode(startCon.textContent);\r\n                    } else {\r\n                        beforeNode = util.createTextNode(startCon.substringData(0, startOff));\r\n                    }\r\n\r\n                    if (beforeNode.length > 0) {\r\n                        startCon.data = beforeNode.data;\r\n                    } else {\r\n                        util.removeItem(startCon);\r\n                    }\r\n\r\n                    continue;\r\n                }\r\n\r\n                if (item === endCon) {\r\n                    if (endCon.nodeType === 1) {\r\n                        afterNode = util.createTextNode(endCon.textContent);\r\n                    } else {\r\n                        afterNode = util.createTextNode(endCon.substringData(endOff, (endCon.length - endOff)));\r\n                    }\r\n\r\n                    if (afterNode.length > 0) {\r\n                        endCon.data = afterNode.data;\r\n                    } else {\r\n                        util.removeItem(endCon);\r\n                    }\r\n\r\n                    continue;\r\n                }\r\n\r\n                util.removeItem(item);\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description appended all selected format Element to the argument element and insert\r\n         * @param {Element} wrapTag - Element of wrap the arguments\r\n         */\r\n        wrapToTags: function (wrapTag) {\r\n            const range = this.getRange();\r\n            const rangeLines = this.getSelectedFormatElements();\r\n\r\n            if (!rangeLines) {\r\n                const inner = util.createElement(util.isCell(this.getSelectionNode()) ? 'DIV' : 'P');\r\n                inner.innerHTML = util.zeroWidthSpace;\r\n                wrapTag.appendChild(inner);\r\n                this.getSelectionNode().appendChild(wrapTag);\r\n                return;\r\n            }\r\n\r\n            let last  = rangeLines[rangeLines.length - 1];\r\n            let standTag, beforeTag, pElement;\r\n\r\n            if (util.isRangeFormatElement(last) || util.isFormatElement(last)) {\r\n                standTag = last;\r\n            } else {\r\n                standTag = util.getRangeFormatElement(last) || util.getFormatElement(last);\r\n            }\r\n\r\n            if (util.isCell(standTag)) {\r\n                beforeTag = null;\r\n                pElement = standTag;\r\n            } else {\r\n                beforeTag = standTag.nextSibling;\r\n                pElement = standTag.parentNode;\r\n            }\r\n\r\n            let listParent = null;\r\n            let line = null;\r\n            let prevNodeName = '';\r\n            \r\n            for (let i = 0, len = rangeLines.length; i < len; i++) {\r\n                line = rangeLines[i];\r\n\r\n                if (/^LI$/i.test(line.nodeName)) {\r\n                    if (listParent === null || !/^LI$/i.test(prevNodeName)) {\r\n                        listParent = util.createElement(line.parentNode.nodeName);\r\n                    }\r\n\r\n                    listParent.appendChild(line);\r\n                    if (i === len - 1 || !/^LI$/i.test(rangeLines[i + 1].nodeName)) wrapTag.appendChild(listParent);\r\n                }\r\n                else {\r\n                    wrapTag.appendChild(line);\r\n                }\r\n\r\n                prevNodeName = line.nodeName;\r\n            }\r\n\r\n            pElement.insertBefore(wrapTag, beforeTag);\r\n            if (!range.collapsed && (util.isRangeFormatElement(range.startContainer) || util.isRangeFormatElement(range.endContainer))) util.removeEmptyNode(pElement);\r\n        },\r\n\r\n        /**\r\n         * @description Add the node received as an argument value to the selected area.\r\n         * 1. When there is the same css value node in the selection area, the tag is stripped.\r\n         * 2. If there is another css value other thanCss attribute values received as arguments on the node, removed only Css attribute values received as arguments\r\n         * 3. If you pass an element whose node name is \"removenode\" as an argument value, it performs a type removal operation. ex) nodeChange(document.createElement('removenode'))\r\n         * @param {Element} appendNode - The dom that will wrap the selected text area\r\n         * @param {Array} checkCSSPropertyArray - The css attribute name Array to check (['font-size'], ['font-family', 'background-color', 'border']...])\r\n         */\r\n        nodeChange: function (appendNode, checkCSSPropertyArray) {\r\n            const range = this.getRange();\r\n            const isRemoveFormat = /removenode/i.test(appendNode.nodeName);\r\n            let tempCon, tempOffset, tempChild, tempArray;\r\n\r\n            /* checked same style property */\r\n            if (range.startContainer === range.endContainer) {\r\n                let sNode = range.startContainer;\r\n                if (isRemoveFormat) {\r\n                    if (util.getFormatElement(sNode) === sNode.parentNode) return;\r\n                } else {\r\n                    let checkCnt = 0;\r\n\r\n                    for (let i = 0; i < checkCSSPropertyArray.length; i++) {\r\n                        while(!util.isFormatElement(sNode) && !util.isWysiwygDiv(sNode)) {\r\n                            if (sNode.nodeType === 1 && sNode.style[checkCSSPropertyArray[i]] === appendNode.style[checkCSSPropertyArray[i]]) {\r\n                                checkCnt++;\r\n                            }\r\n                            sNode = sNode.parentNode;\r\n                        }\r\n                    }\r\n    \r\n                    if (checkCnt >= checkCSSPropertyArray.length) return;\r\n                }\r\n            }\r\n\r\n            /* find text node */\r\n            tempCon = range.startContainer;\r\n            tempOffset = range.startOffset;\r\n\r\n            if (tempCon.nodeType === 1 && tempCon.childNodes.length > 0) {\r\n                while (tempCon && !util.isBreak(tempCon) && tempCon.nodeType === 1) {\r\n                    tempArray = [];\r\n                    tempChild = tempCon.childNodes;\r\n                    for (let i = 0, len = tempChild.length; i < len; i++) {\r\n                        tempArray.push(tempChild[i]);\r\n                    }\r\n                    tempCon = tempArray[tempOffset] || tempCon.nextElementSibling || tempCon.nextSibling;\r\n                    tempOffset = 0;\r\n                }\r\n            }\r\n\r\n            const startCon = tempCon;\r\n            const startOff = tempOffset;\r\n\r\n            tempCon = range.endContainer;\r\n            tempOffset = range.endOffset;\r\n            if (tempCon.nodeType === 1 && tempCon.childNodes.length > 0) {\r\n                while (tempCon && !util.isBreak(tempCon) && tempCon.nodeType === 1) {\r\n                    tempArray = [];\r\n                    tempChild = tempCon.childNodes;\r\n                    for (let i = 0, len = tempChild.length; i < len; i++) {\r\n                        tempArray.push(tempChild[i]);\r\n                    }\r\n                    tempCon = tempArray[tempOffset - 1] || tempArray[0] || tempCon.previousElementSibling || tempCon.previousSibling || startCon;\r\n                }\r\n                tempOffset = tempCon.textContent.length;\r\n            }\r\n\r\n            const endCon = tempCon;\r\n            const endOff = tempOffset;\r\n            const commonCon = range.commonAncestorContainer;\r\n            const newNodeName = appendNode.nodeName;\r\n\r\n            let start = {}, end = {};\r\n            let newNode, regExp;\r\n\r\n            if (checkCSSPropertyArray) {\r\n                regExp = '(?:;|^|\\\\s)(?:' + checkCSSPropertyArray[0];\r\n                for (let i = 1; i < checkCSSPropertyArray.length; i++) {\r\n                    regExp += '|' + checkCSSPropertyArray[i];\r\n                }\r\n                regExp += ')\\\\s*:[^;]*\\\\s*(?:;|$)';\r\n                regExp = new RegExp(regExp, 'ig');\r\n            }\r\n\r\n            /** tag check function*/\r\n            const checkCss = function (vNode) {\r\n                if (isRemoveFormat || vNode.nodeType === 3 || util.isBreak(vNode)) return true;\r\n\r\n                let style = '';\r\n                if (regExp && vNode.style.cssText.length > 0) {\r\n                    style = vNode.style.cssText.replace(regExp, '').trim();\r\n                }\r\n\r\n                if (style.length > 0 || vNode.nodeName !== newNodeName) {\r\n                    if (vNode.style.cssText.length > 0) vNode.style.cssText = style;\r\n                    return true;\r\n                }\r\n\r\n                return false;\r\n            };\r\n\r\n            /** one line */\r\n            if (!util.isWysiwygDiv(commonCon) && !util.isRangeFormatElement(commonCon)) {\r\n                newNode = appendNode.cloneNode(false);\r\n                const newRange = this._nodeChange_oneLine(util.getFormatElement(commonCon), newNode, checkCss, startCon, startOff, endCon, endOff, isRemoveFormat, range.collapsed);\r\n\r\n                start.container = newRange.startContainer;\r\n                start.offset = newRange.startOffset;\r\n                end.container = newRange.endContainer;\r\n                end.offset = newRange.endOffset;\r\n            }\r\n            /** multi line */\r\n            else {\r\n                // get line nodes\r\n                const lineNodes = this.getSelectedFormatElements();\r\n                const endLength = lineNodes.length - 1;\r\n\r\n                // startCon\r\n                newNode = appendNode.cloneNode(false);\r\n                start = this._nodeChange_startLine(lineNodes[0], newNode, checkCss, startCon, startOff, isRemoveFormat);\r\n\r\n                // mid\r\n                for (let i = 1; i < endLength; i++) {\r\n                    newNode = appendNode.cloneNode(false);\r\n                    this._nodeChange_middleLine(lineNodes[i], newNode, checkCss, isRemoveFormat);\r\n                }\r\n\r\n                // endCon\r\n                if (endLength > 0) {\r\n                    newNode = appendNode.cloneNode(false);\r\n                    end = this._nodeChange_endLine(lineNodes[endLength], newNode, checkCss, endCon, endOff, isRemoveFormat);\r\n                } else {\r\n                    end = start;\r\n                }\r\n            }\r\n\r\n            // set range\r\n            this.setRange(start.container, start.offset, end.container, end.offset);\r\n        },\r\n\r\n        /**\r\n         * @description wraps text nodes of line selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Element} startCon - The startContainer property of the selection object.\r\n         * @param {Number} startOff - The startOffset property of the selection object.\r\n         * @param {Element} endCon - The endContainer property of the selection object.\r\n         * @param {Number} endOff - The endOffset property of the selection object.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @param {Boolean} collapsed - range.collapsed\r\n         * @returns {{startContainer: *, startOffset: *, endContainer: *, endOffset: *}}\r\n         * @private\r\n         */\r\n        _nodeChange_oneLine: function (element, newInnerNode, validation, startCon, startOff, endCon, endOff, isRemoveFormat, collapsed) {\r\n            const el = element;\r\n            const pNode = element.cloneNode(false);\r\n            const isSameNode = startCon === endCon;\r\n            let startContainer = startCon;\r\n            let startOffset = startOff;\r\n            let endContainer = endCon;\r\n            let endOffset = endOff;\r\n            let startPass = false;\r\n            let endPass = false;\r\n            let pCurrent, newNode, appendNode, cssText;\r\n\r\n            function checkCss (vNode) {\r\n                const regExp = new RegExp('(?:;|^|\\\\s)(?:' + cssText + 'null)\\\\s*:[^;]*\\\\s*(?:;|$)', 'ig');\r\n                let style = '';\r\n\r\n                if (regExp && vNode.style.cssText.length > 0) {\r\n                    style = regExp.test(vNode.style.cssText);\r\n                }\r\n            \r\n                return !style;\r\n            }\r\n\r\n            (function recursionFunc(current, node) {\r\n                const childNodes = current.childNodes;\r\n\r\n                for (let i = 0, len = childNodes.length; i < len; i++) {\r\n                    let child = childNodes[i];\r\n                    let coverNode = node;\r\n                    let cloneNode;\r\n\r\n                    // startContainer\r\n                    if (!startPass && child === startContainer) {\r\n                        const prevNode = util.createTextNode(startContainer.nodeType === 1 ? '' : startContainer.substringData(0, startOffset));\r\n                        const textNode = util.createTextNode(startContainer.nodeType === 1 ? '' : startContainer.substringData(startOffset, (endOffset > startOffset ? endOffset - startOffset : startOffset - endOffset)));\r\n\r\n                        if (prevNode.data.length > 0) {\r\n                            node.appendChild(prevNode);\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        cssText = '';\r\n                        while (newNode !== pNode && newNode !== el && newNode !== null) {\r\n                            if (validation(newNode) && newNode.nodeType === 1 && checkCss(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                                cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || textNode;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        newInnerNode.appendChild(childNode);\r\n                        pNode.appendChild(newInnerNode);\r\n\r\n                        startContainer = textNode;\r\n                        startOffset = 0;\r\n                        startPass = true;\r\n\r\n                        if (newNode !== textNode) newNode.appendChild(startContainer);\r\n                        if (!isSameNode) continue;\r\n                    }\r\n\r\n                    // endContainer\r\n                    if (!endPass && child === endContainer) {\r\n                        const afterNode = util.createTextNode(endContainer.nodeType === 1 ? '' : endContainer.substringData(endOffset, (endContainer.length - endOffset)));\r\n                        const textNode = util.createTextNode(isSameNode || endContainer.nodeType === 1 ? '' : endContainer.substringData(0, endOffset));\r\n\r\n                        if (afterNode.data.length > 0) {\r\n                            newNode = child;\r\n                            cssText = '';\r\n                            pCurrent = [];\r\n                            while (newNode !== pNode && newNode !== el && newNode !== null) {\r\n                                if (newNode.nodeType === 1 && checkCss(newNode)) {\r\n                                    pCurrent.push(newNode.cloneNode(false));\r\n                                    cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                                }\r\n                                newNode = newNode.parentNode;\r\n                            }\r\n\r\n                            cloneNode = appendNode = newNode = pCurrent.pop() || afterNode;\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                                appendNode = newNode;\r\n                            }\r\n\r\n                            pNode.appendChild(cloneNode);\r\n                            newNode.textContent = afterNode.data;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        cssText = '';\r\n                        while (newNode !== pNode && newNode !== el && newNode !== null) {\r\n                            if (validation(newNode) && newNode.nodeType === 1 && checkCss(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                                cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || textNode;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        newInnerNode.appendChild(childNode);\r\n\r\n                        endContainer = textNode;\r\n                        endOffset = textNode.data.length;\r\n                        endPass = true;\r\n\r\n                        if (!isRemoveFormat && collapsed) {\r\n                            newInnerNode = textNode;\r\n                            textNode.textContent = util.zeroWidthSpace;\r\n                        }\r\n\r\n                        if (newNode !== textNode) newNode.appendChild(endContainer);\r\n                        continue;\r\n                    }\r\n\r\n                    // other\r\n                    if (startPass) {\r\n                        if (child.nodeType === 1 && !util.isBreak(child)) {\r\n                            recursionFunc(child, child);\r\n                            continue;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        cssText = '';\r\n                        while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\r\n                            if (newNode.nodeType === 1 && !util.isBreak(child) && (endPass || validation(newNode)) && checkCss(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                                cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || child;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n                        \r\n                        if (childNode === child) {\r\n                            if (!endPass) node = newInnerNode;\r\n                            else node = pNode;\r\n                        } else if (endPass) {\r\n                            pNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            newInnerNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        }\r\n                    }\r\n\r\n                    cloneNode = child.cloneNode(false);\r\n                    node.appendChild(cloneNode);\r\n                    if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            if (isRemoveFormat) {\r\n                startContainer = util.createTextNode(collapsed ? util.zeroWidthSpace : newInnerNode.textContent);\r\n                pNode.insertBefore(startContainer, newInnerNode);\r\n                pNode.removeChild(newInnerNode);\r\n                if (collapsed) startOffset = 1;\r\n            } else if (collapsed) {\r\n                startContainer = endContainer = newInnerNode;\r\n                startOffset = 1;\r\n                endOffset = 1;\r\n            }\r\n\r\n            util.removeEmptyNode(pNode);\r\n            element.parentNode.insertBefore(pNode, element);\r\n            util.removeItem(element);\r\n\r\n            return {\r\n                startContainer: startContainer,\r\n                startOffset: startOffset,\r\n                endContainer: isRemoveFormat || !endContainer.textContent ? startContainer : endContainer,\r\n                endOffset: isRemoveFormat || !endContainer.textContent ? startContainer.textContent.length : endOffset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description wraps mid lines selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @private\r\n         */\r\n        _nodeChange_middleLine: function (element, newInnerNode, validation, isRemoveFormat) {\r\n            if (isRemoveFormat) {\r\n                newInnerNode = util.createTextNode(element.textContent ? element.textContent : util.zeroWidthSpace);\r\n            } else {\r\n                (function recursionFunc(current, node) {\r\n                    const childNodes = current.childNodes;\r\n    \r\n                    for (let i = 0, len = childNodes.length; i < len; i++) {\r\n                        let child = childNodes[i];\r\n                        let coverNode = node;\r\n                        if (validation(child)) {\r\n                            let cloneNode = child.cloneNode(false);\r\n                            node.appendChild(cloneNode);\r\n                            if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n                        }\r\n                        recursionFunc(child, coverNode);\r\n                    }\r\n                })(element, newInnerNode);\r\n            }\r\n\r\n            element.innerHTML = '';\r\n            element.appendChild(newInnerNode);\r\n        },\r\n\r\n        /**\r\n         * @description wraps first line selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Element} startCon - The startContainer property of the selection object.\r\n         * @param {Number} startOff - The startOffset property of the selection object.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _nodeChange_startLine: function (element, newInnerNode, validation, startCon, startOff, isRemoveFormat) {\r\n            const el = element;\r\n            const pNode = element.cloneNode(false);\r\n\r\n            let container = startCon;\r\n            let offset = startOff;\r\n            let passNode = false;\r\n            let pCurrent, newNode, appendNode;\r\n\r\n            (function recursionFunc(current, node) {\r\n                const childNodes = current.childNodes;\r\n                for (let i = 0, len = childNodes.length; i < len; i++) {\r\n                    const child = childNodes[i];\r\n                    let coverNode = node;\r\n\r\n                    if (passNode && !util.isBreak(child)) {\r\n                        if (child.nodeType === 1) {\r\n                            recursionFunc(child, child);\r\n                            continue;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\r\n                            if (newNode.nodeType === 1 && validation(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        if (pCurrent.length > 0) {\r\n                            const childNode = pCurrent.pop();\r\n                            appendNode = newNode = childNode;\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                                appendNode = newNode;\r\n                            }\r\n                            newInnerNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n                    }\r\n\r\n                    // startContainer\r\n                    if (!passNode && child === container) {\r\n                        const prevNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(0, offset));\r\n                        const textNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(offset, (container.length - offset)));\r\n\r\n                        if (prevNode.data.length > 0) {\r\n                            node.appendChild(prevNode);\r\n                        }\r\n\r\n                        newNode = node;\r\n                        pCurrent = [];\r\n                        while (newNode !== pNode && newNode !== null) {\r\n                            if (newNode.nodeType === 1 && validation(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || node;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        if (childNode !== node) {\r\n                            newInnerNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n\r\n                        if (util.isBreak(child)) newInnerNode.appendChild(child.cloneNode(false));\r\n\r\n                        pNode.appendChild(newInnerNode);\r\n                        container = textNode;\r\n                        offset = 0;\r\n                        passNode = true;\r\n\r\n                        node.appendChild(container);\r\n                        continue;\r\n                    }\r\n\r\n                    if (!passNode || validation(child)) {\r\n                        const cloneNode = child.cloneNode(false);\r\n                        node.appendChild(cloneNode);\r\n                        if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            if (isRemoveFormat) {\r\n                container = util.createTextNode(newInnerNode.textContent);\r\n                pNode.insertBefore(container, newInnerNode);\r\n                pNode.removeChild(newInnerNode);\r\n            }\r\n\r\n            if (!isRemoveFormat && pNode.children.length === 0) {\r\n                if (element.childNodes) {\r\n                    container = element.childNodes[0];\r\n                } else {\r\n                    container = util.createTextNode(util.zeroWidthSpace);\r\n                    element.appendChild(container);\r\n                }\r\n            } else {\r\n                util.removeEmptyNode(pNode);\r\n                element.parentNode.insertBefore(pNode, element);\r\n                util.removeItem(element);\r\n            }\r\n\r\n            return {\r\n                container: container,\r\n                offset: offset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description wraps last line selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Element} endCon - The endContainer property of the selection object.\r\n         * @param {Number} endOff - The endOffset property of the selection object.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _nodeChange_endLine: function (element, newInnerNode, validation, endCon, endOff, isRemoveFormat) {\r\n            const el = element;\r\n            const pNode = element.cloneNode(false);\r\n\r\n            let container = endCon;\r\n            let offset = endOff;\r\n            let passNode = false;\r\n            let pCurrent, newNode, appendNode;\r\n\r\n            (function recursionFunc(current, node) {\r\n                const childNodes = current.childNodes;\r\n                for (let i = childNodes.length -1; 0 <= i; i--) {\r\n                    const child = childNodes[i];\r\n                    let coverNode = node;\r\n\r\n                    if (passNode && !util.isBreak(child)) {\r\n                        if (child.nodeType === 1) {\r\n                            recursionFunc(child, child);\r\n                            continue;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\r\n                            if (validation(newNode) && newNode.nodeType === 1) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        if (pCurrent.length > 0) {\r\n                            const childNode = pCurrent.pop();\r\n                            appendNode = newNode = childNode;\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                                appendNode = newNode;\r\n                            }\r\n                            newInnerNode.insertBefore(childNode, newInnerNode.firstChild);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n                    }\r\n\r\n                    // endContainer\r\n                    if (!passNode && child === container) {\r\n                        const afterNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(offset, (container.length - offset)));\r\n                        const textNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(0, offset));\r\n\r\n                        if (afterNode.data.length > 0) {\r\n                            node.insertBefore(afterNode, node.firstChild);\r\n                        }\r\n\r\n                        newNode = node;\r\n                        pCurrent = [];\r\n                        while (newNode !== pNode && newNode !== null) {\r\n                            if (validation(newNode) && newNode.nodeType === 1) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || node;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        if (childNode !== node) {\r\n                            newInnerNode.insertBefore(childNode, newInnerNode.firstChild);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n\r\n                        if (util.isBreak(child)) newInnerNode.appendChild(child.cloneNode(false));\r\n\r\n                        pNode.insertBefore(newInnerNode, pNode.firstChild);\r\n                        container = textNode;\r\n                        offset = textNode.data.length;\r\n                        passNode = true;\r\n\r\n                        node.insertBefore(container, node.firstChild);\r\n                        continue;\r\n                    }\r\n\r\n                    if (!passNode || validation(child)) {\r\n                        const cloneNode = child.cloneNode(false);\r\n                        node.insertBefore(cloneNode, node.firstChild);\r\n                        if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            if (isRemoveFormat) {\r\n                container = util.createTextNode(newInnerNode.textContent);\r\n                offset = container.textContent.length;\r\n                pNode.insertBefore(container, newInnerNode);\r\n                pNode.removeChild(newInnerNode);\r\n            }\r\n\r\n            if (!isRemoveFormat && pNode.childNodes.length === 0) {\r\n                if (element.childNodes) {\r\n                    container = element.childNodes[0];\r\n                } else {\r\n                    container = util.createTextNode(util.zeroWidthSpace);\r\n                    element.appendChild(container);\r\n                }\r\n            } else {\r\n                util.removeEmptyNode(pNode);\r\n                element.parentNode.insertBefore(pNode, element);\r\n                util.removeItem(element);\r\n            }\r\n\r\n            return {\r\n                container: container,\r\n                offset: offset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description Execute command of command button(All Buttons except submenu and dialog)\r\n         * (redo, undo, bold, underline, italic, strikethrough, subscript, superscript, removeFormat, indent, outdent, fullscreen, showBlocks, codeview, preview, print)\r\n         * @param {Element} target - The element of command button\r\n         * @param {String} command - Property of command button (data-value)\r\n         */\r\n        commandHandler: function (target, command) {\r\n            switch (command) {\r\n                case 'codeView':\r\n                    this.controllersOff();\r\n                    this.toggleCodeView();\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'fullScreen':\r\n                    this.controllersOff();\r\n                    this.toggleFullScreen(target);\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'indent':\r\n                case 'outdent':\r\n                    this.indent(command);\r\n                    break;\r\n                case 'redo':\r\n                case 'undo':\r\n                    this.execCommand(command, false, null);\r\n                    break;\r\n                case 'removeFormat':\r\n                    this.removeFormat();\r\n                    break;\r\n                case 'preview':\r\n                case 'print':\r\n                    this.openWindowContents(command);\r\n                    break;\r\n                case 'showBlocks':\r\n                    this.toggleDisplayBlocks();\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'subscript':\r\n                    if (util.hasClass(context.tool.superscript, 'on')) {\r\n                        this.execCommand('superscript', false, null);\r\n                        util.removeClass(context.tool.superscript, 'on');\r\n                    }\r\n                    this.execCommand(command, false, null);\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'superscript':\r\n                    if (util.hasClass(context.tool.subscript, 'on')) {\r\n                        this.execCommand('subscript', false, null);\r\n                        util.removeClass(context.tool.subscript, 'on');\r\n                    }\r\n                    this.execCommand(command, false, null);\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                default : // 'bold', 'underline', 'italic', 'strike'\r\n                    this.execCommand(command, false, target.getAttribute('data-value'));\r\n                    util.toggleClass(target, 'on');\r\n            }\r\n\r\n            this.focus();\r\n        },\r\n\r\n        /**\r\n         * @description Remove format of the currently selected range (IE, Edge not working)\r\n         */\r\n        removeFormat: function () {\r\n            this.nodeChange(util.createElement('REMOVENODE'));\r\n        },\r\n\r\n        /**\r\n         * @description This method implements indentation to selected range.\r\n         * Setted \"margin-left\" to \"25px\" in the top \"P\" tag of the parameter node.\r\n         * @param command {String} - Separator (\"indent\" or \"outdent\")\r\n         */\r\n        indent: function (command) {\r\n            const rangeLines = this.getSelectedFormatElements();\r\n            let p, margin;\r\n\r\n            for (let i = 0, len = rangeLines.length; i < len; i++) {\r\n                p = rangeLines[i];\r\n                margin = /\\d+/.test(p.style.marginLeft) ? p.style.marginLeft.match(/\\d+/)[0] * 1 : 0;\r\n\r\n                if ('indent' === command) {\r\n                    margin += 25;\r\n                } else {\r\n                    margin -= 25;\r\n                }\r\n    \r\n                p.style.marginLeft = (margin < 0 ? 0 : margin) + 'px';\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Add or remove the class name of \"body\" so that the code block is visible\r\n         */\r\n        toggleDisplayBlocks: function () {\r\n            util.toggleClass(context.element.wysiwyg, 'sun-editor-show-block');\r\n        },\r\n\r\n        /**\r\n         * @description Changes to code view or wysiwyg view\r\n         */\r\n        toggleCodeView: function () {\r\n            const wysiwygActive = this._variable.wysiwygActive;\r\n            const disButtons = this.codeViewDisabledButtons;\r\n            for (let i = 0, len = disButtons.length; i < len; i++) {\r\n                disButtons[i].disabled = wysiwygActive;\r\n            }\r\n\r\n            if (!wysiwygActive) {\r\n                const code_html = context.element.code.value.trim();\r\n                context.element.wysiwyg.innerHTML = code_html.length > 0 ? util.convertContentsForEditor(code_html) : '<p>' + util.zeroWidthSpace + '</p>';\r\n                context.element.wysiwyg.scrollTop = 0;\r\n                context.element.code.style.display = 'none';\r\n                context.element.wysiwyg.style.display = 'block';\r\n                if (context.option.height === 'auto') context.element.code.style.height = '0px';\r\n                this._variable.wysiwygActive = true;\r\n                this.focus();\r\n            }\r\n            else {\r\n                context.element.code.value = util.convertHTMLForCodeView(context.element.wysiwyg.innerHTML.trim());\r\n                context.element.wysiwyg.style.display = 'none';\r\n                context.element.code.style.display = 'block';\r\n                if (context.option.height === 'auto') context.element.code.style.height = context.element.code.scrollHeight > 0 ? (context.element.code.scrollHeight + 'px') : 'auto';\r\n                this._variable.wysiwygActive = false;\r\n                context.element.code.focus();\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Changes to full screen or default screen\r\n         * @param {Element} element - full screen button\r\n         */\r\n        toggleFullScreen: function (element) {\r\n            if (!this._variable.isFullScreen) {\r\n                this._variable.isFullScreen = true;\r\n\r\n                context.element.topArea.style.position = 'fixed';\r\n                context.element.topArea.style.top = '0';\r\n                context.element.topArea.style.left = '0';\r\n                context.element.topArea.style.width = '100%';\r\n                context.element.topArea.style.height = '100%';\r\n                context.element.topArea.style.zIndex = '2147483647';\r\n\r\n                this._variable._bodyOverflow = _d.body.style.overflow;\r\n                _d.body.style.overflow = 'hidden';\r\n\r\n                this._variable._editorAreaOriginCssText = context.element.editorArea.style.cssText;\r\n                this._variable._wysiwygOriginCssText = context.element.wysiwyg.style.cssText;\r\n                this._variable._codeOriginCssText = context.element.code.style.cssText;\r\n\r\n                context.element.editorArea.style.cssText = context.element.toolbar.style.cssText = context.element.wysiwyg.style.cssText = context.element.code.style.cssText = '';\r\n                context.element.toolbar.style.width = context.element.wysiwyg.style.height = context.element.code.style.height = '100%';\r\n                context.element.toolbar.style.position = 'relative';\r\n\r\n                this._variable.innerHeight_fullScreen = (_w.innerHeight - context.element.toolbar.offsetHeight);\r\n                context.element.editorArea.style.height = this._variable.innerHeight_fullScreen + 'px';\r\n\r\n                util.removeClass(element.firstElementChild, 'icon-expansion');\r\n                util.addClass(element.firstElementChild, 'icon-reduction');\r\n            }\r\n            else {\r\n                this._variable.isFullScreen = false;\r\n\r\n                context.element.code.style.cssText = this._variable._codeOriginCssText;\r\n                context.element.wysiwyg.style.cssText = this._variable._wysiwygOriginCssText;\r\n                context.element.toolbar.style.cssText = '';\r\n                context.element.editorArea.style.cssText = this._variable._editorAreaOriginCssText;\r\n                context.element.topArea.style.cssText = this._variable._originCssText;\r\n                _d.body.style.overflow = this._variable._bodyOverflow;\r\n\r\n                if (context.option.stickyToolbar > -1) {\r\n                    util.removeClass(context.element.toolbar, 'sun-editor-sticky');\r\n                    event.onScroll_window();\r\n                }\r\n\r\n                util.removeClass(element.firstElementChild, 'icon-reduction');\r\n                util.addClass(element.firstElementChild, 'icon-expansion');\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Open the preview window or open the print window\r\n         * @param {String} mode - 'preview' or 'print'\r\n         */\r\n        openWindowContents: function (mode) {\r\n            const isPrint = mode === 'print';\r\n            const windowObject = _w.open('', '_blank');\r\n            windowObject.mimeType = 'text/html';\r\n            windowObject.document.write('' +\r\n                '<!doctype html><html>' +\r\n                '<head>' +\r\n                '<meta charset=\"utf-8\" />' +\r\n                '<meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">' +\r\n                '<title>' + (isPrint ? lang.toolbar.print : lang.toolbar.preview) + '</title>' +\r\n                '<link rel=\"stylesheet\" type=\"text/css\" href=\"' + util.getIncludePath(['suneditor-contents', 'suneditor'], 'css') + '\">' +\r\n                '</head>' +\r\n                '<body>' +\r\n                '<div class=\"sun-editor-editable\" style=\"width:' + context.element.wysiwyg.offsetWidth + 'px; margin:auto;\">' +\r\n                this.getContents() + '</div>' +\r\n                (isPrint ? '<script>_w.print();</script>' : '') + '</body>' +\r\n                '</html>');\r\n        },\r\n\r\n        /**\r\n         * @description Gets the current contents\r\n         * @returns {Object}\r\n         */\r\n        getContents: function () {\r\n            let contents = '';\r\n\r\n            if (context.element.wysiwyg.innerText.trim().length === 0) return contents;\r\n\r\n            if (editor._variable.wysiwygActive) {\r\n                contents = context.element.wysiwyg.innerHTML;\r\n            } else {\r\n                contents = util.convertContentsForEditor(context.element.code.value);\r\n            }\r\n\r\n            const renderHTML = util.createElement('DIV');\r\n            renderHTML.innerHTML = contents;\r\n\r\n            const figcaptions = util.getListChildren(renderHTML, function (current) {\r\n                return /FIGCAPTION/i.test(current.nodeName);\r\n            });\r\n\r\n            for (let i = 0, len = figcaptions.length; i < len; i++) {\r\n                figcaptions[i].outerHTML = figcaptions[i].outerHTML.replace(/(?!^<figcaption\\s+)(contenteditable=\"([a-z]+|\\s*)\")\\s*(?=[^>]*>)/i, '');\r\n            }\r\n\r\n            return renderHTML.innerHTML;\r\n        }\r\n    };\r\n\r\n    /**\r\n     * @description event function\r\n     */\r\n    const event = {\r\n        _shortcutKeyCode: {\r\n            66: ['bold', 'B'],\r\n            83: ['strikethrough', 'STRIKE'],\r\n            85: ['underline', 'U'],\r\n            73: ['italic', 'I'],\r\n            89: ['redo'],\r\n            90: ['undo'],\r\n            219: ['outdent'],\r\n            221: ['indent']\r\n        },\r\n\r\n        _directionKeyKeyCode: new RegExp('^(?:8|13|32|46|33|34|35|36|37|38|39|40|98|100|102|104)$'),\r\n\r\n        _changeButtonClassTagCheck: new RegExp('^(?:B|U|I|STRIKE|SUB|SUP)$'),\r\n\r\n        _findButtonEffectTag: function () {\r\n            const commandMap = editor.commandMap;\r\n            const classOnCheck = this._changeButtonClassTagCheck;\r\n            const commandMapNodes = [];\r\n            const currentNodes = [];\r\n\r\n            let findFormat = true, findFont = true, findSize = true, findA = true;\r\n            let findB = true, findI = true, findU = true, findS = true;\r\n            let cssText = '', nodeName = '';\r\n\r\n            for (let selectionParent = editor.getSelectionNode(); !util.isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\r\n                if (!selectionParent) break;\r\n                if (selectionParent.nodeType !== 1) continue;\r\n                nodeName = selectionParent.nodeName.toUpperCase();\r\n                currentNodes.push(nodeName);\r\n\r\n                /** Format */\r\n                if (findFormat && util.isFormatElement(selectionParent)) {\r\n                    commandMapNodes.push('FORMAT');\r\n                    util.changeTxt(commandMap.FORMAT, nodeName);\r\n                    findFormat = false;\r\n                    continue;\r\n                }\r\n\r\n                /** Font */\r\n                if (findFont && (selectionParent.style.fontFamily.length > 0 || (selectionParent.face && selectionParent.face.length > 0))) {\r\n                    commandMapNodes.push('FONT');\r\n                    const selectFont = (selectionParent.style.fontFamily || selectionParent.face || lang.toolbar.font).replace(/[\"']/g,'');\r\n                    util.changeTxt(commandMap.FONT, selectFont);\r\n                    findFont = false;\r\n                }\r\n\r\n                /** A */\r\n                if (findA && /^A$/.test(nodeName) && selectionParent.getAttribute('data-image-link') === null) {\r\n                    if (!context.link || editor.controllerArray[0] !== context.link.linkBtn) {\r\n                        editor.callPlugin('link', function () {\r\n                            editor.plugins.link.call_controller_linkButton.call(editor, selectionParent);\r\n                        });\r\n                    }\r\n                    findA = false;\r\n                } else if (findA && context.link && editor.controllerArray[0] === context.link.linkBtn) {\r\n                    editor.controllersOff();\r\n                }\r\n\r\n                /** SPAN */\r\n                if (findSize && /^SPAN$/.test(nodeName)) {\r\n                    /** font size */\r\n                    if (selectionParent.style.fontSize.length > 0) {\r\n                        commandMapNodes.push('SIZE');\r\n                        util.changeTxt(commandMap.SIZE, selectionParent.style.fontSize.match(/\\d+/)[0]);\r\n                        findSize = false;\r\n                    }\r\n                }\r\n\r\n                /** command map */\r\n                cssText = selectionParent.style.cssText;\r\n                if (findB && /font\\-weight\\s*:\\s*(?:\\d+|bold|bolder)(?:;|\\s|)/.test(cssText)) {\r\n                    commandMapNodes.push('B');\r\n                    findB = false;\r\n                }\r\n                if (findI && /font\\-style\\s*:\\s*(?:italic|oblique)(?:;|\\s)/.test(cssText)) {\r\n                    commandMapNodes.push('I');\r\n                    findI = false;\r\n                }\r\n                if (findU && /text\\-decoration(?:\\-line)?\\s*:\\s*underline(?:;|\\s|)/.test(cssText)) {\r\n                    commandMapNodes.push('U');\r\n                    findU = false;\r\n                }\r\n                if (findS && /text\\-decoration(?:\\-line)?\\s*:\\s*line-through(?:;|\\s|)/.test(cssText)) {\r\n                    commandMapNodes.push('STRIKE');\r\n                    findS = false;\r\n                }\r\n\r\n                commandMapNodes.push((/^STRONG$/.test(nodeName) ? 'B' : /^EM$/.test(nodeName) ? 'I' : nodeName));\r\n            }\r\n\r\n            /** A Tag edit controller off */\r\n            if (findA) editor.controllersOff();\r\n\r\n            /** toggle class on */\r\n            for (let i = 0; i < commandMapNodes.length; i++) {\r\n                nodeName = commandMapNodes[i];\r\n                if (classOnCheck.test(nodeName)) {\r\n                    util.addClass(commandMap[nodeName], 'on');\r\n                }\r\n            }\r\n\r\n            /** remove class, display text */\r\n            for (let key in commandMap) {\r\n                if (commandMapNodes.indexOf(key) > -1) continue;\r\n                if (/^FONT/i.test(key)) {\r\n                    util.changeTxt(commandMap[key], lang.toolbar.font);\r\n                }\r\n                else if (/^SIZE$/i.test(key)) {\r\n                    util.changeTxt(commandMap[key], lang.toolbar.fontSize);\r\n                }\r\n                else {\r\n                    util.removeClass(commandMap[key], 'on');\r\n                }\r\n            }\r\n\r\n            /** save current nodes */\r\n            editor._variable.currentNodes = currentNodes.reverse();\r\n\r\n            /**  Displays the current node structure to resizingBar */\r\n            if (context.option.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\r\n        },\r\n\r\n        _cancelCaptionEdit: function () {\r\n            this.setAttribute('contenteditable', false);\r\n            this.removeEventListener('blur', event._cancelCaptionEdit);\r\n        },\r\n\r\n        onClick_toolbar: function (e) {\r\n            e.preventDefault();\r\n            e.stopPropagation();\r\n\r\n            let target = e.target;\r\n            let display = target.getAttribute('data-display');\r\n            let command = target.getAttribute('data-command');\r\n            let className = target.className;\r\n\r\n            while (!command && !/editor_tool/.test(className) && !/sun-editor-id-toolbar/.test(className)) {\r\n                target = target.parentNode;\r\n                command = target.getAttribute('data-command');\r\n                display = target.getAttribute('data-display');\r\n                className = target.className;\r\n            }\r\n\r\n            if (!command && !display) return;\r\n            if (target.disabled) return;\r\n            \r\n            /** Dialog, Submenu */\r\n            if (display) {\r\n                if (/submenu/.test(display) && (target.nextElementSibling === null || target !== editor.submenuActiveButton)) {\r\n                    editor.submenuOff();\r\n                    editor.callPlugin(command, function () {\r\n                        editor.submenuOn(target);\r\n                    });\r\n                    return;\r\n                }\r\n                else if (/dialog/.test(display)) {\r\n                    editor.callPlugin(command, function () {\r\n                        editor.plugins.dialog.open.call(editor, command, false);\r\n                    });\r\n                }\r\n\r\n                editor.submenuOff();\r\n                return;\r\n            }\r\n\r\n            editor.submenuOff();\r\n\r\n            /** default command */\r\n            if (command) {\r\n                editor.focus();\r\n                editor.commandHandler(target, command);\r\n            }\r\n        },\r\n\r\n        onClick_wysiwyg: function (e) {\r\n            e.stopPropagation();\r\n            const targetElement = e.target;\r\n            editor.submenuOff();\r\n\r\n            if (/^IMG$/i.test(targetElement.nodeName)) {\r\n                e.preventDefault();\r\n                editor.callPlugin('image', function () {\r\n                    const size = editor.plugins.resizing.call_controller_resize.call(editor, targetElement, 'image');\r\n                    editor.plugins.image.onModifyMode.call(editor, targetElement, size);\r\n                    \r\n                    if (!util.getParentElement(targetElement, '.sun-editor-id-image-container')) {\r\n                        editor.plugins.image.openModify.call(editor, true);\r\n                        editor.plugins.image.update_image.call(editor);\r\n                        editor.controllersOff();\r\n                    }\r\n                });\r\n\r\n                return;\r\n            }\r\n\r\n            if (/sun-editor-id-iframe-inner-resizing-cover/i.test(targetElement.className)) {\r\n                e.preventDefault();\r\n                editor.callPlugin('video', function () {\r\n                    const iframe = util.getChildElement(targetElement.parentNode, 'iframe');\r\n                    const size = editor.plugins.resizing.call_controller_resize.call(editor, iframe, 'video');\r\n                    editor.plugins.video.onModifyMode.call(editor, iframe, size);\r\n                });\r\n\r\n                return;\r\n            }\r\n\r\n            editor._setEditorRange();\r\n            event._findButtonEffectTag();\r\n\r\n            if (editor._isBalloon) {\r\n                const range = editor.getRange();\r\n\r\n                if (range.collapsed) {\r\n                    event._hideToolbar();\r\n                } else {\r\n                    event._showToolbarBalloon(range);\r\n                    return;\r\n                }\r\n            }\r\n\r\n            const figcaption = util.getParentElement(targetElement, 'FIGCAPTION');\r\n            if (figcaption && figcaption.getAttribute('contenteditable') !== 'ture') {\r\n                e.preventDefault();\r\n                figcaption.setAttribute('contenteditable', true);\r\n                figcaption.focus();\r\n\r\n                if (editor._isInline && !editor._inlineToolbarAttr.isShow) {\r\n                    event._showToolbarInline();\r\n\r\n                    const hideToolbar = function () {\r\n                        event._hideToolbar();\r\n                        _d.removeEventListener('click', hideToolbar);\r\n                    }\r\n\r\n                    _d.addEventListener('click', hideToolbar);\r\n                }\r\n            } else {\r\n                const td = util.getParentElement(targetElement, util.isCell);\r\n                if (td) {\r\n                    if (editor.controllerArray.length === 0) {\r\n                        editor.callPlugin('table', editor.plugins.table.call_controller_tableEdit.bind(editor, td));\r\n                    }\r\n                }\r\n            }\r\n\r\n            if (userFunction.onClick) userFunction.onClick(e);\r\n        },\r\n\r\n        _showToolbarBalloon: function (range) {\r\n            const toolbar = context.element.toolbar;\r\n            const childNodes = util.getListChildNodes(range.commonAncestorContainer);\r\n            const selection = _w.getSelection();\r\n            const isDirTop = util.getArrayIndex(childNodes, selection.focusNode) < util.getArrayIndex(childNodes, selection.anchorNode)\r\n\r\n            let rects = range.getClientRects();\r\n            rects = rects[isDirTop ? 0 : rects.length - 1];\r\n            \r\n            toolbar.style.display = 'block';\r\n\r\n            let l = (isDirTop ? rects.left : rects.right) - context.element.topArea.offsetLeft + _w.scrollX - toolbar.offsetWidth / 2;\r\n            let t = (isDirTop ? rects.top - toolbar.offsetHeight - 11 : rects.bottom + 11) - context.element.topArea.offsetTop + _w.scrollY;\r\n            \r\n            toolbar.style.left = (l < 0 ? 20 : l) + 'px';\r\n            toolbar.style.top = (t) + 'px';\r\n\r\n            if (isDirTop) {\r\n                util.removeClass(context.element._arrow, 'arrow-up');\r\n                util.addClass(context.element._arrow, 'arrow-down');\r\n                context.element._arrow.style.top = (toolbar.offsetHeight) + 'px';\r\n            } else {\r\n                util.removeClass(context.element._arrow, 'arrow-down');\r\n                util.addClass(context.element._arrow, 'arrow-up');\r\n                context.element._arrow.style.top = '-11px';\r\n            }\r\n\r\n            const arrow_width = context.element._arrow.offsetWidth;\r\n            const arrow_left = (toolbar.offsetWidth / 2 + (l < 0 ? l - arrow_width : 0));\r\n            context.element._arrow.style.left = (arrow_left < arrow_width / 2 ? arrow_width / 2 - 1 : arrow_left) + 'px';\r\n        },\r\n\r\n        _showToolbarInline: function () {\r\n            const toolbar = context.element.toolbar;\r\n            toolbar.style.display = 'block';\r\n            editor._inlineToolbarAttr.width = toolbar.style.width = context.option.toolbarWidth;\r\n            editor._inlineToolbarAttr.top = toolbar.style.top = (-1 - toolbar.offsetHeight) + 'px';\r\n            event.onScroll_window();\r\n            editor._inlineToolbarAttr.isShow = true;\r\n        },\r\n\r\n        _hideToolbar: function () {\r\n            context.element.toolbar.style.display = 'none';\r\n            editor._inlineToolbarAttr.isShow = false;\r\n        },\r\n\r\n        onKeyDown_wysiwyg: function (e) {\r\n            const keyCode = e.keyCode;\r\n            const shift = e.shiftKey;\r\n            const ctrl = e.ctrlKey || e.metaKey;\r\n            const alt = e.altKey;\r\n            e.stopPropagation();\r\n\r\n            if (editor._isBalloon) {\r\n                event._hideToolbar();\r\n            }\r\n\r\n            function shortcutCommand(keyCode) {\r\n                const key = event._shortcutKeyCode[keyCode];\r\n                if (!key) return false;\r\n\r\n                editor.commandHandler(util.getFormatElement(editor.getSelectionNode()), key[0]);\r\n                util.toggleClass(editor.commandMap[key[1]], 'on');\r\n\r\n                return true;\r\n            }\r\n\r\n            /** Shortcuts */\r\n            if (ctrl && !/^(?:16|17|18)$/.test(keyCode)) {\r\n                if (!(shift && keyCode !== 83) && shortcutCommand(keyCode)) {\r\n                    e.preventDefault();\r\n                    return;\r\n                }\r\n            }\r\n\r\n            /** default key action */\r\n            const selectionNode = editor.getSelectionNode();\r\n            switch (keyCode) {\r\n                case 8: /**backspace key*/\r\n                    if (util.isFormatElement(selectionNode) && util.isWysiwygDiv(selectionNode.parentNode) && selectionNode.previousSibling === null) {\r\n                        e.preventDefault();\r\n                        e.stopPropagation();\r\n                        selectionNode.innerHTML = util.zeroWidthSpace;\r\n                        return false;\r\n                    }\r\n                    \r\n                    break;\r\n                case 9:\r\n                    /**tab key*/\r\n                    e.preventDefault();\r\n                    if (ctrl || alt) break;\r\n\r\n                    editor.controllersOff();\r\n\r\n                    let currentNode = selectionNode || editor.getSelectionNode();\r\n                    while (!util.isCell(currentNode) && !util.isWysiwygDiv(currentNode)) {\r\n                        currentNode = currentNode.parentNode;\r\n                    }\r\n\r\n                    if (currentNode && util.isCell(currentNode)) {\r\n                        const table = util.getParentElement(currentNode, 'table');\r\n                        const cells = util.getListChildren(table, util.isCell);\r\n                        let idx = shift ? util.prevIdx(cells, currentNode) : util.nextIdx(cells, currentNode);\r\n\r\n                        if (idx === cells.length && !shift) idx = 0;\r\n                        if (idx === -1 && shift) idx = cells.length - 1;\r\n\r\n                        const moveCell = cells[idx];\r\n                        if (!moveCell) return false;\r\n\r\n                        editor.setRange(moveCell, 0, moveCell, 0);\r\n\r\n                        break;\r\n                    }\r\n\r\n                    /** format Tag */\r\n                    const lines = editor.getSelectedFormatElements();\r\n\r\n                    if (!shift) {\r\n                        const tabText = util.createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\r\n                        if (lines.length === 1) {\r\n                            editor.insertNode(tabText);\r\n                            editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize);\r\n                        } else {\r\n                            for (let i = 0, len = lines.length; i < len; i++) {\r\n                                lines[i].insertBefore(tabText.cloneNode(false), lines[i].firstChild);\r\n                            }\r\n                        }\r\n                    } else {\r\n                        for (let i = 0, len = lines.length, child; i < len; i++) {\r\n                            child = lines[i].firstChild;\r\n                            if (/^\\s{1,4}$/.test(child.textContent)) {\r\n                                util.removeItem(child);\r\n                            } else if (/^\\s{1,4}/.test(child.textContent)) {\r\n                                child.textContent = child.textContent.replace(/^\\s{1,4}/, '');\r\n                            }\r\n                        }\r\n                    }\r\n\r\n                    break;\r\n            }\r\n\r\n            if (userFunction.onKeyDown) userFunction.onKeyDown(e);\r\n        },\r\n\r\n        onKeyUp_wysiwyg: function (e) {\r\n            editor._setEditorRange();\r\n            editor.controllersOff();\r\n            const selectionNode = editor.getSelectionNode();\r\n\r\n            /** when format tag deleted */\r\n            if (e.keyCode === 8 && util.isWysiwygDiv(selectionNode) && context.element.wysiwyg.textContent.length === 0) {\r\n                e.preventDefault();\r\n                e.stopPropagation();\r\n\r\n                const oFormatTag = util.createElement(util.isFormatElement(editor._variable.currentNodes[0]) ? editor._variable.currentNodes[0] : 'P');\r\n                oFormatTag.innerHTML = util.zeroWidthSpace;\r\n\r\n                selectionNode.appendChild(oFormatTag);\r\n                editor.setSelectionNode(oFormatTag);\r\n                editor.setRange(oFormatTag, 0, oFormatTag, 0);\r\n                return;\r\n            }\r\n\r\n            if ((util.isWysiwygDiv(selectionNode.parentElement) || util.isRangeFormatElement(selectionNode.parentElement)) && selectionNode.nodeType === 3) {\r\n                editor.execCommand('formatBlock', false, util.isWysiwygDiv(selectionNode.parentElement) ? 'P' : 'DIV');\r\n                editor._setEditorRange();\r\n                event._findButtonEffectTag();\r\n                return;\r\n            }\r\n\r\n            if (event._directionKeyKeyCode.test(e.keyCode)) {\r\n                event._findButtonEffectTag();\r\n            }\r\n\r\n            if (userFunction.onKeyUp) userFunction.onKeyUp(e);\r\n        },\r\n\r\n        onScroll_wysiwyg: function (e) {\r\n            editor.controllersOff();\r\n            if (userFunction.onScroll) userFunction.onScroll(e);\r\n        },\r\n\r\n        onDrop_wysiwyg: function (e) {\r\n            const files = e.dataTransfer.files;\r\n\r\n            if (files.length > 0) {\r\n                e.stopPropagation();\r\n                e.preventDefault();\r\n                \r\n                editor.focus();\r\n    \r\n                editor.callPlugin('image', function () {\r\n                    context.image.imgInputFile.files = files;\r\n                    editor.plugins.image.onRender_imgInput.call(editor);\r\n                    context.image.imgInputFile.files = null;\r\n                });\r\n            }\r\n\r\n            if (userFunction.onDrop) userFunction.onDrop(e);\r\n        },\r\n\r\n        onMouseDown_resizingBar: function (e) {\r\n            e.stopPropagation();\r\n\r\n            editor._variable.resizeClientY = e.clientY;\r\n            context.element.resizeBackground.style.display = 'block';\r\n\r\n            function closureFunc() {\r\n                context.element.resizeBackground.style.display = 'none';\r\n                _d.removeEventListener('mousemove', event._resize_editor);\r\n                _d.removeEventListener('mouseup', closureFunc);\r\n            }\r\n\r\n            _d.addEventListener('mousemove', event._resize_editor);\r\n            _d.addEventListener('mouseup', closureFunc);\r\n        },\r\n\r\n        _resize_editor: function (e) {\r\n            const resizeInterval = context.element.editorArea.offsetHeight + (e.clientY - editor._variable.resizeClientY);\r\n            context.element.wysiwyg.style.height = context.element.code.style.height = (resizeInterval < editor._variable.minResizingSize ? editor._variable.minResizingSize : resizeInterval) + 'px';\r\n            editor._variable.resizeClientY = e.clientY;\r\n        },\r\n\r\n        onResize_window: function () {\r\n            if (context.element.topArea.offsetWidth === 0) return;\r\n\r\n            if (editor._variable.isFullScreen) {\r\n                editor._variable.innerHeight_fullScreen += (_w.innerHeight - context.element.toolbar.offsetHeight) - editor._variable.innerHeight_fullScreen;\r\n                context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\r\n            }\r\n            else if (editor._variable._sticky) {\r\n                context.element.toolbar.style.width = (context.element.topArea.offsetWidth - 2) + 'px';\r\n                event.onScroll_window();\r\n            }\r\n\r\n            editor.controllersOff();\r\n        },\r\n\r\n        onScroll_window: function () {\r\n            if (editor._variable.isFullScreen || context.element.topArea.offsetWidth === 0) return;\r\n\r\n            const element = context.element;\r\n            const editorHeight = element.editorArea.offsetHeight;\r\n            const editorTop = element.topArea.offsetTop - (editor._isInline ? element.toolbar.offsetHeight : 0);\r\n            const y = (this.scrollY || _d.documentElement.scrollTop) + context.option.stickyToolbar;\r\n            \r\n            if (y < editorTop) {\r\n                event._offStickyToolbar(element);\r\n            }\r\n            else if (y + editor._variable.minResizingSize >= editorHeight + editorTop) {\r\n                if (!editor._variable._sticky) event._onStickyToolbar(element);\r\n                element.toolbar.style.top = (editorHeight + editorTop + context.option.stickyToolbar -y - editor._variable.minResizingSize) + 'px';\r\n            }\r\n            else if (y >= editorTop) {\r\n                event._onStickyToolbar(element);\r\n            }\r\n        },\r\n\r\n        _onStickyToolbar: function (element) {\r\n            if (!editor._isInline) {\r\n                element._stickyDummy.style.height = element.toolbar.offsetHeight + 'px';\r\n                element._stickyDummy.style.display = 'block';\r\n            }\r\n\r\n            element.toolbar.style.top = context.option.stickyToolbar + 'px';\r\n            element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : element.toolbar.offsetWidth + 'px';\r\n            util.addClass(element.toolbar, 'sun-editor-sticky');\r\n            editor._variable._sticky = true;\r\n        },\r\n\r\n        _offStickyToolbar: function (element) {\r\n            element._stickyDummy.style.display = 'none';\r\n            element.toolbar.style.top = editor._isInline ? editor._inlineToolbarAttr.top : '';\r\n            element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : '';\r\n            element.editorArea.style.marginTop = '';\r\n            util.removeClass(element.toolbar, 'sun-editor-sticky');\r\n            editor._variable._sticky = false;\r\n        },\r\n\r\n        _codeViewAutoScroll: function () {\r\n            context.element.code.style.height = context.element.code.scrollHeight + 'px';\r\n        },\r\n\r\n        onPaste_wysiwyg: function (e) {\r\n            if (!e.clipboardData.getData) return true;\r\n\r\n            const cleanData = util.cleanHTML(e.clipboardData.getData('text/html'));\r\n            \r\n            if (cleanData) {\r\n                editor.execCommand('insertHTML', false, cleanData);\r\n                e.stopPropagation();\r\n                e.preventDefault();\r\n            }\r\n        }\r\n    };\r\n\r\n    /** add event listeners */\r\n    /** toolbar event */\r\n    context.element.toolbar.addEventListener('click', event.onClick_toolbar, false);\r\n    context.element.toolbar.addEventListener('mousedown', function (e) { e.preventDefault() }, false);\r\n    /** editor area */\r\n    context.element.relative.addEventListener('click', editor.focus.bind(editor), false);\r\n    context.element.wysiwyg.addEventListener('click', event.onClick_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('keydown', event.onKeyDown_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('keyup', event.onKeyUp_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('drop', event.onDrop_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('paste', event.onPaste_wysiwyg, false);\r\n    \r\n    /** code view area auto line */\r\n    if (context.option.height === 'auto') context.element.code.addEventListener('keyup', event._codeViewAutoScroll, false);\r\n\r\n    /** resizingBar */\r\n    if (context.element.resizingBar) {\r\n        if (/\\d+/.test(context.option.height)) {\r\n            context.element.resizingBar.addEventListener('mousedown', event.onMouseDown_resizingBar, false);\r\n        } else {\r\n            util.addClass(context.element.resizingBar, 'none-resize');\r\n        }\r\n    }\r\n\r\n    /** inlineToolbar */\r\n    if (editor._isInline) {\r\n        context.element.wysiwyg.addEventListener('focus', event._showToolbarInline, false);\r\n    }\r\n\r\n    if (editor._isInline || editor._isBalloon) {\r\n        context.element.wysiwyg.addEventListener('blur', event._hideToolbar, false);\r\n    }\r\n    \r\n    /** window event */\r\n    _w.addEventListener('resize', event.onResize_window, false);\r\n    if (context.option.stickyToolbar > -1) _w.addEventListener('scroll', event.onScroll_window, false);\r\n\r\n    /** add plugin to plugins object */\r\n    if (plugins) {\r\n        Object.keys(plugins).map(function(key) {\r\n            let plugin = plugins[key];\r\n            editor.plugins[plugin.name] = util.copyObj(plugin);\r\n        });\r\n    }\r\n\r\n    /** User function */\r\n    const userFunction = {\r\n        /**\r\n         * @description Event functions\r\n         * @param {Object} event - Event Object\r\n         */\r\n        onScroll: null,\r\n        onClick: null,\r\n        onKeyDown: null,\r\n        onKeyUp: null,\r\n        onDrop: null,\r\n\r\n        /**\r\n         * @description Called when the image is uploaded or the uploaded image is deleted\r\n         * @param {Element} targetImgElement - Current img element\r\n         * @param {Number} index - Uploaded index\r\n         * @param {Boolean} isDelete - Whether or not it was called after the delete operation\r\n         */\r\n        onImageUpload: null,\r\n\r\n        /**\r\n         * @description Open a notice area\r\n         * @param {String} message - Notice message\r\n         */\r\n        noticeOpen: function (message) {\r\n            editor.addModule([notice]);\r\n            notice.open.call(editor, message);\r\n        },\r\n\r\n        /**\r\n         * @description Close a notice area\r\n         */\r\n        noticeClose: function () {\r\n            editor.addModule([notice]);\r\n            notice.close.call(editor);\r\n        },\r\n\r\n        /**\r\n         * @description Copying the contents of the editor to the original textarea\r\n         */\r\n        save: function () {\r\n            context.element.originElement.value = editor.getContents();\r\n        },\r\n\r\n        /**\r\n         * @description Gets the suneditor's context object. Contains settings, plugins, and cached element objects\r\n         * @returns {Object}\r\n         */\r\n        getContext: function () {\r\n            return context;\r\n        },\r\n\r\n        /**\r\n         * @description Gets the contents of the suneditor\r\n         * @returns {String}\r\n         */\r\n        getContents: function () {\r\n            return editor.getContents();\r\n        },\r\n\r\n        /**\r\n         * @description Gets uploaded images informations\r\n         * @returns {Array}\r\n         */\r\n        getImagesInfo: function () {\r\n            return editor._variable._imagesInfo;\r\n        },\r\n\r\n        /**\r\n         * @description Inserts an HTML element or HTML string or plain string at the current cursor position\r\n         * @param {Element|String} html - HTML Element or HTML string or plain string\r\n         */\r\n        insertHTML: function (html) {\r\n            if (!html.nodeType || html.nodeType !== 1) {\r\n                const template = util.createElement('template');\r\n                template.innerHTML = html;\r\n                html = template.firstChild || template.content.firstChild;\r\n            }\r\n\r\n            editor.insertNode(html);\r\n            editor.focus();\r\n        },\r\n\r\n        /**\r\n         * @description Change the contents of the suneditor\r\n         * @param {String} contents - Contents to Input\r\n         */\r\n        setContents: function (contents) {\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.wysiwyg.innerHTML = util.convertContentsForEditor(contents);\r\n            } else {\r\n                context.element.code.value = contents;\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Add contents to the suneditor\r\n         * @param {String} contents - Contents to Input\r\n         */\r\n        appendContents: function (contents) {\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.wysiwyg.innerHTML += util.convertContentsForEditor(contents);\r\n            } else {\r\n                context.element.code.value += contents;\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Disable the suneditor\r\n         */\r\n        disabled: function () {\r\n            context.tool.cover.style.display = 'block';\r\n            context.element.wysiwyg.setAttribute('contenteditable', false);\r\n            context.element.code.setAttribute('disabled', 'disabled');\r\n        },\r\n\r\n        /**\r\n         * @description Enabled the suneditor\r\n         */\r\n        enabled: function () {\r\n            context.tool.cover.style.display = 'none';\r\n            context.element.wysiwyg.setAttribute('contenteditable', true);\r\n            context.element.code.removeAttribute('disabled');\r\n        },\r\n\r\n        /**\r\n         * @description Show the suneditor\r\n         */\r\n        show: function () {\r\n            const topAreaStyle = context.element.topArea.style;\r\n            if (topAreaStyle.display === 'none') topAreaStyle.display = context.option.display;\r\n        },\r\n\r\n        /**\r\n         * @description Hide the suneditor\r\n         */\r\n        hide: function () {\r\n            context.element.topArea.style.display = 'none';\r\n        },\r\n\r\n        /**\r\n         * @description Destroy the suneditor\r\n         */\r\n        destroy: function () {\r\n            /** remove window event listeners */\r\n            _w.removeEventListener('resize', event.onResize_window);\r\n            _w.removeEventListener('scroll', event.onScroll_window);\r\n            \r\n            /** remove element */\r\n            util.removeItem(context.element.topArea);\r\n\r\n            this.onScroll = null;\r\n            this.onClick = null;\r\n            this.onKeyDown = null;\r\n            this.onKeyUp = null;\r\n            this.onDrop = null;\r\n            this.save = null;\r\n            this.onImageUpload = null;\r\n            this.noticeOpen = null;\r\n            this.noticeClose = null;\r\n            this.getContext = null;\r\n            this.getContents = null;\r\n            this.getImagesInfo = null;\r\n            this.insertHTML = null;\r\n            this.setContents = null;\r\n            this.appendContents = null;\r\n            this.disabled = null;\r\n            this.enabled = null;\r\n            this.show = null;\r\n            this.hide = null;\r\n            this.destroy = null;\r\n\r\n            context = null;\r\n            plugins = null;\r\n            lang = null;\r\n        }\r\n    };\r\n\r\n    return userFunction;\r\n};\r\n\r\nexport default core;"],"mappings":"AAAA;AAAA;AAAA;AAAA;;;;;;;AAOA;AACA;AACA;AACA;AAEA;;;;;;;;AAOA;AACA;AACA;AACA;;;;;AAIA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;;AAIA;AACA;AACA;;;;AAIA;AACA;AACA;;;;AAIA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AAYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AACA;AAWA;;;;;;;;;;;;;AAaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAjBA;AACA;AAmBA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;;;;;;;AAOA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;;;;AAQA;AACA;AACA;AACA;AACA;AACA;AAEA;AAGA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAXA;AAaA;AACA;AACA;AACA;AAGA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;AAQA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AAWA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AAcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAJA;AAMA;AACA;AACA;;;;;;;;AAQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;AAWA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;;;;;;AAWA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AAhDA;AACA;AAkDA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AAAA;AACA;AAaA;AACA;AACA;;;;AAIA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA9iDA;AAijDA;;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AARA;AAWA;AAEA;AAEA;AACA;AACA;AACA;AACA;AAEA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AAEA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAAA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AADA;AAEA;AACA;AAAA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAAA;AACA;AAAA;AAEA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAEA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA9DA;AACA;AAgEA;AACA;AAEA;AACA;AACA;AAAA;AACA;AAEA;AACA;AAAA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAhiBA;AAmiBA;AACA;AAAA;AACA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAAA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AAAA;AAEA;AACA;AACA;AADA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AA9KA;AAiLA;AACA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/lib/core.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _util__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./util */ \"./src/lib/util.js\");\n/* harmony import */ var _plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../plugins/modules/notice */ \"./src/plugins/modules/notice.js\");\n/*\r\n * wysiwyg web editor\r\n *\r\n * suneditor.js\r\n * Copyright 2017 JiHong Lee.\r\n * MIT license.\r\n */\n\n\n\n\n/**\r\n * @description SunEditor core closure\r\n * @param context\r\n * @param plugins\r\n * @param lang\r\n * @returns {{save: save, getContext: getContext, getContent: getContent, setContent: setContent, appendContent: appendContent, disabled: disabled, enabled: enabled, show: show, hide: hide, destroy: destroy}}\r\n */\n\nconst core = function (context, plugins, lang) {\n const _d = document;\n const _w = window;\n /**\r\n * @description editor core object\r\n * should always bind this object when registering an event in the plug-in.\r\n */\n\n const editor = {\n /**\r\n * @description Elements and user options parameters of the suneditor\r\n */\n context: context,\n\n /**\r\n * @description loaded plugins\r\n */\n plugins: {},\n\n /**\r\n * @description Whether the plugin is initialized\r\n */\n initPlugins: {},\n\n /**\r\n * @description loaded language\r\n */\n lang: lang,\n\n /**\r\n * @description dialog element\r\n */\n dialogForm: null,\n\n /**\r\n * @description submenu element\r\n */\n submenu: null,\n\n /**\r\n * @description active button element in submenu\r\n */\n submenuActiveButton: null,\n\n /**\r\n * @description The elements array to be processed unvisible when the controllersOff function is executed (resizing, link modified button, table controller)\r\n */\n controllerArray: [],\n\n /**\r\n * @description The functions array to be executed when the controllersOff function is executed ex) init function of table plugin\r\n */\n controllerFunction: [],\n\n /**\r\n * @description An array of buttons whose class name is not \"code-view-enabled\"\r\n */\n codeViewDisabledButtons: context.element.toolbar.querySelectorAll('.sun-editor-id-toolbar button:not([class~=\"code-view-enabled\"])'),\n\n /**\r\n * @description Is inline mode?\r\n * @private\r\n */\n _isInline: /inline/i.test(context.option.mode),\n\n /**\r\n * @description Is balloon mode?\r\n * @private\r\n */\n _isBalloon: /balloon/i.test(context.option.mode),\n\n /**\r\n * @description Required value when using inline mode to sticky toolbar\r\n * @private\r\n */\n _inlineToolbarAttr: {\n width: 0,\n height: 0,\n isShow: false\n },\n\n /**\r\n * @description An user event function when image uploaded success or remove image\r\n * @private\r\n */\n _imageUpload: function (targetImgElement, index, isDelete) {\n if (userFunction.onImageUpload) userFunction.onImageUpload(targetImgElement, index, isDelete);\n },\n\n /**\r\n * @description Elements that need to change text or className for each selection change\r\n * @property {Element} FORMAT - format button\r\n * @property {Element} FONT - font family button\r\n * @property {Element} SIZE - font size button\r\n * @property {Element} B - bold button\r\n * @property {Element} U - underline button\r\n * @property {Element} I - italic button\r\n * @property {Element} STRIKE - strike button\r\n * @property {Element} SUB - subscript button\r\n * @property {Element} SUP - superscript button\r\n */\n commandMap: {\n FORMAT: context.tool.format,\n FONT: context.tool.font,\n SIZE: context.tool.fontSize,\n B: context.tool.bold,\n U: context.tool.underline,\n I: context.tool.italic,\n STRIKE: context.tool.strike,\n SUB: context.tool.subscript,\n SUP: context.tool.superscript\n },\n\n /**\r\n * @description Variables used internally in editor operation\r\n * @property {(Element|null)} selectionNode - Contains selection node\r\n * @property {(Object|null)} range - The current range object\r\n * @property {Boolean} wysiwygActive - The wysiwyg frame or code view state\r\n * @property {Boolean} isFullScreen - State of full screen\r\n * @property {Number} innerHeight_fullScreen - InnerHeight in editor when in full screen\r\n * @property {Number} resizeClientY - Remember the vertical size of the editor before resizing the editor (Used when calculating during resize operation)\r\n * @property {Number} tabSize - Indented size when tab button clicked (4)\r\n * @property {Number} minResizingSize - Minimum size of editing area when resized (65)\r\n * @property {Array} currentNodes - An array of the current cursor's node structure\r\n * @private\r\n */\n _variable: {\n selectionNode: null,\n range: null,\n wysiwygActive: true,\n isFullScreen: false,\n innerHeight_fullScreen: 0,\n resizeClientY: 0,\n tabSize: 4,\n minResizingSize: 65,\n currentNodes: [],\n _originCssText: context.element.topArea.style.cssText,\n _bodyOverflow: '',\n _editorAreaOriginCssText: '',\n _wysiwygOriginCssText: '',\n _codeOriginCssText: '',\n _sticky: false,\n _imagesInfo: [],\n _imageIndex: 0\n },\n\n /**\r\n * @description If the plugin is not added, add the plugin and call the 'add' function.\r\n * If the plugin is added call callBack function.\r\n * @param {String} pluginName - The name of the plugin to call\r\n * @param {function} callBackFunction - Function to be executed immediately after module call\r\n */\n callPlugin: function (pluginName, callBackFunction) {\n if (!this.plugins[pluginName]) {\n throw Error('[SUNEDITOR.core.callPlugin.fail] The called plugin does not exist or is in an invalid format. (pluginName:\"' + pluginName + '\")');\n } else if (!this.initPlugins[pluginName]) {\n this.plugins[pluginName].add(this, this.plugins[pluginName].buttonElement);\n this.initPlugins[pluginName] = true;\n }\n\n callBackFunction();\n },\n\n /**\r\n * @description If the module is not added, add the module and call the 'add' function\r\n * @param {Array} moduleArray - module object's Array [dialog, resizing]\r\n */\n addModule: function (moduleArray) {\n let moduleName = '';\n\n for (let i = 0, len = moduleArray.length; i < len; i++) {\n moduleName = moduleArray[i].name;\n\n if (!this.plugins[moduleName]) {\n this.plugins[moduleName] = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].copyObj(moduleArray[i]);\n this.plugins[moduleName].add(this);\n }\n }\n },\n\n /**\r\n * @description Enabled submenu\r\n * @param {Element} element - Submenu element to call\r\n */\n submenuOn: function (element) {\n const submenuName = element.getAttribute('data-command');\n if (this.plugins[submenuName].on) this.plugins[submenuName].on.call(this);\n this.submenu = element.nextElementSibling;\n this.submenu.style.display = 'block';\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(element, 'on');\n this.submenuActiveButton = element;\n const overLeft = this.context.element.toolbar.offsetWidth - (element.parentElement.offsetLeft + this.submenu.offsetWidth);\n if (overLeft < 0) this.submenu.style.left = overLeft + 'px';else this.submenu.style.left = '1px';\n },\n\n /**\r\n * @description Disable submenu\r\n */\n submenuOff: function () {\n if (this.submenu) {\n this.submenu.style.display = 'none';\n this.submenu = null;\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(this.submenuActiveButton, 'on');\n this.submenuActiveButton = null;\n }\n\n this.controllersOff();\n },\n\n /**\r\n * @description Disable controller in editor area (link button, image resize button)\r\n */\n controllersOff: function () {\n const len = this.controllerArray.length;\n const fLen = this.controllerFunction.length;\n\n if (len > 0) {\n for (let i = 0; i < len; i++) {\n this.controllerArray[i].style.display = 'none';\n }\n\n this.controllerArray = [];\n }\n\n if (fLen > 0) {\n for (let i = 0; i < fLen; i++) {\n this.controllerFunction[i]();\n }\n\n this.controllerFunction = [];\n }\n },\n\n /**\r\n * @description javascript execCommand\r\n * @param {String} command - javascript execCommand function property\r\n * @param {Boolean} showDefaultUI - javascript execCommand function property\r\n * @param {String} value - javascript execCommand function property\r\n */\n execCommand: function (command, showDefaultUI, value) {\n _d.execCommand(command, showDefaultUI, command === 'formatBlock' ? '<' + value + '>' : value);\n },\n\n /**\r\n * @description Focus to wysiwyg area\r\n */\n focus: function () {\n if (context.element.wysiwyg.style.display === 'none') return;\n const caption = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(this.getSelectionNode(), 'figcaption');\n\n if (caption) {\n caption.focus();\n } else {\n context.element.wysiwyg.focus();\n }\n\n this._setEditorRange();\n\n event._findButtonEffectTag();\n },\n\n /**\r\n * @description Saving the range object and the currently selected node of editor\r\n * @private\r\n */\n _setEditorRange: function () {\n const selection = _w.getSelection();\n\n let range = null;\n\n if (selection.rangeCount > 0) {\n range = selection.getRangeAt(0);\n } else {\n range = this._createDefaultRange();\n }\n\n this._variable.range = range;\n\n if (range.collapsed) {\n this.setSelectionNode(range.commonAncestorContainer);\n } else {\n this.setSelectionNode(selection.extentNode || selection.anchorNode);\n }\n },\n\n /**\r\n * @description Return the range object of editor's first child node\r\n * @returns {Object}\r\n * @private\r\n */\n _createDefaultRange: function () {\n const range = _d.createRange();\n\n range.setStart(context.element.wysiwyg.firstChild, 0);\n range.setEnd(context.element.wysiwyg.firstChild, 0);\n return range;\n },\n\n /**\r\n * @description Set current editor's range object\r\n * @param {Element} startCon - The startContainer property of the selection object.\r\n * @param {Number} startOff - The startOffset property of the selection object.\r\n * @param {Element} endCon - The endContainer property of the selection object.\r\n * @param {Element} endOff - The endOffset property of the selection object.\r\n */\n setRange: function (startCon, startOff, endCon, endOff) {\n const range = _d.createRange();\n\n range.setStart(startCon, startOff);\n range.setEnd(endCon, endOff);\n\n const selection = _w.getSelection();\n\n if (selection.removeAllRanges) {\n selection.removeAllRanges();\n }\n\n selection.addRange(range);\n this._variable.range = range;\n\n this._setEditorRange();\n },\n\n /**\r\n * @description Get current editor's range object\r\n * @returns {Object}\r\n */\n getRange: function () {\n return this._variable.range || this._createDefaultRange();\n },\n\n /**\r\n * @description Set the selected node. (Used by getSelectionNode function)\r\n * @param {Node} node - node object\r\n */\n setSelectionNode: function (node) {\n this._variable.selectionNode = node;\n },\n\n /**\r\n * @description Get current select node\r\n * @returns {Node}\r\n */\n getSelectionNode: function () {\n if (this._variable.selectionNode) {\n return this._variable.selectionNode;\n }\n\n return context.element.wysiwyg.firstChild;\n },\n\n /**\r\n * @description Returns a \"formatElement\"(P, DIV, H[1-6], LI) array from the currently selected range.\r\n * @returns {Array}\r\n */\n getSelectedFormatElements: function () {\n let range = this.getRange();\n\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(range.startContainer)) {\n const children = context.element.wysiwyg.children;\n this.setRange(children[0], 0, children[children.length - 1], children[children.length - 1].textContent.length);\n range = this.getRange();\n }\n\n const startCon = range.startContainer;\n const endCon = range.endContainer;\n const commonCon = range.commonAncestorContainer;\n const rangeFormatElements = [];\n if (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(commonCon) && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(commonCon)) return [_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(commonCon)]; // get line nodes\n\n const lineNodes = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildren(commonCon, function (current) {\n return _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(current);\n });\n if (startCon === endCon) return lineNodes[0];\n let startLine = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(startCon);\n let endLine = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(endCon);\n let startIdx = 0;\n let endIdx = 0;\n\n for (let i = 0, len = lineNodes.length; i < len; i++) {\n if (startLine === lineNodes[i]) {\n startIdx = i;\n continue;\n }\n\n if (endLine === lineNodes[i]) {\n endIdx = i;\n break;\n }\n }\n\n for (let i = startIdx; i <= endIdx; i++) {\n rangeFormatElements.push(lineNodes[i]);\n }\n\n return rangeFormatElements;\n },\n\n /**\r\n * @description Returns a \"rangeFormatElement\"(blockquote, TABLE, TR, TD, OL, UL, PRE) array from the currently selected range.\r\n * @returns {Array}\r\n */\n getSelectedRangeFormatElements: function () {\n let range = this.getRange();\n\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(range.startContainer)) {\n const children = context.element.wysiwyg.children;\n this.setRange(children[0], 0, children[children.length - 1], children[children.length - 1].textContent.length);\n range = this.getRange();\n }\n\n const startCon = range.startContainer;\n const endCon = range.endContainer;\n const commonCon = range.commonAncestorContainer;\n const rangeFormatElements = [];\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(commonCon)) return [commonCon];\n\n if (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(commonCon)) {\n const el = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getRangeFormatElement(commonCon);\n return el ? [el] : [];\n } // get range Elements\n\n\n const rangeElements = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildren(commonCon, function (current) {\n return _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(current);\n });\n if (startCon === endCon) return rangeElements[0];\n let startLine = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getRangeFormatElement(startCon);\n let endLine = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getRangeFormatElement(endCon);\n let startIdx = 0;\n let endIdx = 0;\n\n for (let i = 0, len = rangeElements.length; i < len; i++) {\n if (startLine === rangeElements[i]) {\n startIdx = i;\n continue;\n }\n\n if (endLine === rangeElements[i]) {\n endIdx = i;\n break;\n }\n }\n\n for (let i = startIdx; i <= endIdx; i++) {\n if (rangeElements[i]) rangeFormatElements.push(rangeElements[i]);\n }\n\n return rangeFormatElements;\n },\n\n /**\r\n * @description Determine if this offset is the edge offset of container\r\n * @param {Object} container - The container property of the selection object.\r\n * @param {Number} offset - The offset property of the selection object.\r\n * @returns {Boolean}\r\n */\n isEdgePoint: function (container, offset) {\n return offset === 0 || offset === container.nodeValue.length;\n },\n\n /**\r\n * @description Show loading box\r\n */\n showLoading: function () {\n context.element.loading.style.display = 'block';\n },\n\n /**\r\n * @description Close loading box\r\n */\n closeLoading: function () {\n context.element.loading.style.display = 'none';\n },\n\n /**\r\n * @description Append format element to sibling node of argument element.\r\n * If the \"formatNodeName\" argument value is present, the tag of that argument value is inserted,\r\n * If not, the currently selected format tag is inserted.\r\n * @param {Element} element - Insert as siblings of that element\r\n * @param {String|null} formatNodeName - Node name to be inserted\r\n * @returns {Element}\r\n */\n appendFormatTag: function (element, formatNodeName) {\n const formatEl = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getRangeFormatElement(element) || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(element);\n const currentFormatEl = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(this.getSelectionNode());\n const oFormatName = formatNodeName ? formatNodeName : _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(currentFormatEl) ? currentFormatEl.nodeName : 'P';\n const oFormat = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement(oFormatName);\n oFormat.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(formatEl)) formatEl.insertBefore(oFormat, element.nextElementSibling);else formatEl.parentNode.insertBefore(oFormat, formatEl.nextElementSibling);\n return oFormat;\n },\n\n /**\r\n * @description Delete selected node and insert argument value node\r\n * @param {Element} oNode - Node to be inserted\r\n * @param {(Element|null)} rightNode - If the node exists, it is inserted after the node\r\n */\n insertNode: function (oNode, rightNode) {\n const range = this.getRange();\n let parentNode = null;\n\n if (!rightNode) {\n const startCon = range.startContainer;\n const startOff = range.startOffset;\n const endCon = range.endContainer;\n const endOff = range.endOffset;\n const commonCon = range.commonAncestorContainer;\n parentNode = startCon;\n\n if (startCon.nodeType === 3) {\n parentNode = startCon.parentNode;\n }\n /** No Select range node */\n\n\n if (range.collapsed) {\n if (commonCon.nodeType === 3) {\n rightNode = commonCon.splitText(endOff);\n } else {\n if (parentNode.lastChild !== null && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(parentNode.lastChild)) {\n parentNode.removeChild(parentNode.lastChild);\n }\n\n rightNode = null;\n }\n }\n /** Select range nodes */\n else {\n const isSameContainer = startCon === endCon;\n\n if (isSameContainer) {\n if (this.isEdgePoint(endCon, endOff)) rightNode = endCon.nextSibling;else rightNode = endCon.splitText(endOff);\n let removeNode = startCon;\n if (!this.isEdgePoint(startCon, startOff)) removeNode = startCon.splitText(startOff);\n parentNode.removeChild(removeNode);\n } else {\n this.removeNode();\n parentNode = commonCon;\n rightNode = endCon;\n\n while (rightNode.parentNode !== commonCon) {\n rightNode = rightNode.parentNode;\n }\n }\n }\n } else {\n parentNode = rightNode.parentNode;\n rightNode = rightNode.nextSibling;\n }\n\n try {\n parentNode.insertBefore(oNode, rightNode);\n } catch (e) {\n parentNode.appendChild(oNode);\n }\n },\n\n /**\r\n * @description Delete the currently selected node\r\n */\n removeNode: function () {\n const range = this.getRange();\n\n if (range.deleteContents) {\n range.deleteContents();\n return;\n }\n\n const startCon = range.startContainer;\n const startOff = range.startOffset;\n const endCon = range.endContainer;\n const endOff = range.endOffset;\n const commonCon = range.commonAncestorContainer;\n let beforeNode = null;\n let afterNode = null;\n const childNodes = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildNodes(commonCon);\n let startIndex = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getArrayIndex(childNodes, startCon);\n let endIndex = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getArrayIndex(childNodes, endCon);\n\n for (let i = startIndex + 1, startNode = startCon; i >= 0; i--) {\n if (childNodes[i] === startNode.parentNode && childNodes[i].firstChild === startNode && startOff === 0) {\n startIndex = i;\n startNode = startNode.parentNode;\n }\n }\n\n for (let i = endIndex - 1, endNode = endCon; i > startIndex; i--) {\n if (childNodes[i] === endNode.parentNode && childNodes[i].nodeType === 1) {\n childNodes.splice(i, 1);\n endNode = endNode.parentNode;\n --endIndex;\n }\n }\n\n for (let i = startIndex; i <= endIndex; i++) {\n const item = childNodes[i];\n\n if (item.length === 0 || item.nodeType === 3 && item.data === undefined) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(item);\n continue;\n }\n\n if (item === startCon) {\n if (startCon.nodeType === 1) {\n beforeNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(startCon.textContent);\n } else {\n beforeNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(startCon.substringData(0, startOff));\n }\n\n if (beforeNode.length > 0) {\n startCon.data = beforeNode.data;\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(startCon);\n }\n\n continue;\n }\n\n if (item === endCon) {\n if (endCon.nodeType === 1) {\n afterNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(endCon.textContent);\n } else {\n afterNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(endCon.substringData(endOff, endCon.length - endOff));\n }\n\n if (afterNode.length > 0) {\n endCon.data = afterNode.data;\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(endCon);\n }\n\n continue;\n }\n\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(item);\n }\n },\n\n /**\r\n * @description appended all selected format Element to the argument element and insert\r\n * @param {Element} wrapTag - Element of wrap the arguments\r\n */\n wrapToTags: function (wrapTag) {\n const range = this.getRange();\n const rangeLines = this.getSelectedFormatElements();\n\n if (!rangeLines) {\n const inner = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(this.getSelectionNode()) ? 'DIV' : 'P');\n inner.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n wrapTag.appendChild(inner);\n this.getSelectionNode().appendChild(wrapTag);\n return;\n }\n\n let last = rangeLines[rangeLines.length - 1];\n let standTag, beforeTag, pElement;\n\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(last) || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(last)) {\n standTag = last;\n } else {\n standTag = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getRangeFormatElement(last) || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(last);\n }\n\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(standTag)) {\n beforeTag = null;\n pElement = standTag;\n } else {\n beforeTag = standTag.nextSibling;\n pElement = standTag.parentNode;\n }\n\n let listParent = null;\n let line = null;\n let prevNodeName = '';\n\n for (let i = 0, len = rangeLines.length; i < len; i++) {\n line = rangeLines[i];\n\n if (/^LI$/i.test(line.nodeName)) {\n if (listParent === null || !/^LI$/i.test(prevNodeName)) {\n listParent = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement(line.parentNode.nodeName);\n }\n\n listParent.appendChild(line);\n if (i === len - 1 || !/^LI$/i.test(rangeLines[i + 1].nodeName)) wrapTag.appendChild(listParent);\n } else {\n wrapTag.appendChild(line);\n }\n\n prevNodeName = line.nodeName;\n }\n\n pElement.insertBefore(wrapTag, beforeTag);\n if (!range.collapsed && (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(range.startContainer) || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(range.endContainer))) _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeEmptyNode(pElement);\n },\n\n /**\r\n * @description Add the node received as an argument value to the selected area.\r\n * 1. When there is the same css value node in the selection area, the tag is stripped.\r\n * 2. If there is another css value other thanCss attribute values received as arguments on the node, removed only Css attribute values received as arguments\r\n * 3. If you pass an element whose node name is \"removenode\" as an argument value, it performs a type removal operation. ex) nodeChange(document.createElement('removenode'))\r\n * @param {Element} appendNode - The dom that will wrap the selected text area\r\n * @param {Array} checkCSSPropertyArray - The css attribute name Array to check (['font-size'], ['font-family', 'background-color', 'border']...])\r\n */\n nodeChange: function (appendNode, checkCSSPropertyArray) {\n const range = this.getRange();\n const isRemoveFormat = /removenode/i.test(appendNode.nodeName);\n let tempCon, tempOffset, tempChild, tempArray;\n /* checked same style property */\n\n if (range.startContainer === range.endContainer) {\n let sNode = range.startContainer;\n\n if (isRemoveFormat) {\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(sNode) === sNode.parentNode) return;\n } else {\n let checkCnt = 0;\n\n for (let i = 0; i < checkCSSPropertyArray.length; i++) {\n while (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(sNode) && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(sNode)) {\n if (sNode.nodeType === 1 && sNode.style[checkCSSPropertyArray[i]] === appendNode.style[checkCSSPropertyArray[i]]) {\n checkCnt++;\n }\n\n sNode = sNode.parentNode;\n }\n }\n\n if (checkCnt >= checkCSSPropertyArray.length) return;\n }\n }\n /* find text node */\n\n\n tempCon = range.startContainer;\n tempOffset = range.startOffset;\n\n if (tempCon.nodeType === 1 && tempCon.childNodes.length > 0) {\n while (tempCon && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(tempCon) && tempCon.nodeType === 1) {\n tempArray = [];\n tempChild = tempCon.childNodes;\n\n for (let i = 0, len = tempChild.length; i < len; i++) {\n tempArray.push(tempChild[i]);\n }\n\n tempCon = tempArray[tempOffset] || tempCon.nextElementSibling || tempCon.nextSibling;\n tempOffset = 0;\n }\n }\n\n const startCon = tempCon;\n const startOff = tempOffset;\n tempCon = range.endContainer;\n tempOffset = range.endOffset;\n\n if (tempCon.nodeType === 1 && tempCon.childNodes.length > 0) {\n while (tempCon && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(tempCon) && tempCon.nodeType === 1) {\n tempArray = [];\n tempChild = tempCon.childNodes;\n\n for (let i = 0, len = tempChild.length; i < len; i++) {\n tempArray.push(tempChild[i]);\n }\n\n tempCon = tempArray[tempOffset - 1] || tempArray[0] || tempCon.previousElementSibling || tempCon.previousSibling || startCon;\n }\n\n tempOffset = tempCon.textContent.length;\n }\n\n const endCon = tempCon;\n const endOff = tempOffset;\n const commonCon = range.commonAncestorContainer;\n const newNodeName = appendNode.nodeName;\n let start = {},\n end = {};\n let newNode, regExp;\n\n if (checkCSSPropertyArray) {\n regExp = '(?:;|^|\\\\s)(?:' + checkCSSPropertyArray[0];\n\n for (let i = 1; i < checkCSSPropertyArray.length; i++) {\n regExp += '|' + checkCSSPropertyArray[i];\n }\n\n regExp += ')\\\\s*:[^;]*\\\\s*(?:;|$)';\n regExp = new RegExp(regExp, 'ig');\n }\n /** tag check function*/\n\n\n const checkCss = function (vNode) {\n if (isRemoveFormat || vNode.nodeType === 3 || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(vNode)) return true;\n let style = '';\n\n if (regExp && vNode.style.cssText.length > 0) {\n style = vNode.style.cssText.replace(regExp, '').trim();\n }\n\n if (style.length > 0 || vNode.nodeName !== newNodeName) {\n if (vNode.style.cssText.length > 0) vNode.style.cssText = style;\n return true;\n }\n\n return false;\n };\n /** one line */\n\n\n if (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(commonCon) && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(commonCon)) {\n newNode = appendNode.cloneNode(false);\n\n const newRange = this._nodeChange_oneLine(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(commonCon), newNode, checkCss, startCon, startOff, endCon, endOff, isRemoveFormat, range.collapsed);\n\n start.container = newRange.startContainer;\n start.offset = newRange.startOffset;\n end.container = newRange.endContainer;\n end.offset = newRange.endOffset;\n }\n /** multi line */\n else {\n // get line nodes\n const lineNodes = this.getSelectedFormatElements();\n const endLength = lineNodes.length - 1; // startCon\n\n newNode = appendNode.cloneNode(false);\n start = this._nodeChange_startLine(lineNodes[0], newNode, checkCss, startCon, startOff, isRemoveFormat); // mid\n\n for (let i = 1; i < endLength; i++) {\n newNode = appendNode.cloneNode(false);\n\n this._nodeChange_middleLine(lineNodes[i], newNode, checkCss, isRemoveFormat);\n } // endCon\n\n\n if (endLength > 0) {\n newNode = appendNode.cloneNode(false);\n end = this._nodeChange_endLine(lineNodes[endLength], newNode, checkCss, endCon, endOff, isRemoveFormat);\n } else {\n end = start;\n }\n } // set range\n\n\n this.setRange(start.container, start.offset, end.container, end.offset);\n },\n\n /**\r\n * @description wraps text nodes of line selected text.\r\n * @param {Element} element - The node of the line that contains the selected text node.\r\n * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n * @param {function} validation - Check if the node should be stripped.\r\n * @param {Element} startCon - The startContainer property of the selection object.\r\n * @param {Number} startOff - The startOffset property of the selection object.\r\n * @param {Element} endCon - The endContainer property of the selection object.\r\n * @param {Number} endOff - The endOffset property of the selection object.\r\n * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n * @param {Boolean} collapsed - range.collapsed\r\n * @returns {{startContainer: *, startOffset: *, endContainer: *, endOffset: *}}\r\n * @private\r\n */\n _nodeChange_oneLine: function (element, newInnerNode, validation, startCon, startOff, endCon, endOff, isRemoveFormat, collapsed) {\n const el = element;\n const pNode = element.cloneNode(false);\n const isSameNode = startCon === endCon;\n let startContainer = startCon;\n let startOffset = startOff;\n let endContainer = endCon;\n let endOffset = endOff;\n let startPass = false;\n let endPass = false;\n let pCurrent, newNode, appendNode, cssText;\n\n function checkCss(vNode) {\n const regExp = new RegExp('(?:;|^|\\\\s)(?:' + cssText + 'null)\\\\s*:[^;]*\\\\s*(?:;|$)', 'ig');\n let style = '';\n\n if (regExp && vNode.style.cssText.length > 0) {\n style = regExp.test(vNode.style.cssText);\n }\n\n return !style;\n }\n\n (function recursionFunc(current, node) {\n const childNodes = current.childNodes;\n\n for (let i = 0, len = childNodes.length; i < len; i++) {\n let child = childNodes[i];\n let coverNode = node;\n let cloneNode; // startContainer\n\n if (!startPass && child === startContainer) {\n const prevNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(startContainer.nodeType === 1 ? '' : startContainer.substringData(0, startOffset));\n const textNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(startContainer.nodeType === 1 ? '' : startContainer.substringData(startOffset, endOffset > startOffset ? endOffset - startOffset : startOffset - endOffset));\n\n if (prevNode.data.length > 0) {\n node.appendChild(prevNode);\n }\n\n newNode = child;\n pCurrent = [];\n cssText = '';\n\n while (newNode !== pNode && newNode !== el && newNode !== null) {\n if (validation(newNode) && newNode.nodeType === 1 && checkCss(newNode)) {\n pCurrent.push(newNode.cloneNode(false));\n cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\n }\n\n newNode = newNode.parentNode;\n }\n\n const childNode = pCurrent.pop() || textNode;\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n newInnerNode.appendChild(childNode);\n pNode.appendChild(newInnerNode);\n startContainer = textNode;\n startOffset = 0;\n startPass = true;\n if (newNode !== textNode) newNode.appendChild(startContainer);\n if (!isSameNode) continue;\n } // endContainer\n\n\n if (!endPass && child === endContainer) {\n const afterNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(endContainer.nodeType === 1 ? '' : endContainer.substringData(endOffset, endContainer.length - endOffset));\n const textNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(isSameNode || endContainer.nodeType === 1 ? '' : endContainer.substringData(0, endOffset));\n\n if (afterNode.data.length > 0) {\n newNode = child;\n cssText = '';\n pCurrent = [];\n\n while (newNode !== pNode && newNode !== el && newNode !== null) {\n if (newNode.nodeType === 1 && checkCss(newNode)) {\n pCurrent.push(newNode.cloneNode(false));\n cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\n }\n\n newNode = newNode.parentNode;\n }\n\n cloneNode = appendNode = newNode = pCurrent.pop() || afterNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n pNode.appendChild(cloneNode);\n newNode.textContent = afterNode.data;\n }\n\n newNode = child;\n pCurrent = [];\n cssText = '';\n\n while (newNode !== pNode && newNode !== el && newNode !== null) {\n if (validation(newNode) && newNode.nodeType === 1 && checkCss(newNode)) {\n pCurrent.push(newNode.cloneNode(false));\n cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\n }\n\n newNode = newNode.parentNode;\n }\n\n const childNode = pCurrent.pop() || textNode;\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n newInnerNode.appendChild(childNode);\n endContainer = textNode;\n endOffset = textNode.data.length;\n endPass = true;\n\n if (!isRemoveFormat && collapsed) {\n newInnerNode = textNode;\n textNode.textContent = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n }\n\n if (newNode !== textNode) newNode.appendChild(endContainer);\n continue;\n } // other\n\n\n if (startPass) {\n if (child.nodeType === 1 && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) {\n recursionFunc(child, child);\n continue;\n }\n\n newNode = child;\n pCurrent = [];\n cssText = '';\n\n while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\n if (newNode.nodeType === 1 && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child) && (endPass || validation(newNode)) && checkCss(newNode)) {\n pCurrent.push(newNode.cloneNode(false));\n cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\n }\n\n newNode = newNode.parentNode;\n }\n\n const childNode = pCurrent.pop() || child;\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n if (childNode === child) {\n if (!endPass) node = newInnerNode;else node = pNode;\n } else if (endPass) {\n pNode.appendChild(childNode);\n node = newNode;\n } else {\n newInnerNode.appendChild(childNode);\n node = newNode;\n }\n }\n\n cloneNode = child.cloneNode(false);\n node.appendChild(cloneNode);\n if (child.nodeType === 1 && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) coverNode = cloneNode;\n recursionFunc(child, coverNode);\n }\n })(element, pNode);\n\n if (isRemoveFormat) {\n startContainer = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(collapsed ? _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace : newInnerNode.textContent);\n pNode.insertBefore(startContainer, newInnerNode);\n pNode.removeChild(newInnerNode);\n if (collapsed) startOffset = 1;\n } else if (collapsed) {\n startContainer = endContainer = newInnerNode;\n startOffset = 1;\n endOffset = 1;\n }\n\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeEmptyNode(pNode);\n element.parentNode.insertBefore(pNode, element);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(element);\n return {\n startContainer: startContainer,\n startOffset: startOffset,\n endContainer: isRemoveFormat || !endContainer.textContent ? startContainer : endContainer,\n endOffset: isRemoveFormat || !endContainer.textContent ? startContainer.textContent.length : endOffset\n };\n },\n\n /**\r\n * @description wraps mid lines selected text.\r\n * @param {Element} element - The node of the line that contains the selected text node.\r\n * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n * @param {function} validation - Check if the node should be stripped.\r\n * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n * @private\r\n */\n _nodeChange_middleLine: function (element, newInnerNode, validation, isRemoveFormat) {\n if (isRemoveFormat) {\n newInnerNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(element.textContent ? element.textContent : _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace);\n } else {\n (function recursionFunc(current, node) {\n const childNodes = current.childNodes;\n\n for (let i = 0, len = childNodes.length; i < len; i++) {\n let child = childNodes[i];\n let coverNode = node;\n\n if (validation(child)) {\n let cloneNode = child.cloneNode(false);\n node.appendChild(cloneNode);\n if (child.nodeType === 1 && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(element, newInnerNode);\n }\n\n element.innerHTML = '';\n element.appendChild(newInnerNode);\n },\n\n /**\r\n * @description wraps first line selected text.\r\n * @param {Element} element - The node of the line that contains the selected text node.\r\n * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n * @param {function} validation - Check if the node should be stripped.\r\n * @param {Element} startCon - The startContainer property of the selection object.\r\n * @param {Number} startOff - The startOffset property of the selection object.\r\n * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n * @returns {{container: *, offset: *}}\r\n * @private\r\n */\n _nodeChange_startLine: function (element, newInnerNode, validation, startCon, startOff, isRemoveFormat) {\n const el = element;\n const pNode = element.cloneNode(false);\n let container = startCon;\n let offset = startOff;\n let passNode = false;\n let pCurrent, newNode, appendNode;\n\n (function recursionFunc(current, node) {\n const childNodes = current.childNodes;\n\n for (let i = 0, len = childNodes.length; i < len; i++) {\n const child = childNodes[i];\n let coverNode = node;\n\n if (passNode && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) {\n if (child.nodeType === 1) {\n recursionFunc(child, child);\n continue;\n }\n\n newNode = child;\n pCurrent = [];\n\n while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\n if (newNode.nodeType === 1 && validation(newNode)) {\n pCurrent.push(newNode.cloneNode(false));\n }\n\n newNode = newNode.parentNode;\n }\n\n if (pCurrent.length > 0) {\n const childNode = pCurrent.pop();\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n newInnerNode.appendChild(childNode);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n } // startContainer\n\n\n if (!passNode && child === container) {\n const prevNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(container.nodeType === 1 ? '' : container.substringData(0, offset));\n const textNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(container.nodeType === 1 ? '' : container.substringData(offset, container.length - offset));\n\n if (prevNode.data.length > 0) {\n node.appendChild(prevNode);\n }\n\n newNode = node;\n pCurrent = [];\n\n while (newNode !== pNode && newNode !== null) {\n if (newNode.nodeType === 1 && validation(newNode)) {\n pCurrent.push(newNode.cloneNode(false));\n }\n\n newNode = newNode.parentNode;\n }\n\n const childNode = pCurrent.pop() || node;\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n if (childNode !== node) {\n newInnerNode.appendChild(childNode);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) newInnerNode.appendChild(child.cloneNode(false));\n pNode.appendChild(newInnerNode);\n container = textNode;\n offset = 0;\n passNode = true;\n node.appendChild(container);\n continue;\n }\n\n if (!passNode || validation(child)) {\n const cloneNode = child.cloneNode(false);\n node.appendChild(cloneNode);\n if (child.nodeType === 1 && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(element, pNode);\n\n if (isRemoveFormat) {\n container = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(newInnerNode.textContent);\n pNode.insertBefore(container, newInnerNode);\n pNode.removeChild(newInnerNode);\n }\n\n if (!isRemoveFormat && pNode.children.length === 0) {\n if (element.childNodes) {\n container = element.childNodes[0];\n } else {\n container = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace);\n element.appendChild(container);\n }\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeEmptyNode(pNode);\n element.parentNode.insertBefore(pNode, element);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(element);\n }\n\n return {\n container: container,\n offset: offset\n };\n },\n\n /**\r\n * @description wraps last line selected text.\r\n * @param {Element} element - The node of the line that contains the selected text node.\r\n * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n * @param {function} validation - Check if the node should be stripped.\r\n * @param {Element} endCon - The endContainer property of the selection object.\r\n * @param {Number} endOff - The endOffset property of the selection object.\r\n * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n * @returns {{container: *, offset: *}}\r\n * @private\r\n */\n _nodeChange_endLine: function (element, newInnerNode, validation, endCon, endOff, isRemoveFormat) {\n const el = element;\n const pNode = element.cloneNode(false);\n let container = endCon;\n let offset = endOff;\n let passNode = false;\n let pCurrent, newNode, appendNode;\n\n (function recursionFunc(current, node) {\n const childNodes = current.childNodes;\n\n for (let i = childNodes.length - 1; 0 <= i; i--) {\n const child = childNodes[i];\n let coverNode = node;\n\n if (passNode && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) {\n if (child.nodeType === 1) {\n recursionFunc(child, child);\n continue;\n }\n\n newNode = child;\n pCurrent = [];\n\n while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\n if (validation(newNode) && newNode.nodeType === 1) {\n pCurrent.push(newNode.cloneNode(false));\n }\n\n newNode = newNode.parentNode;\n }\n\n if (pCurrent.length > 0) {\n const childNode = pCurrent.pop();\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n newInnerNode.insertBefore(childNode, newInnerNode.firstChild);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n } // endContainer\n\n\n if (!passNode && child === container) {\n const afterNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(container.nodeType === 1 ? '' : container.substringData(offset, container.length - offset));\n const textNode = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(container.nodeType === 1 ? '' : container.substringData(0, offset));\n\n if (afterNode.data.length > 0) {\n node.insertBefore(afterNode, node.firstChild);\n }\n\n newNode = node;\n pCurrent = [];\n\n while (newNode !== pNode && newNode !== null) {\n if (validation(newNode) && newNode.nodeType === 1) {\n pCurrent.push(newNode.cloneNode(false));\n }\n\n newNode = newNode.parentNode;\n }\n\n const childNode = pCurrent.pop() || node;\n appendNode = newNode = childNode;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n appendNode = newNode;\n }\n\n if (childNode !== node) {\n newInnerNode.insertBefore(childNode, newInnerNode.firstChild);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) newInnerNode.appendChild(child.cloneNode(false));\n pNode.insertBefore(newInnerNode, pNode.firstChild);\n container = textNode;\n offset = textNode.data.length;\n passNode = true;\n node.insertBefore(container, node.firstChild);\n continue;\n }\n\n if (!passNode || validation(child)) {\n const cloneNode = child.cloneNode(false);\n node.insertBefore(cloneNode, node.firstChild);\n if (child.nodeType === 1 && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isBreak(child)) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(element, pNode);\n\n if (isRemoveFormat) {\n container = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(newInnerNode.textContent);\n offset = container.textContent.length;\n pNode.insertBefore(container, newInnerNode);\n pNode.removeChild(newInnerNode);\n }\n\n if (!isRemoveFormat && pNode.childNodes.length === 0) {\n if (element.childNodes) {\n container = element.childNodes[0];\n } else {\n container = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace);\n element.appendChild(container);\n }\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeEmptyNode(pNode);\n element.parentNode.insertBefore(pNode, element);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(element);\n }\n\n return {\n container: container,\n offset: offset\n };\n },\n\n /**\r\n * @description Execute command of command button(All Buttons except submenu and dialog)\r\n * (redo, undo, bold, underline, italic, strikethrough, subscript, superscript, removeFormat, indent, outdent, fullscreen, showBlocks, codeview, preview, print)\r\n * @param {Element} target - The element of command button\r\n * @param {String} command - Property of command button (data-value)\r\n */\n commandHandler: function (target, command) {\n switch (command) {\n case 'codeView':\n this.controllersOff();\n this.toggleCodeView();\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(target, 'on');\n break;\n\n case 'fullScreen':\n this.controllersOff();\n this.toggleFullScreen(target);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(target, 'on');\n break;\n\n case 'indent':\n case 'outdent':\n this.indent(command);\n break;\n\n case 'redo':\n case 'undo':\n this.execCommand(command, false, null);\n break;\n\n case 'removeFormat':\n this.removeFormat();\n break;\n\n case 'preview':\n case 'print':\n this.openWindowContents(command);\n break;\n\n case 'showBlocks':\n this.toggleDisplayBlocks();\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(target, 'on');\n break;\n\n case 'subscript':\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].hasClass(context.tool.superscript, 'on')) {\n this.execCommand('superscript', false, null);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.tool.superscript, 'on');\n }\n\n this.execCommand(command, false, null);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(target, 'on');\n break;\n\n case 'superscript':\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].hasClass(context.tool.subscript, 'on')) {\n this.execCommand('subscript', false, null);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.tool.subscript, 'on');\n }\n\n this.execCommand(command, false, null);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(target, 'on');\n break;\n\n default:\n // 'bold', 'underline', 'italic', 'strike'\n this.execCommand(command, false, target.getAttribute('data-value'));\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(target, 'on');\n }\n\n this.focus();\n },\n\n /**\r\n * @description Remove format of the currently selected range (IE, Edge not working)\r\n */\n removeFormat: function () {\n this.nodeChange(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement('REMOVENODE'));\n },\n\n /**\r\n * @description This method implements indentation to selected range.\r\n * Setted \"margin-left\" to \"25px\" in the top \"P\" tag of the parameter node.\r\n * @param command {String} - Separator (\"indent\" or \"outdent\")\r\n */\n indent: function (command) {\n const rangeLines = this.getSelectedFormatElements();\n let p, margin;\n\n for (let i = 0, len = rangeLines.length; i < len; i++) {\n p = rangeLines[i];\n margin = /\\d+/.test(p.style.marginLeft) ? p.style.marginLeft.match(/\\d+/)[0] * 1 : 0;\n\n if ('indent' === command) {\n margin += 25;\n } else {\n margin -= 25;\n }\n\n p.style.marginLeft = (margin < 0 ? 0 : margin) + 'px';\n }\n },\n\n /**\r\n * @description Add or remove the class name of \"body\" so that the code block is visible\r\n */\n toggleDisplayBlocks: function () {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(context.element.wysiwyg, 'sun-editor-show-block');\n },\n\n /**\r\n * @description Changes to code view or wysiwyg view\r\n */\n toggleCodeView: function () {\n const wysiwygActive = this._variable.wysiwygActive;\n const disButtons = this.codeViewDisabledButtons;\n\n for (let i = 0, len = disButtons.length; i < len; i++) {\n disButtons[i].disabled = wysiwygActive;\n }\n\n if (!wysiwygActive) {\n const code_html = context.element.code.value.trim();\n context.element.wysiwyg.innerHTML = code_html.length > 0 ? _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(code_html) : '

    ' + _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace + '

    ';\n context.element.wysiwyg.scrollTop = 0;\n context.element.code.style.display = 'none';\n context.element.wysiwyg.style.display = 'block';\n if (context.option.height === 'auto') context.element.code.style.height = '0px';\n this._variable.wysiwygActive = true;\n this.focus();\n } else {\n context.element.code.value = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertHTMLForCodeView(context.element.wysiwyg.innerHTML.trim());\n context.element.wysiwyg.style.display = 'none';\n context.element.code.style.display = 'block';\n if (context.option.height === 'auto') context.element.code.style.height = context.element.code.scrollHeight > 0 ? context.element.code.scrollHeight + 'px' : 'auto';\n this._variable.wysiwygActive = false;\n context.element.code.focus();\n }\n },\n\n /**\r\n * @description Changes to full screen or default screen\r\n * @param {Element} element - full screen button\r\n */\n toggleFullScreen: function (element) {\n if (!this._variable.isFullScreen) {\n this._variable.isFullScreen = true;\n context.element.topArea.style.position = 'fixed';\n context.element.topArea.style.top = '0';\n context.element.topArea.style.left = '0';\n context.element.topArea.style.width = '100%';\n context.element.topArea.style.height = '100%';\n context.element.topArea.style.zIndex = '2147483647';\n this._variable._bodyOverflow = _d.body.style.overflow;\n _d.body.style.overflow = 'hidden';\n this._variable._editorAreaOriginCssText = context.element.editorArea.style.cssText;\n this._variable._wysiwygOriginCssText = context.element.wysiwyg.style.cssText;\n this._variable._codeOriginCssText = context.element.code.style.cssText;\n context.element.editorArea.style.cssText = context.element.toolbar.style.cssText = context.element.wysiwyg.style.cssText = context.element.code.style.cssText = '';\n context.element.toolbar.style.width = context.element.wysiwyg.style.height = context.element.code.style.height = '100%';\n context.element.toolbar.style.position = 'relative';\n this._variable.innerHeight_fullScreen = _w.innerHeight - context.element.toolbar.offsetHeight;\n context.element.editorArea.style.height = this._variable.innerHeight_fullScreen + 'px';\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(element.firstElementChild, 'icon-expansion');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(element.firstElementChild, 'icon-reduction');\n } else {\n this._variable.isFullScreen = false;\n context.element.code.style.cssText = this._variable._codeOriginCssText;\n context.element.wysiwyg.style.cssText = this._variable._wysiwygOriginCssText;\n context.element.toolbar.style.cssText = '';\n context.element.editorArea.style.cssText = this._variable._editorAreaOriginCssText;\n context.element.topArea.style.cssText = this._variable._originCssText;\n _d.body.style.overflow = this._variable._bodyOverflow;\n\n if (context.option.stickyToolbar > -1) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.element.toolbar, 'sun-editor-sticky');\n event.onScroll_window();\n }\n\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(element.firstElementChild, 'icon-reduction');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(element.firstElementChild, 'icon-expansion');\n }\n },\n\n /**\r\n * @description Open the preview window or open the print window\r\n * @param {String} mode - 'preview' or 'print'\r\n */\n openWindowContents: function (mode) {\n const isPrint = mode === 'print';\n\n const windowObject = _w.open('', '_blank');\n\n windowObject.mimeType = 'text/html';\n windowObject.document.write('' + '' + '' + '' + '' + '' + (isPrint ? lang.toolbar.print : lang.toolbar.preview) + '' + '' + '' + '' + '
    ' + this.getContents() + '
    ' + (isPrint ? '' : '') + '' + '');\n },\n\n /**\r\n * @description Gets the current contents\r\n * @returns {Object}\r\n */\n getContents: function () {\n let contents = '';\n if (context.element.wysiwyg.innerText.trim().length === 0) return contents;\n\n if (editor._variable.wysiwygActive) {\n contents = context.element.wysiwyg.innerHTML;\n } else {\n contents = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(context.element.code.value);\n }\n\n const renderHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement('DIV');\n renderHTML.innerHTML = contents;\n const figcaptions = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildren(renderHTML, function (current) {\n return /FIGCAPTION/i.test(current.nodeName);\n });\n\n for (let i = 0, len = figcaptions.length; i < len; i++) {\n figcaptions[i].outerHTML = figcaptions[i].outerHTML.replace(/(?!^]*>)/i, '');\n }\n\n return renderHTML.innerHTML;\n }\n };\n /**\r\n * @description event function\r\n */\n\n const event = {\n _shortcutKeyCode: {\n 66: ['bold', 'B'],\n 83: ['strikethrough', 'STRIKE'],\n 85: ['underline', 'U'],\n 73: ['italic', 'I'],\n 89: ['redo'],\n 90: ['undo'],\n 219: ['outdent'],\n 221: ['indent']\n },\n _directionKeyKeyCode: new RegExp('^(?:8|13|32|46|33|34|35|36|37|38|39|40|98|100|102|104)$'),\n _changeButtonClassTagCheck: new RegExp('^(?:B|U|I|STRIKE|SUB|SUP)$'),\n _findButtonEffectTag: function () {\n const commandMap = editor.commandMap;\n const classOnCheck = this._changeButtonClassTagCheck;\n const commandMapNodes = [];\n const currentNodes = [];\n let findFormat = true,\n findFont = true,\n findSize = true,\n findA = true;\n let findB = true,\n findI = true,\n findU = true,\n findS = true;\n let cssText = '',\n nodeName = '';\n\n for (let selectionParent = editor.getSelectionNode(); !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\n if (!selectionParent) break;\n if (selectionParent.nodeType !== 1) continue;\n nodeName = selectionParent.nodeName.toUpperCase();\n currentNodes.push(nodeName);\n /** Format */\n\n if (findFormat && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(selectionParent)) {\n commandMapNodes.push('FORMAT');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap.FORMAT, nodeName);\n findFormat = false;\n continue;\n }\n /** Font */\n\n\n if (findFont && (selectionParent.style.fontFamily.length > 0 || selectionParent.face && selectionParent.face.length > 0)) {\n commandMapNodes.push('FONT');\n const selectFont = (selectionParent.style.fontFamily || selectionParent.face || lang.toolbar.font).replace(/[\"']/g, '');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap.FONT, selectFont);\n findFont = false;\n }\n /** A */\n\n\n if (findA && /^A$/.test(nodeName) && selectionParent.getAttribute('data-image-link') === null) {\n if (!context.link || editor.controllerArray[0] !== context.link.linkBtn) {\n editor.callPlugin('link', function () {\n editor.plugins.link.call_controller_linkButton.call(editor, selectionParent);\n });\n }\n\n findA = false;\n } else if (findA && context.link && editor.controllerArray[0] === context.link.linkBtn) {\n editor.controllersOff();\n }\n /** SPAN */\n\n\n if (findSize && /^SPAN$/.test(nodeName)) {\n /** font size */\n if (selectionParent.style.fontSize.length > 0) {\n commandMapNodes.push('SIZE');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap.SIZE, selectionParent.style.fontSize.match(/\\d+/)[0]);\n findSize = false;\n }\n }\n /** command map */\n\n\n cssText = selectionParent.style.cssText;\n\n if (findB && /font\\-weight\\s*:\\s*(?:\\d+|bold|bolder)(?:;|\\s|)/.test(cssText)) {\n commandMapNodes.push('B');\n findB = false;\n }\n\n if (findI && /font\\-style\\s*:\\s*(?:italic|oblique)(?:;|\\s)/.test(cssText)) {\n commandMapNodes.push('I');\n findI = false;\n }\n\n if (findU && /text\\-decoration(?:\\-line)?\\s*:\\s*underline(?:;|\\s|)/.test(cssText)) {\n commandMapNodes.push('U');\n findU = false;\n }\n\n if (findS && /text\\-decoration(?:\\-line)?\\s*:\\s*line-through(?:;|\\s|)/.test(cssText)) {\n commandMapNodes.push('STRIKE');\n findS = false;\n }\n\n commandMapNodes.push(/^STRONG$/.test(nodeName) ? 'B' : /^EM$/.test(nodeName) ? 'I' : nodeName);\n }\n /** A Tag edit controller off */\n\n\n if (findA) editor.controllersOff();\n /** toggle class on */\n\n for (let i = 0; i < commandMapNodes.length; i++) {\n nodeName = commandMapNodes[i];\n\n if (classOnCheck.test(nodeName)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(commandMap[nodeName], 'on');\n }\n }\n /** remove class, display text */\n\n\n for (let key in commandMap) {\n if (commandMapNodes.indexOf(key) > -1) continue;\n\n if (/^FONT/i.test(key)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap[key], lang.toolbar.font);\n } else if (/^SIZE$/i.test(key)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].changeTxt(commandMap[key], lang.toolbar.fontSize);\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(commandMap[key], 'on');\n }\n }\n /** save current nodes */\n\n\n editor._variable.currentNodes = currentNodes.reverse();\n /** Displays the current node structure to resizingBar */\n\n if (context.option.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\n },\n _cancelCaptionEdit: function () {\n this.setAttribute('contenteditable', false);\n this.removeEventListener('blur', event._cancelCaptionEdit);\n },\n onClick_toolbar: function (e) {\n e.preventDefault();\n e.stopPropagation();\n let target = e.target;\n let display = target.getAttribute('data-display');\n let command = target.getAttribute('data-command');\n let className = target.className;\n\n while (!command && !/editor_tool/.test(className) && !/sun-editor-id-toolbar/.test(className)) {\n target = target.parentNode;\n command = target.getAttribute('data-command');\n display = target.getAttribute('data-display');\n className = target.className;\n }\n\n if (!command && !display) return;\n if (target.disabled) return;\n /** Dialog, Submenu */\n\n if (display) {\n if (/submenu/.test(display) && (target.nextElementSibling === null || target !== editor.submenuActiveButton)) {\n editor.submenuOff();\n editor.callPlugin(command, function () {\n editor.submenuOn(target);\n });\n return;\n } else if (/dialog/.test(display)) {\n editor.callPlugin(command, function () {\n editor.plugins.dialog.open.call(editor, command, false);\n });\n }\n\n editor.submenuOff();\n return;\n }\n\n editor.submenuOff();\n /** default command */\n\n if (command) {\n editor.focus();\n editor.commandHandler(target, command);\n }\n },\n onClick_wysiwyg: function (e) {\n e.stopPropagation();\n const targetElement = e.target;\n editor.submenuOff();\n\n if (/^IMG$/i.test(targetElement.nodeName)) {\n e.preventDefault();\n editor.callPlugin('image', function () {\n const size = editor.plugins.resizing.call_controller_resize.call(editor, targetElement, 'image');\n editor.plugins.image.onModifyMode.call(editor, targetElement, size);\n\n if (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(targetElement, '.sun-editor-id-image-container')) {\n editor.plugins.image.openModify.call(editor, true);\n editor.plugins.image.update_image.call(editor);\n editor.controllersOff();\n }\n });\n return;\n }\n\n if (/sun-editor-id-iframe-inner-resizing-cover/i.test(targetElement.className)) {\n e.preventDefault();\n editor.callPlugin('video', function () {\n const iframe = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getChildElement(targetElement.parentNode, 'iframe');\n const size = editor.plugins.resizing.call_controller_resize.call(editor, iframe, 'video');\n editor.plugins.video.onModifyMode.call(editor, iframe, size);\n });\n return;\n }\n\n editor._setEditorRange();\n\n event._findButtonEffectTag();\n\n if (editor._isBalloon) {\n if (editor.getRange().collapsed) {\n event._hideToolbar();\n } else {\n event._showToolbarBalloon();\n\n return;\n }\n }\n\n const figcaption = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(targetElement, 'FIGCAPTION');\n\n if (figcaption && figcaption.getAttribute('contenteditable') !== 'ture') {\n e.preventDefault();\n figcaption.setAttribute('contenteditable', true);\n figcaption.focus();\n\n if (editor._isInline && !editor._inlineToolbarAttr.isShow) {\n event._showToolbarInline();\n\n const hideToolbar = function () {\n event._hideToolbar();\n\n _d.removeEventListener('click', hideToolbar);\n };\n\n _d.addEventListener('click', hideToolbar);\n }\n } else {\n const td = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(targetElement, _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell);\n\n if (td) {\n if (editor.controllerArray.length === 0) {\n editor.callPlugin('table', editor.plugins.table.call_controller_tableEdit.bind(editor, td));\n }\n }\n }\n\n if (userFunction.onClick) userFunction.onClick(e);\n },\n _showToolbarBalloon: function () {\n const range = editor.getRange();\n const padding = 20;\n const toolbar = context.element.toolbar;\n\n const selection = _w.getSelection();\n\n let isDirTop;\n\n if (selection.focusNode === selection.anchorNode) {\n isDirTop = selection.focusOffset < selection.anchorOffset;\n } else {\n const childNodes = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildNodes(range.commonAncestorContainer);\n isDirTop = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getArrayIndex(childNodes, selection.focusNode) < _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getArrayIndex(childNodes, selection.anchorNode);\n }\n\n let rects = range.getClientRects();\n rects = rects[isDirTop ? 0 : rects.length - 1];\n toolbar.style.display = 'block';\n const toolbarWidth = toolbar.offsetWidth;\n const toolbarHeight = toolbar.offsetHeight;\n let l = (isDirTop ? rects.left : rects.right) - context.element.topArea.offsetLeft + (_w.scrollX || _d.documentElement.scrollLeft) - toolbarWidth / 2;\n let t = (isDirTop ? rects.top - toolbarHeight - 11 : rects.bottom + 11) - context.element.topArea.offsetTop + (_w.scrollY || _d.documentElement.scrollTop);\n const overRight = l + toolbarWidth - context.element.topArea.offsetWidth;\n toolbar.style.left = (l < 0 ? padding : overRight < 0 ? l : l - overRight - padding) + 'px';\n toolbar.style.top = t + 'px';\n\n if (isDirTop) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.element._arrow, 'arrow-up');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(context.element._arrow, 'arrow-down');\n context.element._arrow.style.top = toolbarHeight + 'px';\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(context.element._arrow, 'arrow-down');\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(context.element._arrow, 'arrow-up');\n context.element._arrow.style.top = '-11px';\n }\n\n const arrow_width = context.element._arrow.offsetWidth;\n const arrow_left = toolbarWidth / 2 + (l < 0 ? l - arrow_width : overRight < 0 ? 0 : overRight + arrow_width);\n const arrow_point_width = arrow_width / 2;\n context.element._arrow.style.left = (arrow_left < arrow_point_width ? arrow_point_width : arrow_left + arrow_point_width >= toolbarWidth ? arrow_left - arrow_point_width : arrow_left) + 'px';\n },\n _showToolbarInline: function () {\n const toolbar = context.element.toolbar;\n toolbar.style.display = 'block';\n editor._inlineToolbarAttr.width = toolbar.style.width = context.option.toolbarWidth;\n editor._inlineToolbarAttr.top = toolbar.style.top = -1 - toolbar.offsetHeight + 'px';\n event.onScroll_window();\n editor._inlineToolbarAttr.isShow = true;\n },\n _hideToolbar: function () {\n context.element.toolbar.style.display = 'none';\n editor._inlineToolbarAttr.isShow = false;\n },\n onKeyDown_wysiwyg: function (e) {\n const keyCode = e.keyCode;\n const shift = e.shiftKey;\n const ctrl = e.ctrlKey || e.metaKey;\n const alt = e.altKey;\n e.stopPropagation();\n\n if (editor._isBalloon) {\n event._hideToolbar();\n }\n\n function shortcutCommand(keyCode) {\n const key = event._shortcutKeyCode[keyCode];\n if (!key) return false;\n editor.commandHandler(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getFormatElement(editor.getSelectionNode()), key[0]);\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].toggleClass(editor.commandMap[key[1]], 'on');\n return true;\n }\n /** Shortcuts */\n\n\n if (ctrl && !/^(?:16|17|18)$/.test(keyCode)) {\n if (!(!shift && keyCode === 83) && shortcutCommand(keyCode)) {\n e.preventDefault();\n return;\n }\n }\n /** default key action */\n\n\n const selectionNode = editor.getSelectionNode();\n\n switch (keyCode) {\n case 8:\n /**backspace key*/\n if (_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(selectionNode) && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode.parentNode) && selectionNode.previousSibling === null) {\n e.preventDefault();\n e.stopPropagation();\n selectionNode.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n return false;\n }\n\n break;\n\n case 9:\n /**tab key*/\n e.preventDefault();\n if (ctrl || alt) break;\n editor.controllersOff();\n let currentNode = selectionNode || editor.getSelectionNode();\n\n while (!_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(currentNode) && !_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(currentNode)) {\n currentNode = currentNode.parentNode;\n }\n\n if (currentNode && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell(currentNode)) {\n const table = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getParentElement(currentNode, 'table');\n const cells = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].getListChildren(table, _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isCell);\n let idx = shift ? _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].prevIdx(cells, currentNode) : _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].nextIdx(cells, currentNode);\n if (idx === cells.length && !shift) idx = 0;\n if (idx === -1 && shift) idx = cells.length - 1;\n const moveCell = cells[idx];\n if (!moveCell) return false;\n editor.setRange(moveCell, 0, moveCell, 0);\n break;\n }\n /** format Tag */\n\n\n const lines = editor.getSelectedFormatElements();\n\n if (!shift) {\n const tabText = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\n\n if (lines.length === 1) {\n editor.insertNode(tabText);\n editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize);\n } else {\n for (let i = 0, len = lines.length; i < len; i++) {\n lines[i].insertBefore(tabText.cloneNode(false), lines[i].firstChild);\n }\n }\n } else {\n for (let i = 0, len = lines.length, child; i < len; i++) {\n child = lines[i].firstChild;\n\n if (/^\\s{1,4}$/.test(child.textContent)) {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(child);\n } else if (/^\\s{1,4}/.test(child.textContent)) {\n child.textContent = child.textContent.replace(/^\\s{1,4}/, '');\n }\n }\n }\n\n break;\n }\n\n if (userFunction.onKeyDown) userFunction.onKeyDown(e);\n },\n onKeyUp_wysiwyg: function (e) {\n editor._setEditorRange();\n\n editor.controllersOff();\n const selectionNode = editor.getSelectionNode();\n\n if (editor._isBalloon && !editor.getRange().collapsed) {\n event._showToolbarBalloon();\n\n return;\n }\n /** when format tag deleted */\n\n\n if (e.keyCode === 8 && _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode) && context.element.wysiwyg.textContent.length === 0) {\n e.preventDefault();\n e.stopPropagation();\n const oFormatTag = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement(_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isFormatElement(editor._variable.currentNodes[0]) ? editor._variable.currentNodes[0] : 'P');\n oFormatTag.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].zeroWidthSpace;\n selectionNode.appendChild(oFormatTag);\n editor.setSelectionNode(oFormatTag);\n editor.setRange(oFormatTag, 0, oFormatTag, 0);\n return;\n }\n\n if ((_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode.parentElement) || _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isRangeFormatElement(selectionNode.parentElement)) && selectionNode.nodeType === 3) {\n editor.execCommand('formatBlock', false, _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].isWysiwygDiv(selectionNode.parentElement) ? 'P' : 'DIV');\n\n editor._setEditorRange();\n\n event._findButtonEffectTag();\n\n return;\n }\n\n if (event._directionKeyKeyCode.test(e.keyCode)) {\n event._findButtonEffectTag();\n }\n\n if (userFunction.onKeyUp) userFunction.onKeyUp(e);\n },\n onScroll_wysiwyg: function (e) {\n editor.controllersOff();\n if (userFunction.onScroll) userFunction.onScroll(e);\n },\n onDrop_wysiwyg: function (e) {\n const files = e.dataTransfer.files;\n\n if (files.length > 0) {\n e.stopPropagation();\n e.preventDefault();\n editor.focus();\n editor.callPlugin('image', function () {\n context.image.imgInputFile.files = files;\n editor.plugins.image.onRender_imgInput.call(editor);\n context.image.imgInputFile.files = null;\n });\n }\n\n if (userFunction.onDrop) userFunction.onDrop(e);\n },\n onMouseDown_resizingBar: function (e) {\n e.stopPropagation();\n editor._variable.resizeClientY = e.clientY;\n context.element.resizeBackground.style.display = 'block';\n\n function closureFunc() {\n context.element.resizeBackground.style.display = 'none';\n\n _d.removeEventListener('mousemove', event._resize_editor);\n\n _d.removeEventListener('mouseup', closureFunc);\n }\n\n _d.addEventListener('mousemove', event._resize_editor);\n\n _d.addEventListener('mouseup', closureFunc);\n },\n _resize_editor: function (e) {\n const resizeInterval = context.element.editorArea.offsetHeight + (e.clientY - editor._variable.resizeClientY);\n context.element.wysiwyg.style.height = context.element.code.style.height = (resizeInterval < editor._variable.minResizingSize ? editor._variable.minResizingSize : resizeInterval) + 'px';\n editor._variable.resizeClientY = e.clientY;\n },\n onResize_window: function () {\n if (context.element.toolbar.offsetWidth === 0) return;\n\n if (editor._variable.isFullScreen) {\n editor._variable.innerHeight_fullScreen += _w.innerHeight - context.element.toolbar.offsetHeight - editor._variable.innerHeight_fullScreen;\n context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\n } else if (editor._variable._sticky) {\n context.element.toolbar.style.width = context.element.topArea.offsetWidth - 2 + 'px';\n event.onScroll_window();\n }\n\n editor.controllersOff();\n },\n onScroll_window: function () {\n if (editor._variable.isFullScreen || context.element.toolbar.offsetWidth === 0) return;\n const element = context.element;\n const editorHeight = element.editorArea.offsetHeight;\n const editorTop = element.topArea.offsetTop - (editor._isInline ? element.toolbar.offsetHeight : 0);\n const y = (this.scrollY || _d.documentElement.scrollTop) + context.option.stickyToolbar;\n\n if (y < editorTop) {\n event._offStickyToolbar(element);\n } else if (y + editor._variable.minResizingSize >= editorHeight + editorTop) {\n if (!editor._variable._sticky) event._onStickyToolbar(element);\n element.toolbar.style.top = editorHeight + editorTop + context.option.stickyToolbar - y - editor._variable.minResizingSize + 'px';\n } else if (y >= editorTop) {\n event._onStickyToolbar(element);\n }\n },\n _onStickyToolbar: function (element) {\n if (!editor._isInline) {\n element._stickyDummy.style.height = element.toolbar.offsetHeight + 'px';\n element._stickyDummy.style.display = 'block';\n }\n\n element.toolbar.style.top = context.option.stickyToolbar + 'px';\n element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : element.toolbar.offsetWidth + 'px';\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(element.toolbar, 'sun-editor-sticky');\n editor._variable._sticky = true;\n },\n _offStickyToolbar: function (element) {\n element._stickyDummy.style.display = 'none';\n element.toolbar.style.top = editor._isInline ? editor._inlineToolbarAttr.top : '';\n element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : '';\n element.editorArea.style.marginTop = '';\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeClass(element.toolbar, 'sun-editor-sticky');\n editor._variable._sticky = false;\n },\n _codeViewAutoScroll: function () {\n context.element.code.style.height = context.element.code.scrollHeight + 'px';\n },\n onPaste_wysiwyg: function (e) {\n if (!e.clipboardData.getData) return true;\n const cleanData = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].cleanHTML(e.clipboardData.getData('text/html'));\n\n if (cleanData) {\n editor.execCommand('insertHTML', false, cleanData);\n e.stopPropagation();\n e.preventDefault();\n }\n }\n };\n /** add event listeners */\n\n /** toolbar event */\n\n context.element.toolbar.addEventListener('click', event.onClick_toolbar, false);\n context.element.toolbar.addEventListener('mousedown', function (e) {\n e.preventDefault();\n }, false);\n /** editor area */\n\n context.element.relative.addEventListener('click', editor.focus.bind(editor), false);\n context.element.wysiwyg.addEventListener('click', event.onClick_wysiwyg, false);\n context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\n context.element.wysiwyg.addEventListener('keydown', event.onKeyDown_wysiwyg, false);\n context.element.wysiwyg.addEventListener('keyup', event.onKeyUp_wysiwyg, false);\n context.element.wysiwyg.addEventListener('drop', event.onDrop_wysiwyg, false);\n context.element.wysiwyg.addEventListener('paste', event.onPaste_wysiwyg, false);\n /** code view area auto line */\n\n if (context.option.height === 'auto') context.element.code.addEventListener('keyup', event._codeViewAutoScroll, false);\n /** resizingBar */\n\n if (context.element.resizingBar) {\n if (/\\d+/.test(context.option.height)) {\n context.element.resizingBar.addEventListener('mousedown', event.onMouseDown_resizingBar, false);\n } else {\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].addClass(context.element.resizingBar, 'none-resize');\n }\n }\n /** inlineToolbar */\n\n\n if (editor._isInline) {\n context.element.wysiwyg.addEventListener('focus', event._showToolbarInline, false);\n }\n\n if (editor._isInline || editor._isBalloon) {\n context.element.wysiwyg.addEventListener('blur', event._hideToolbar, false);\n }\n /** window event */\n\n\n _w.addEventListener('resize', event.onResize_window, false);\n\n if (context.option.stickyToolbar > -1) _w.addEventListener('scroll', event.onScroll_window, false);\n /** add plugin to plugins object */\n\n if (plugins) {\n Object.keys(plugins).map(function (key) {\n let plugin = plugins[key];\n editor.plugins[plugin.name] = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].copyObj(plugin);\n });\n }\n /** User function */\n\n\n const userFunction = {\n /**\r\n * @description Event functions\r\n * @param {Object} event - Event Object\r\n */\n onScroll: null,\n onClick: null,\n onKeyDown: null,\n onKeyUp: null,\n onDrop: null,\n\n /**\r\n * @description Called when the image is uploaded or the uploaded image is deleted\r\n * @param {Element} targetImgElement - Current img element\r\n * @param {Number} index - Uploaded index\r\n * @param {Boolean} isDelete - Whether or not it was called after the delete operation\r\n */\n onImageUpload: null,\n\n /**\r\n * @description Open a notice area\r\n * @param {String} message - Notice message\r\n */\n noticeOpen: function (message) {\n editor.addModule([_plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"]]);\n _plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"].open.call(editor, message);\n },\n\n /**\r\n * @description Close a notice area\r\n */\n noticeClose: function () {\n editor.addModule([_plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"]]);\n _plugins_modules_notice__WEBPACK_IMPORTED_MODULE_1__[\"default\"].close.call(editor);\n },\n\n /**\r\n * @description Copying the contents of the editor to the original textarea\r\n */\n save: function () {\n context.element.originElement.value = editor.getContents();\n },\n\n /**\r\n * @description Gets the suneditor's context object. Contains settings, plugins, and cached element objects\r\n * @returns {Object}\r\n */\n getContext: function () {\n return context;\n },\n\n /**\r\n * @description Gets the contents of the suneditor\r\n * @returns {String}\r\n */\n getContents: function () {\n return editor.getContents();\n },\n\n /**\r\n * @description Gets uploaded images informations\r\n * @returns {Array}\r\n */\n getImagesInfo: function () {\n return editor._variable._imagesInfo;\n },\n\n /**\r\n * @description Inserts an HTML element or HTML string or plain string at the current cursor position\r\n * @param {Element|String} html - HTML Element or HTML string or plain string\r\n */\n insertHTML: function (html) {\n if (!html.nodeType || html.nodeType !== 1) {\n const template = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].createElement('template');\n template.innerHTML = html;\n html = template.firstChild || template.content.firstChild;\n }\n\n editor.insertNode(html);\n editor.focus();\n },\n\n /**\r\n * @description Change the contents of the suneditor\r\n * @param {String} contents - Contents to Input\r\n */\n setContents: function (contents) {\n if (editor._variable.wysiwygActive) {\n context.element.wysiwyg.innerHTML = _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(contents);\n } else {\n context.element.code.value = contents;\n }\n },\n\n /**\r\n * @description Add contents to the suneditor\r\n * @param {String} contents - Contents to Input\r\n */\n appendContents: function (contents) {\n if (editor._variable.wysiwygActive) {\n context.element.wysiwyg.innerHTML += _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].convertContentsForEditor(contents);\n } else {\n context.element.code.value += contents;\n }\n },\n\n /**\r\n * @description Disable the suneditor\r\n */\n disabled: function () {\n context.tool.cover.style.display = 'block';\n context.element.wysiwyg.setAttribute('contenteditable', false);\n context.element.code.setAttribute('disabled', 'disabled');\n },\n\n /**\r\n * @description Enabled the suneditor\r\n */\n enabled: function () {\n context.tool.cover.style.display = 'none';\n context.element.wysiwyg.setAttribute('contenteditable', true);\n context.element.code.removeAttribute('disabled');\n },\n\n /**\r\n * @description Show the suneditor\r\n */\n show: function () {\n const topAreaStyle = context.element.topArea.style;\n if (topAreaStyle.display === 'none') topAreaStyle.display = context.option.display;\n },\n\n /**\r\n * @description Hide the suneditor\r\n */\n hide: function () {\n context.element.topArea.style.display = 'none';\n },\n\n /**\r\n * @description Destroy the suneditor\r\n */\n destroy: function () {\n /** remove window event listeners */\n _w.removeEventListener('resize', event.onResize_window);\n\n _w.removeEventListener('scroll', event.onScroll_window);\n /** remove element */\n\n\n _util__WEBPACK_IMPORTED_MODULE_0__[\"default\"].removeItem(context.element.topArea);\n this.onScroll = null;\n this.onClick = null;\n this.onKeyDown = null;\n this.onKeyUp = null;\n this.onDrop = null;\n this.save = null;\n this.onImageUpload = null;\n this.noticeOpen = null;\n this.noticeClose = null;\n this.getContext = null;\n this.getContents = null;\n this.getImagesInfo = null;\n this.insertHTML = null;\n this.setContents = null;\n this.appendContents = null;\n this.disabled = null;\n this.enabled = null;\n this.show = null;\n this.hide = null;\n this.destroy = null;\n context = null;\n plugins = null;\n lang = null;\n }\n };\n return userFunction;\n};\n\n/* harmony default export */ __webpack_exports__[\"default\"] = (core);//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/lib/core.js.js","sources":["webpack:///./src/lib/core.js?8a8d"],"sourcesContent":["/*\r\n * wysiwyg web editor\r\n *\r\n * suneditor.js\r\n * Copyright 2017 JiHong Lee.\r\n * MIT license.\r\n */\r\n'use strict';\r\n\r\nimport util from './util';\r\nimport notice from '../plugins/modules/notice';\r\n\r\n/**\r\n * @description SunEditor core closure\r\n * @param context\r\n * @param plugins\r\n * @param lang\r\n * @returns {{save: save, getContext: getContext, getContent: getContent, setContent: setContent, appendContent: appendContent, disabled: disabled, enabled: enabled, show: show, hide: hide, destroy: destroy}}\r\n */\r\nconst core = function (context, plugins, lang) {\r\n    const _d = document;\r\n    const _w = window;\r\n    /**\r\n     * @description editor core object\r\n     * should always bind this object when registering an event in the plug-in.\r\n     */\r\n    const editor = {\r\n        /**\r\n         * @description Elements and user options parameters of the suneditor\r\n         */\r\n        context: context,\r\n\r\n        /**\r\n         * @description loaded plugins\r\n         */\r\n        plugins: {},\r\n\r\n        /**\r\n         * @description Whether the plugin is initialized\r\n         */\r\n        initPlugins: {},\r\n\r\n        /**\r\n         * @description loaded language\r\n         */\r\n        lang: lang,\r\n\r\n        /**\r\n         * @description dialog element\r\n         */\r\n        dialogForm: null,\r\n\r\n        /**\r\n         * @description submenu element\r\n         */\r\n        submenu: null,\r\n\r\n        /**\r\n         * @description active button element in submenu\r\n         */\r\n        submenuActiveButton: null,\r\n\r\n        /**\r\n         * @description The elements array to be processed unvisible when the controllersOff function is executed (resizing, link modified button, table controller)\r\n         */\r\n        controllerArray: [],\r\n\r\n        /**\r\n         * @description The functions array to be executed when the controllersOff function is executed ex) init function of table plugin\r\n         */\r\n        controllerFunction: [],\r\n\r\n        /**\r\n         * @description An array of buttons whose class name is not \"code-view-enabled\"\r\n         */\r\n        codeViewDisabledButtons: context.element.toolbar.querySelectorAll('.sun-editor-id-toolbar button:not([class~=\"code-view-enabled\"])'),\r\n\r\n        /**\r\n         * @description Is inline mode?\r\n         * @private\r\n         */\r\n        _isInline: /inline/i.test(context.option.mode),\r\n\r\n        /**\r\n         * @description Is balloon mode?\r\n         * @private\r\n         */\r\n        _isBalloon: /balloon/i.test(context.option.mode),\r\n\r\n        /**\r\n         * @description Required value when using inline mode to sticky toolbar\r\n         * @private\r\n         */\r\n        _inlineToolbarAttr: {width: 0, height: 0, isShow: false},\r\n\r\n        /**\r\n         * @description An user event function when image uploaded success or remove image\r\n         * @private\r\n         */\r\n        _imageUpload: function (targetImgElement, index, isDelete) {\r\n            if (userFunction.onImageUpload) userFunction.onImageUpload(targetImgElement, index, isDelete);\r\n        },\r\n\r\n        /**\r\n         * @description Elements that need to change text or className for each selection change\r\n         * @property {Element} FORMAT - format button\r\n         * @property {Element} FONT - font family button\r\n         * @property {Element} SIZE - font size button\r\n         * @property {Element} B - bold button\r\n         * @property {Element} U - underline button\r\n         * @property {Element} I - italic button\r\n         * @property {Element} STRIKE - strike button\r\n         * @property {Element} SUB - subscript button\r\n         * @property {Element} SUP - superscript button\r\n         */\r\n        commandMap: {\r\n            FORMAT: context.tool.format,\r\n            FONT: context.tool.font,\r\n            SIZE: context.tool.fontSize,\r\n            B: context.tool.bold,\r\n            U: context.tool.underline,\r\n            I: context.tool.italic,\r\n            STRIKE: context.tool.strike,\r\n            SUB: context.tool.subscript,\r\n            SUP: context.tool.superscript\r\n        },\r\n\r\n        /**\r\n         * @description Variables used internally in editor operation\r\n         * @property {(Element|null)} selectionNode - Contains selection node\r\n         * @property {(Object|null)} range - The current range object\r\n         * @property {Boolean} wysiwygActive - The wysiwyg frame or code view state\r\n         * @property {Boolean} isFullScreen - State of full screen\r\n         * @property {Number} innerHeight_fullScreen - InnerHeight in editor when in full screen\r\n         * @property {Number} resizeClientY - Remember the vertical size of the editor before resizing the editor (Used when calculating during resize operation)\r\n         * @property {Number} tabSize - Indented size when tab button clicked (4)\r\n         * @property {Number} minResizingSize - Minimum size of editing area when resized (65)\r\n         * @property {Array} currentNodes -  An array of the current cursor's node structure\r\n         * @private\r\n         */\r\n        _variable: {\r\n            selectionNode: null,\r\n            range: null,\r\n            wysiwygActive: true,\r\n            isFullScreen: false,\r\n            innerHeight_fullScreen: 0,\r\n            resizeClientY: 0,\r\n            tabSize: 4,\r\n            minResizingSize: 65,\r\n            currentNodes: [],\r\n            _originCssText: context.element.topArea.style.cssText,\r\n            _bodyOverflow: '',\r\n            _editorAreaOriginCssText: '',\r\n            _wysiwygOriginCssText: '',\r\n            _codeOriginCssText: '',\r\n            _sticky: false,\r\n            _imagesInfo: [],\r\n            _imageIndex: 0\r\n        },\r\n\r\n        /**\r\n         * @description If the plugin is not added, add the plugin and call the 'add' function.\r\n         * If the plugin is added call callBack function.\r\n         * @param {String} pluginName - The name of the plugin to call\r\n         * @param {function} callBackFunction - Function to be executed immediately after module call\r\n         */\r\n        callPlugin: function (pluginName, callBackFunction) {\r\n            if (!this.plugins[pluginName]) {\r\n                throw Error('[SUNEDITOR.core.callPlugin.fail] The called plugin does not exist or is in an invalid format. (pluginName:\"' + pluginName + '\")');\r\n            } else if (!this.initPlugins[pluginName]){\r\n                this.plugins[pluginName].add(this, this.plugins[pluginName].buttonElement);\r\n                this.initPlugins[pluginName] = true;\r\n            }\r\n                \r\n            callBackFunction();\r\n        },\r\n\r\n        /**\r\n         * @description If the module is not added, add the module and call the 'add' function\r\n         * @param {Array} moduleArray - module object's Array [dialog, resizing]\r\n         */\r\n        addModule: function (moduleArray) {\r\n            let moduleName = '';\r\n            for (let i = 0, len = moduleArray.length; i < len; i++) {\r\n                moduleName = moduleArray[i].name;\r\n                if (!this.plugins[moduleName]) {\r\n                    this.plugins[moduleName] = util.copyObj(moduleArray[i]);\r\n                    this.plugins[moduleName].add(this);\r\n                }\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Enabled submenu\r\n         * @param {Element} element - Submenu element to call\r\n         */\r\n        submenuOn: function (element) {\r\n            const submenuName = element.getAttribute('data-command');\r\n            if (this.plugins[submenuName].on) this.plugins[submenuName].on.call(this);\r\n\r\n            this.submenu = element.nextElementSibling;\r\n            this.submenu.style.display = 'block';\r\n            util.addClass(element, 'on');\r\n            this.submenuActiveButton = element;\r\n\r\n            const overLeft = this.context.element.toolbar.offsetWidth - (element.parentElement.offsetLeft + this.submenu.offsetWidth);\r\n            if (overLeft < 0) this.submenu.style.left = overLeft + 'px';\r\n            else this.submenu.style.left = '1px';\r\n        },\r\n\r\n        /**\r\n         * @description Disable submenu\r\n         */\r\n        submenuOff: function () {\r\n            if (this.submenu) {\r\n                this.submenu.style.display = 'none';\r\n                this.submenu = null;\r\n                util.removeClass(this.submenuActiveButton, 'on');\r\n                this.submenuActiveButton = null;\r\n            }\r\n\r\n            this.controllersOff();\r\n        },\r\n\r\n        /**\r\n         * @description Disable controller in editor area (link button, image resize button)\r\n         */\r\n        controllersOff: function () {\r\n            const len = this.controllerArray.length;\r\n            const fLen = this.controllerFunction.length;\r\n\r\n            if (len > 0) {\r\n                for (let i = 0; i < len; i++) {\r\n                    this.controllerArray[i].style.display = 'none';\r\n                }\r\n                this.controllerArray = [];\r\n            }\r\n\r\n            if (fLen > 0) {\r\n                for (let i = 0; i < fLen; i++) {\r\n                    this.controllerFunction[i]();\r\n                }\r\n                this.controllerFunction = [];\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description javascript execCommand\r\n         * @param {String} command - javascript execCommand function property\r\n         * @param {Boolean} showDefaultUI - javascript execCommand function property\r\n         * @param {String} value - javascript execCommand function property\r\n         */\r\n        execCommand: function (command, showDefaultUI, value) {\r\n            _d.execCommand(command, showDefaultUI, (command === 'formatBlock' ? '<' + value + '>' : value));\r\n        },\r\n\r\n        /**\r\n         * @description Focus to wysiwyg area\r\n         */\r\n        focus: function () {\r\n            if (context.element.wysiwyg.style.display === 'none') return;\r\n\r\n            const caption = util.getParentElement(this.getSelectionNode(), 'figcaption');\r\n            if (caption) {\r\n                caption.focus();\r\n            } else {\r\n                context.element.wysiwyg.focus();\r\n            }\r\n\r\n            this._setEditorRange();\r\n            event._findButtonEffectTag();\r\n        },\r\n\r\n        /**\r\n         * @description Saving the range object and the currently selected node of editor\r\n         * @private\r\n         */\r\n        _setEditorRange: function () {\r\n            const selection = _w.getSelection();\r\n            let range = null;\r\n\r\n            if (selection.rangeCount > 0) {\r\n                range = selection.getRangeAt(0);\r\n            }\r\n            else {\r\n                range = this._createDefaultRange();\r\n            }\r\n\r\n            this._variable.range = range;\r\n\r\n            if (range.collapsed) {\r\n                this.setSelectionNode(range.commonAncestorContainer);\r\n            } else {\r\n                this.setSelectionNode(selection.extentNode || selection.anchorNode);\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Return the range object of editor's first child node\r\n         * @returns {Object}\r\n         * @private\r\n         */\r\n        _createDefaultRange: function () {\r\n            const range = _d.createRange();\r\n            range.setStart(context.element.wysiwyg.firstChild, 0);\r\n            range.setEnd(context.element.wysiwyg.firstChild, 0);\r\n            return range;\r\n        },\r\n\r\n        /**\r\n         * @description Set current editor's range object\r\n         * @param {Element} startCon - The startContainer property of the selection object.\r\n         * @param {Number} startOff - The startOffset property of the selection object.\r\n         * @param {Element} endCon - The endContainer property of the selection object.\r\n         * @param {Element} endOff - The endOffset property of the selection object.\r\n         */\r\n        setRange: function (startCon, startOff, endCon, endOff) {\r\n            const range = _d.createRange();\r\n            range.setStart(startCon, startOff);\r\n            range.setEnd(endCon, endOff);\r\n\r\n            const selection = _w.getSelection();\r\n\r\n            if (selection.removeAllRanges) {\r\n                selection.removeAllRanges();\r\n            }\r\n\r\n            selection.addRange(range);\r\n            this._variable.range = range;\r\n            this._setEditorRange();\r\n        },\r\n\r\n        /**\r\n         * @description Get current editor's range object\r\n         * @returns {Object}\r\n         */\r\n        getRange: function () {\r\n            return this._variable.range || this._createDefaultRange();\r\n        },\r\n\r\n        /**\r\n         * @description Set the selected node. (Used by getSelectionNode function)\r\n         * @param {Node} node - node object\r\n         */\r\n        setSelectionNode: function (node) {\r\n            this._variable.selectionNode = node;\r\n        },\r\n\r\n        /**\r\n         * @description Get current select node\r\n         * @returns {Node}\r\n         */\r\n        getSelectionNode: function () {\r\n            if (this._variable.selectionNode) {\r\n                return this._variable.selectionNode;\r\n            }\r\n\r\n            return context.element.wysiwyg.firstChild;\r\n        },\r\n\r\n        /**\r\n         * @description Returns a \"formatElement\"(P, DIV, H[1-6], LI) array from the currently selected range.\r\n         * @returns {Array}\r\n         */\r\n        getSelectedFormatElements: function () {\r\n            let range = this.getRange();\r\n\r\n            if (util.isWysiwygDiv(range.startContainer)) {\r\n                const children = context.element.wysiwyg.children;\r\n                this.setRange(children[0], 0, children[children.length - 1], children[children.length - 1].textContent.length);\r\n                range = this.getRange();\r\n            }\r\n\r\n            const startCon = range.startContainer;\r\n            const endCon = range.endContainer;\r\n            const commonCon = range.commonAncestorContainer;\r\n            const rangeFormatElements = [];\r\n\r\n            if (!util.isWysiwygDiv(commonCon) && !util.isRangeFormatElement(commonCon)) return [util.getFormatElement(commonCon)];\r\n\r\n            // get line nodes\r\n            const lineNodes = util.getListChildren(commonCon, function (current) {\r\n                return util.isFormatElement(current);\r\n            });\r\n\r\n            if (startCon === endCon) return lineNodes[0];\r\n\r\n            let startLine = util.getFormatElement(startCon);\r\n            let endLine = util.getFormatElement(endCon);\r\n            let startIdx = 0;\r\n            let endIdx = 0;\r\n\r\n            for (let i = 0, len = lineNodes.length; i < len; i++) {\r\n                if (startLine === lineNodes[i]) {\r\n                    startIdx = i;\r\n                    continue;\r\n                }\r\n                if (endLine === lineNodes[i]) {\r\n                    endIdx = i;\r\n                    break;\r\n                }\r\n            }\r\n\r\n            for (let i = startIdx; i <= endIdx; i++) {\r\n                rangeFormatElements.push(lineNodes[i]);\r\n            }\r\n\r\n            return rangeFormatElements;\r\n        },\r\n\r\n        /**\r\n         * @description Returns a \"rangeFormatElement\"(blockquote, TABLE, TR, TD, OL, UL, PRE) array from the currently selected range.\r\n         * @returns {Array}\r\n         */\r\n        getSelectedRangeFormatElements: function () {\r\n            let range = this.getRange();\r\n\r\n            if (util.isWysiwygDiv(range.startContainer)) {\r\n                const children = context.element.wysiwyg.children;\r\n                this.setRange(children[0], 0, children[children.length - 1], children[children.length - 1].textContent.length);\r\n                range = this.getRange();\r\n            }\r\n\r\n            const startCon = range.startContainer;\r\n            const endCon = range.endContainer;\r\n            const commonCon = range.commonAncestorContainer;\r\n            const rangeFormatElements = [];\r\n\r\n            if (util.isRangeFormatElement(commonCon)) return [commonCon];\r\n            if (!util.isWysiwygDiv(commonCon)) {\r\n                const el = util.getRangeFormatElement(commonCon);\r\n                return el ? [el] : [];\r\n            }\r\n\r\n            // get range Elements\r\n            const rangeElements = util.getListChildren(commonCon, function (current) {\r\n                return util.isRangeFormatElement(current);\r\n            });\r\n\r\n            if (startCon === endCon) return rangeElements[0];\r\n\r\n            let startLine = util.getRangeFormatElement(startCon);\r\n            let endLine = util.getRangeFormatElement(endCon);\r\n            let startIdx = 0;\r\n            let endIdx = 0;\r\n\r\n            for (let i = 0, len = rangeElements.length; i < len; i++) {\r\n                if (startLine === rangeElements[i]) {\r\n                    startIdx = i;\r\n                    continue;\r\n                }\r\n                if (endLine === rangeElements[i]) {\r\n                    endIdx = i;\r\n                    break;\r\n                }\r\n            }\r\n\r\n            for (let i = startIdx; i <= endIdx; i++) {\r\n                if (rangeElements[i]) rangeFormatElements.push(rangeElements[i]);\r\n            }\r\n\r\n            return rangeFormatElements;\r\n        },\r\n\r\n        /**\r\n         * @description Determine if this offset is the edge offset of container\r\n         * @param {Object} container - The container property of the selection object.\r\n         * @param {Number} offset - The offset property of the selection object.\r\n         * @returns {Boolean}\r\n         */\r\n        isEdgePoint: function (container, offset) {\r\n            return (offset === 0) || (offset === container.nodeValue.length);\r\n        },\r\n\r\n        /**\r\n         * @description Show loading box\r\n         */\r\n        showLoading: function () {\r\n            context.element.loading.style.display = 'block';\r\n        },\r\n\r\n        /**\r\n         * @description Close loading box\r\n         */\r\n        closeLoading: function () {\r\n            context.element.loading.style.display = 'none';\r\n        },\r\n\r\n        /**\r\n         * @description Append format element to sibling node of argument element.\r\n         * If the \"formatNodeName\" argument value is present, the tag of that argument value is inserted,\r\n         * If not, the currently selected format tag is inserted.\r\n         * @param {Element} element - Insert as siblings of that element\r\n         * @param {String|null} formatNodeName - Node name to be inserted\r\n         * @returns {Element}\r\n         */\r\n        appendFormatTag: function (element, formatNodeName) {\r\n            const formatEl = util.getRangeFormatElement(element) || util.getFormatElement(element);\r\n            const currentFormatEl = util.getFormatElement(this.getSelectionNode());\r\n            const oFormatName = formatNodeName ? formatNodeName : util.isFormatElement(currentFormatEl) ? currentFormatEl.nodeName : 'P';\r\n            const oFormat = util.createElement(oFormatName);\r\n            oFormat.innerHTML = util.zeroWidthSpace;\r\n\r\n            if (util.isCell(formatEl)) formatEl.insertBefore(oFormat, element.nextElementSibling);\r\n            else formatEl.parentNode.insertBefore(oFormat, formatEl.nextElementSibling);\r\n\r\n            return oFormat;\r\n        },\r\n\r\n        /**\r\n         * @description Delete selected node and insert argument value node\r\n         * @param {Element} oNode - Node to be inserted\r\n         * @param {(Element|null)} rightNode - If the node exists, it is inserted after the node\r\n         */\r\n        insertNode: function (oNode, rightNode) {\r\n            const range = this.getRange();\r\n            let parentNode = null;\r\n\r\n            if (!rightNode) {\r\n                const startCon = range.startContainer;\r\n                const startOff = range.startOffset;\r\n                const endCon = range.endContainer;\r\n                const endOff = range.endOffset;\r\n                const commonCon = range.commonAncestorContainer;\r\n\r\n                parentNode = startCon;\r\n                if (startCon.nodeType === 3) {\r\n                    parentNode = startCon.parentNode;\r\n                }\r\n\r\n                /** No Select range node */\r\n                if (range.collapsed) {\r\n                    if (commonCon.nodeType === 3) {\r\n                        rightNode = commonCon.splitText(endOff);\r\n                    }\r\n                    else {\r\n                        if (parentNode.lastChild !== null && util.isBreak(parentNode.lastChild)) {\r\n                            parentNode.removeChild(parentNode.lastChild);\r\n                        }\r\n                        rightNode = null;\r\n                    }\r\n                }\r\n                /** Select range nodes */\r\n                else {\r\n                    const isSameContainer = startCon === endCon;\r\n\r\n                    if (isSameContainer) {\r\n                        if (this.isEdgePoint(endCon, endOff)) rightNode = endCon.nextSibling;\r\n                        else rightNode = endCon.splitText(endOff);\r\n\r\n                        let removeNode = startCon;\r\n                        if (!this.isEdgePoint(startCon, startOff)) removeNode = startCon.splitText(startOff);\r\n\r\n                        parentNode.removeChild(removeNode);\r\n                    }\r\n                    else {\r\n                        this.removeNode();\r\n                        parentNode = commonCon;\r\n                        rightNode = endCon;\r\n\r\n                        while (rightNode.parentNode !== commonCon) {\r\n                            rightNode = rightNode.parentNode;\r\n                        }\r\n                    }\r\n                }\r\n            }\r\n            else {\r\n                parentNode = rightNode.parentNode;\r\n                rightNode = rightNode.nextSibling;\r\n            }\r\n\r\n            try {\r\n                parentNode.insertBefore(oNode, rightNode);\r\n            } catch (e) {\r\n                parentNode.appendChild(oNode);\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Delete the currently selected node\r\n         */\r\n        removeNode: function () {\r\n            const range = this.getRange();\r\n\r\n            if (range.deleteContents) {\r\n                range.deleteContents();\r\n                return;\r\n            }\r\n\r\n            const startCon = range.startContainer;\r\n            const startOff = range.startOffset;\r\n            const endCon = range.endContainer;\r\n            const endOff = range.endOffset;\r\n            const commonCon = range.commonAncestorContainer;\r\n\r\n            let beforeNode = null;\r\n            let afterNode = null;\r\n\r\n            const childNodes = util.getListChildNodes(commonCon);\r\n            let startIndex = util.getArrayIndex(childNodes, startCon);\r\n            let endIndex = util.getArrayIndex(childNodes, endCon);\r\n\r\n            for (let i = startIndex + 1, startNode = startCon; i >= 0; i--) {\r\n                if (childNodes[i] === startNode.parentNode && childNodes[i].firstChild === startNode && startOff === 0) {\r\n                    startIndex = i;\r\n                    startNode = startNode.parentNode;\r\n                }\r\n            }\r\n\r\n            for (let i = endIndex - 1, endNode = endCon; i > startIndex; i--) {\r\n                if (childNodes[i] === endNode.parentNode && childNodes[i].nodeType === 1) {\r\n                    childNodes.splice(i, 1);\r\n                    endNode = endNode.parentNode;\r\n                    --endIndex;\r\n                }\r\n            }\r\n\r\n            for (let i = startIndex; i <= endIndex; i++) {\r\n                const item = childNodes[i];\r\n\r\n                if (item.length === 0 || (item.nodeType === 3 && item.data === undefined)) {\r\n                    util.removeItem(item);\r\n                    continue;\r\n                }\r\n\r\n                if (item === startCon) {\r\n                    if (startCon.nodeType === 1) {\r\n                        beforeNode = util.createTextNode(startCon.textContent);\r\n                    } else {\r\n                        beforeNode = util.createTextNode(startCon.substringData(0, startOff));\r\n                    }\r\n\r\n                    if (beforeNode.length > 0) {\r\n                        startCon.data = beforeNode.data;\r\n                    } else {\r\n                        util.removeItem(startCon);\r\n                    }\r\n\r\n                    continue;\r\n                }\r\n\r\n                if (item === endCon) {\r\n                    if (endCon.nodeType === 1) {\r\n                        afterNode = util.createTextNode(endCon.textContent);\r\n                    } else {\r\n                        afterNode = util.createTextNode(endCon.substringData(endOff, (endCon.length - endOff)));\r\n                    }\r\n\r\n                    if (afterNode.length > 0) {\r\n                        endCon.data = afterNode.data;\r\n                    } else {\r\n                        util.removeItem(endCon);\r\n                    }\r\n\r\n                    continue;\r\n                }\r\n\r\n                util.removeItem(item);\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description appended all selected format Element to the argument element and insert\r\n         * @param {Element} wrapTag - Element of wrap the arguments\r\n         */\r\n        wrapToTags: function (wrapTag) {\r\n            const range = this.getRange();\r\n            const rangeLines = this.getSelectedFormatElements();\r\n\r\n            if (!rangeLines) {\r\n                const inner = util.createElement(util.isCell(this.getSelectionNode()) ? 'DIV' : 'P');\r\n                inner.innerHTML = util.zeroWidthSpace;\r\n                wrapTag.appendChild(inner);\r\n                this.getSelectionNode().appendChild(wrapTag);\r\n                return;\r\n            }\r\n\r\n            let last  = rangeLines[rangeLines.length - 1];\r\n            let standTag, beforeTag, pElement;\r\n\r\n            if (util.isRangeFormatElement(last) || util.isFormatElement(last)) {\r\n                standTag = last;\r\n            } else {\r\n                standTag = util.getRangeFormatElement(last) || util.getFormatElement(last);\r\n            }\r\n\r\n            if (util.isCell(standTag)) {\r\n                beforeTag = null;\r\n                pElement = standTag;\r\n            } else {\r\n                beforeTag = standTag.nextSibling;\r\n                pElement = standTag.parentNode;\r\n            }\r\n\r\n            let listParent = null;\r\n            let line = null;\r\n            let prevNodeName = '';\r\n            \r\n            for (let i = 0, len = rangeLines.length; i < len; i++) {\r\n                line = rangeLines[i];\r\n\r\n                if (/^LI$/i.test(line.nodeName)) {\r\n                    if (listParent === null || !/^LI$/i.test(prevNodeName)) {\r\n                        listParent = util.createElement(line.parentNode.nodeName);\r\n                    }\r\n\r\n                    listParent.appendChild(line);\r\n                    if (i === len - 1 || !/^LI$/i.test(rangeLines[i + 1].nodeName)) wrapTag.appendChild(listParent);\r\n                }\r\n                else {\r\n                    wrapTag.appendChild(line);\r\n                }\r\n\r\n                prevNodeName = line.nodeName;\r\n            }\r\n\r\n            pElement.insertBefore(wrapTag, beforeTag);\r\n            if (!range.collapsed && (util.isRangeFormatElement(range.startContainer) || util.isRangeFormatElement(range.endContainer))) util.removeEmptyNode(pElement);\r\n        },\r\n\r\n        /**\r\n         * @description Add the node received as an argument value to the selected area.\r\n         * 1. When there is the same css value node in the selection area, the tag is stripped.\r\n         * 2. If there is another css value other thanCss attribute values received as arguments on the node, removed only Css attribute values received as arguments\r\n         * 3. If you pass an element whose node name is \"removenode\" as an argument value, it performs a type removal operation. ex) nodeChange(document.createElement('removenode'))\r\n         * @param {Element} appendNode - The dom that will wrap the selected text area\r\n         * @param {Array} checkCSSPropertyArray - The css attribute name Array to check (['font-size'], ['font-family', 'background-color', 'border']...])\r\n         */\r\n        nodeChange: function (appendNode, checkCSSPropertyArray) {\r\n            const range = this.getRange();\r\n            const isRemoveFormat = /removenode/i.test(appendNode.nodeName);\r\n            let tempCon, tempOffset, tempChild, tempArray;\r\n\r\n            /* checked same style property */\r\n            if (range.startContainer === range.endContainer) {\r\n                let sNode = range.startContainer;\r\n                if (isRemoveFormat) {\r\n                    if (util.getFormatElement(sNode) === sNode.parentNode) return;\r\n                } else {\r\n                    let checkCnt = 0;\r\n\r\n                    for (let i = 0; i < checkCSSPropertyArray.length; i++) {\r\n                        while(!util.isFormatElement(sNode) && !util.isWysiwygDiv(sNode)) {\r\n                            if (sNode.nodeType === 1 && sNode.style[checkCSSPropertyArray[i]] === appendNode.style[checkCSSPropertyArray[i]]) {\r\n                                checkCnt++;\r\n                            }\r\n                            sNode = sNode.parentNode;\r\n                        }\r\n                    }\r\n    \r\n                    if (checkCnt >= checkCSSPropertyArray.length) return;\r\n                }\r\n            }\r\n\r\n            /* find text node */\r\n            tempCon = range.startContainer;\r\n            tempOffset = range.startOffset;\r\n\r\n            if (tempCon.nodeType === 1 && tempCon.childNodes.length > 0) {\r\n                while (tempCon && !util.isBreak(tempCon) && tempCon.nodeType === 1) {\r\n                    tempArray = [];\r\n                    tempChild = tempCon.childNodes;\r\n                    for (let i = 0, len = tempChild.length; i < len; i++) {\r\n                        tempArray.push(tempChild[i]);\r\n                    }\r\n                    tempCon = tempArray[tempOffset] || tempCon.nextElementSibling || tempCon.nextSibling;\r\n                    tempOffset = 0;\r\n                }\r\n            }\r\n\r\n            const startCon = tempCon;\r\n            const startOff = tempOffset;\r\n\r\n            tempCon = range.endContainer;\r\n            tempOffset = range.endOffset;\r\n            if (tempCon.nodeType === 1 && tempCon.childNodes.length > 0) {\r\n                while (tempCon && !util.isBreak(tempCon) && tempCon.nodeType === 1) {\r\n                    tempArray = [];\r\n                    tempChild = tempCon.childNodes;\r\n                    for (let i = 0, len = tempChild.length; i < len; i++) {\r\n                        tempArray.push(tempChild[i]);\r\n                    }\r\n                    tempCon = tempArray[tempOffset - 1] || tempArray[0] || tempCon.previousElementSibling || tempCon.previousSibling || startCon;\r\n                }\r\n                tempOffset = tempCon.textContent.length;\r\n            }\r\n\r\n            const endCon = tempCon;\r\n            const endOff = tempOffset;\r\n            const commonCon = range.commonAncestorContainer;\r\n            const newNodeName = appendNode.nodeName;\r\n\r\n            let start = {}, end = {};\r\n            let newNode, regExp;\r\n\r\n            if (checkCSSPropertyArray) {\r\n                regExp = '(?:;|^|\\\\s)(?:' + checkCSSPropertyArray[0];\r\n                for (let i = 1; i < checkCSSPropertyArray.length; i++) {\r\n                    regExp += '|' + checkCSSPropertyArray[i];\r\n                }\r\n                regExp += ')\\\\s*:[^;]*\\\\s*(?:;|$)';\r\n                regExp = new RegExp(regExp, 'ig');\r\n            }\r\n\r\n            /** tag check function*/\r\n            const checkCss = function (vNode) {\r\n                if (isRemoveFormat || vNode.nodeType === 3 || util.isBreak(vNode)) return true;\r\n\r\n                let style = '';\r\n                if (regExp && vNode.style.cssText.length > 0) {\r\n                    style = vNode.style.cssText.replace(regExp, '').trim();\r\n                }\r\n\r\n                if (style.length > 0 || vNode.nodeName !== newNodeName) {\r\n                    if (vNode.style.cssText.length > 0) vNode.style.cssText = style;\r\n                    return true;\r\n                }\r\n\r\n                return false;\r\n            };\r\n\r\n            /** one line */\r\n            if (!util.isWysiwygDiv(commonCon) && !util.isRangeFormatElement(commonCon)) {\r\n                newNode = appendNode.cloneNode(false);\r\n                const newRange = this._nodeChange_oneLine(util.getFormatElement(commonCon), newNode, checkCss, startCon, startOff, endCon, endOff, isRemoveFormat, range.collapsed);\r\n\r\n                start.container = newRange.startContainer;\r\n                start.offset = newRange.startOffset;\r\n                end.container = newRange.endContainer;\r\n                end.offset = newRange.endOffset;\r\n            }\r\n            /** multi line */\r\n            else {\r\n                // get line nodes\r\n                const lineNodes = this.getSelectedFormatElements();\r\n                const endLength = lineNodes.length - 1;\r\n\r\n                // startCon\r\n                newNode = appendNode.cloneNode(false);\r\n                start = this._nodeChange_startLine(lineNodes[0], newNode, checkCss, startCon, startOff, isRemoveFormat);\r\n\r\n                // mid\r\n                for (let i = 1; i < endLength; i++) {\r\n                    newNode = appendNode.cloneNode(false);\r\n                    this._nodeChange_middleLine(lineNodes[i], newNode, checkCss, isRemoveFormat);\r\n                }\r\n\r\n                // endCon\r\n                if (endLength > 0) {\r\n                    newNode = appendNode.cloneNode(false);\r\n                    end = this._nodeChange_endLine(lineNodes[endLength], newNode, checkCss, endCon, endOff, isRemoveFormat);\r\n                } else {\r\n                    end = start;\r\n                }\r\n            }\r\n\r\n            // set range\r\n            this.setRange(start.container, start.offset, end.container, end.offset);\r\n        },\r\n\r\n        /**\r\n         * @description wraps text nodes of line selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Element} startCon - The startContainer property of the selection object.\r\n         * @param {Number} startOff - The startOffset property of the selection object.\r\n         * @param {Element} endCon - The endContainer property of the selection object.\r\n         * @param {Number} endOff - The endOffset property of the selection object.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @param {Boolean} collapsed - range.collapsed\r\n         * @returns {{startContainer: *, startOffset: *, endContainer: *, endOffset: *}}\r\n         * @private\r\n         */\r\n        _nodeChange_oneLine: function (element, newInnerNode, validation, startCon, startOff, endCon, endOff, isRemoveFormat, collapsed) {\r\n            const el = element;\r\n            const pNode = element.cloneNode(false);\r\n            const isSameNode = startCon === endCon;\r\n            let startContainer = startCon;\r\n            let startOffset = startOff;\r\n            let endContainer = endCon;\r\n            let endOffset = endOff;\r\n            let startPass = false;\r\n            let endPass = false;\r\n            let pCurrent, newNode, appendNode, cssText;\r\n\r\n            function checkCss (vNode) {\r\n                const regExp = new RegExp('(?:;|^|\\\\s)(?:' + cssText + 'null)\\\\s*:[^;]*\\\\s*(?:;|$)', 'ig');\r\n                let style = '';\r\n\r\n                if (regExp && vNode.style.cssText.length > 0) {\r\n                    style = regExp.test(vNode.style.cssText);\r\n                }\r\n            \r\n                return !style;\r\n            }\r\n\r\n            (function recursionFunc(current, node) {\r\n                const childNodes = current.childNodes;\r\n\r\n                for (let i = 0, len = childNodes.length; i < len; i++) {\r\n                    let child = childNodes[i];\r\n                    let coverNode = node;\r\n                    let cloneNode;\r\n\r\n                    // startContainer\r\n                    if (!startPass && child === startContainer) {\r\n                        const prevNode = util.createTextNode(startContainer.nodeType === 1 ? '' : startContainer.substringData(0, startOffset));\r\n                        const textNode = util.createTextNode(startContainer.nodeType === 1 ? '' : startContainer.substringData(startOffset, (endOffset > startOffset ? endOffset - startOffset : startOffset - endOffset)));\r\n\r\n                        if (prevNode.data.length > 0) {\r\n                            node.appendChild(prevNode);\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        cssText = '';\r\n                        while (newNode !== pNode && newNode !== el && newNode !== null) {\r\n                            if (validation(newNode) && newNode.nodeType === 1 && checkCss(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                                cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || textNode;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        newInnerNode.appendChild(childNode);\r\n                        pNode.appendChild(newInnerNode);\r\n\r\n                        startContainer = textNode;\r\n                        startOffset = 0;\r\n                        startPass = true;\r\n\r\n                        if (newNode !== textNode) newNode.appendChild(startContainer);\r\n                        if (!isSameNode) continue;\r\n                    }\r\n\r\n                    // endContainer\r\n                    if (!endPass && child === endContainer) {\r\n                        const afterNode = util.createTextNode(endContainer.nodeType === 1 ? '' : endContainer.substringData(endOffset, (endContainer.length - endOffset)));\r\n                        const textNode = util.createTextNode(isSameNode || endContainer.nodeType === 1 ? '' : endContainer.substringData(0, endOffset));\r\n\r\n                        if (afterNode.data.length > 0) {\r\n                            newNode = child;\r\n                            cssText = '';\r\n                            pCurrent = [];\r\n                            while (newNode !== pNode && newNode !== el && newNode !== null) {\r\n                                if (newNode.nodeType === 1 && checkCss(newNode)) {\r\n                                    pCurrent.push(newNode.cloneNode(false));\r\n                                    cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                                }\r\n                                newNode = newNode.parentNode;\r\n                            }\r\n\r\n                            cloneNode = appendNode = newNode = pCurrent.pop() || afterNode;\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                                appendNode = newNode;\r\n                            }\r\n\r\n                            pNode.appendChild(cloneNode);\r\n                            newNode.textContent = afterNode.data;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        cssText = '';\r\n                        while (newNode !== pNode && newNode !== el && newNode !== null) {\r\n                            if (validation(newNode) && newNode.nodeType === 1 && checkCss(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                                cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || textNode;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        newInnerNode.appendChild(childNode);\r\n\r\n                        endContainer = textNode;\r\n                        endOffset = textNode.data.length;\r\n                        endPass = true;\r\n\r\n                        if (!isRemoveFormat && collapsed) {\r\n                            newInnerNode = textNode;\r\n                            textNode.textContent = util.zeroWidthSpace;\r\n                        }\r\n\r\n                        if (newNode !== textNode) newNode.appendChild(endContainer);\r\n                        continue;\r\n                    }\r\n\r\n                    // other\r\n                    if (startPass) {\r\n                        if (child.nodeType === 1 && !util.isBreak(child)) {\r\n                            recursionFunc(child, child);\r\n                            continue;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        cssText = '';\r\n                        while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\r\n                            if (newNode.nodeType === 1 && !util.isBreak(child) && (endPass || validation(newNode)) && checkCss(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                                cssText += newNode.style.cssText.substr(0, newNode.style.cssText.indexOf(':')) + '|';\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || child;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n                        \r\n                        if (childNode === child) {\r\n                            if (!endPass) node = newInnerNode;\r\n                            else node = pNode;\r\n                        } else if (endPass) {\r\n                            pNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            newInnerNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        }\r\n                    }\r\n\r\n                    cloneNode = child.cloneNode(false);\r\n                    node.appendChild(cloneNode);\r\n                    if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            if (isRemoveFormat) {\r\n                startContainer = util.createTextNode(collapsed ? util.zeroWidthSpace : newInnerNode.textContent);\r\n                pNode.insertBefore(startContainer, newInnerNode);\r\n                pNode.removeChild(newInnerNode);\r\n                if (collapsed) startOffset = 1;\r\n            } else if (collapsed) {\r\n                startContainer = endContainer = newInnerNode;\r\n                startOffset = 1;\r\n                endOffset = 1;\r\n            }\r\n\r\n            util.removeEmptyNode(pNode);\r\n            element.parentNode.insertBefore(pNode, element);\r\n            util.removeItem(element);\r\n\r\n            return {\r\n                startContainer: startContainer,\r\n                startOffset: startOffset,\r\n                endContainer: isRemoveFormat || !endContainer.textContent ? startContainer : endContainer,\r\n                endOffset: isRemoveFormat || !endContainer.textContent ? startContainer.textContent.length : endOffset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description wraps mid lines selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @private\r\n         */\r\n        _nodeChange_middleLine: function (element, newInnerNode, validation, isRemoveFormat) {\r\n            if (isRemoveFormat) {\r\n                newInnerNode = util.createTextNode(element.textContent ? element.textContent : util.zeroWidthSpace);\r\n            } else {\r\n                (function recursionFunc(current, node) {\r\n                    const childNodes = current.childNodes;\r\n    \r\n                    for (let i = 0, len = childNodes.length; i < len; i++) {\r\n                        let child = childNodes[i];\r\n                        let coverNode = node;\r\n                        if (validation(child)) {\r\n                            let cloneNode = child.cloneNode(false);\r\n                            node.appendChild(cloneNode);\r\n                            if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n                        }\r\n                        recursionFunc(child, coverNode);\r\n                    }\r\n                })(element, newInnerNode);\r\n            }\r\n\r\n            element.innerHTML = '';\r\n            element.appendChild(newInnerNode);\r\n        },\r\n\r\n        /**\r\n         * @description wraps first line selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Element} startCon - The startContainer property of the selection object.\r\n         * @param {Number} startOff - The startOffset property of the selection object.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _nodeChange_startLine: function (element, newInnerNode, validation, startCon, startOff, isRemoveFormat) {\r\n            const el = element;\r\n            const pNode = element.cloneNode(false);\r\n\r\n            let container = startCon;\r\n            let offset = startOff;\r\n            let passNode = false;\r\n            let pCurrent, newNode, appendNode;\r\n\r\n            (function recursionFunc(current, node) {\r\n                const childNodes = current.childNodes;\r\n                for (let i = 0, len = childNodes.length; i < len; i++) {\r\n                    const child = childNodes[i];\r\n                    let coverNode = node;\r\n\r\n                    if (passNode && !util.isBreak(child)) {\r\n                        if (child.nodeType === 1) {\r\n                            recursionFunc(child, child);\r\n                            continue;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\r\n                            if (newNode.nodeType === 1 && validation(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        if (pCurrent.length > 0) {\r\n                            const childNode = pCurrent.pop();\r\n                            appendNode = newNode = childNode;\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                                appendNode = newNode;\r\n                            }\r\n                            newInnerNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n                    }\r\n\r\n                    // startContainer\r\n                    if (!passNode && child === container) {\r\n                        const prevNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(0, offset));\r\n                        const textNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(offset, (container.length - offset)));\r\n\r\n                        if (prevNode.data.length > 0) {\r\n                            node.appendChild(prevNode);\r\n                        }\r\n\r\n                        newNode = node;\r\n                        pCurrent = [];\r\n                        while (newNode !== pNode && newNode !== null) {\r\n                            if (newNode.nodeType === 1 && validation(newNode)) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || node;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        if (childNode !== node) {\r\n                            newInnerNode.appendChild(childNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n\r\n                        if (util.isBreak(child)) newInnerNode.appendChild(child.cloneNode(false));\r\n\r\n                        pNode.appendChild(newInnerNode);\r\n                        container = textNode;\r\n                        offset = 0;\r\n                        passNode = true;\r\n\r\n                        node.appendChild(container);\r\n                        continue;\r\n                    }\r\n\r\n                    if (!passNode || validation(child)) {\r\n                        const cloneNode = child.cloneNode(false);\r\n                        node.appendChild(cloneNode);\r\n                        if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            if (isRemoveFormat) {\r\n                container = util.createTextNode(newInnerNode.textContent);\r\n                pNode.insertBefore(container, newInnerNode);\r\n                pNode.removeChild(newInnerNode);\r\n            }\r\n\r\n            if (!isRemoveFormat && pNode.children.length === 0) {\r\n                if (element.childNodes) {\r\n                    container = element.childNodes[0];\r\n                } else {\r\n                    container = util.createTextNode(util.zeroWidthSpace);\r\n                    element.appendChild(container);\r\n                }\r\n            } else {\r\n                util.removeEmptyNode(pNode);\r\n                element.parentNode.insertBefore(pNode, element);\r\n                util.removeItem(element);\r\n            }\r\n\r\n            return {\r\n                container: container,\r\n                offset: offset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description wraps last line selected text.\r\n         * @param {Element} element - The node of the line that contains the selected text node.\r\n         * @param {Element} newInnerNode - The dom that will wrap the selected text area\r\n         * @param {function} validation - Check if the node should be stripped.\r\n         * @param {Element} endCon - The endContainer property of the selection object.\r\n         * @param {Number} endOff - The endOffset property of the selection object.\r\n         * @param {Boolean} isRemoveFormat - Is the remove format command ?\r\n         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _nodeChange_endLine: function (element, newInnerNode, validation, endCon, endOff, isRemoveFormat) {\r\n            const el = element;\r\n            const pNode = element.cloneNode(false);\r\n\r\n            let container = endCon;\r\n            let offset = endOff;\r\n            let passNode = false;\r\n            let pCurrent, newNode, appendNode;\r\n\r\n            (function recursionFunc(current, node) {\r\n                const childNodes = current.childNodes;\r\n                for (let i = childNodes.length -1; 0 <= i; i--) {\r\n                    const child = childNodes[i];\r\n                    let coverNode = node;\r\n\r\n                    if (passNode && !util.isBreak(child)) {\r\n                        if (child.nodeType === 1) {\r\n                            recursionFunc(child, child);\r\n                            continue;\r\n                        }\r\n\r\n                        newNode = child;\r\n                        pCurrent = [];\r\n                        while (newNode.parentNode !== null && newNode !== el && newNode !== newInnerNode) {\r\n                            if (validation(newNode) && newNode.nodeType === 1) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        if (pCurrent.length > 0) {\r\n                            const childNode = pCurrent.pop();\r\n                            appendNode = newNode = childNode;\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                                appendNode = newNode;\r\n                            }\r\n                            newInnerNode.insertBefore(childNode, newInnerNode.firstChild);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n                    }\r\n\r\n                    // endContainer\r\n                    if (!passNode && child === container) {\r\n                        const afterNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(offset, (container.length - offset)));\r\n                        const textNode = util.createTextNode(container.nodeType === 1 ? '' : container.substringData(0, offset));\r\n\r\n                        if (afterNode.data.length > 0) {\r\n                            node.insertBefore(afterNode, node.firstChild);\r\n                        }\r\n\r\n                        newNode = node;\r\n                        pCurrent = [];\r\n                        while (newNode !== pNode && newNode !== null) {\r\n                            if (validation(newNode) && newNode.nodeType === 1) {\r\n                                pCurrent.push(newNode.cloneNode(false));\r\n                            }\r\n                            newNode = newNode.parentNode;\r\n                        }\r\n\r\n                        const childNode = pCurrent.pop() || node;\r\n                        appendNode = newNode = childNode;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                            appendNode = newNode;\r\n                        }\r\n\r\n                        if (childNode !== node) {\r\n                            newInnerNode.insertBefore(childNode, newInnerNode.firstChild);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\r\n\r\n                        if (util.isBreak(child)) newInnerNode.appendChild(child.cloneNode(false));\r\n\r\n                        pNode.insertBefore(newInnerNode, pNode.firstChild);\r\n                        container = textNode;\r\n                        offset = textNode.data.length;\r\n                        passNode = true;\r\n\r\n                        node.insertBefore(container, node.firstChild);\r\n                        continue;\r\n                    }\r\n\r\n                    if (!passNode || validation(child)) {\r\n                        const cloneNode = child.cloneNode(false);\r\n                        node.insertBefore(cloneNode, node.firstChild);\r\n                        if (child.nodeType === 1 && !util.isBreak(child)) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            if (isRemoveFormat) {\r\n                container = util.createTextNode(newInnerNode.textContent);\r\n                offset = container.textContent.length;\r\n                pNode.insertBefore(container, newInnerNode);\r\n                pNode.removeChild(newInnerNode);\r\n            }\r\n\r\n            if (!isRemoveFormat && pNode.childNodes.length === 0) {\r\n                if (element.childNodes) {\r\n                    container = element.childNodes[0];\r\n                } else {\r\n                    container = util.createTextNode(util.zeroWidthSpace);\r\n                    element.appendChild(container);\r\n                }\r\n            } else {\r\n                util.removeEmptyNode(pNode);\r\n                element.parentNode.insertBefore(pNode, element);\r\n                util.removeItem(element);\r\n            }\r\n\r\n            return {\r\n                container: container,\r\n                offset: offset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description Execute command of command button(All Buttons except submenu and dialog)\r\n         * (redo, undo, bold, underline, italic, strikethrough, subscript, superscript, removeFormat, indent, outdent, fullscreen, showBlocks, codeview, preview, print)\r\n         * @param {Element} target - The element of command button\r\n         * @param {String} command - Property of command button (data-value)\r\n         */\r\n        commandHandler: function (target, command) {\r\n            switch (command) {\r\n                case 'codeView':\r\n                    this.controllersOff();\r\n                    this.toggleCodeView();\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'fullScreen':\r\n                    this.controllersOff();\r\n                    this.toggleFullScreen(target);\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'indent':\r\n                case 'outdent':\r\n                    this.indent(command);\r\n                    break;\r\n                case 'redo':\r\n                case 'undo':\r\n                    this.execCommand(command, false, null);\r\n                    break;\r\n                case 'removeFormat':\r\n                    this.removeFormat();\r\n                    break;\r\n                case 'preview':\r\n                case 'print':\r\n                    this.openWindowContents(command);\r\n                    break;\r\n                case 'showBlocks':\r\n                    this.toggleDisplayBlocks();\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'subscript':\r\n                    if (util.hasClass(context.tool.superscript, 'on')) {\r\n                        this.execCommand('superscript', false, null);\r\n                        util.removeClass(context.tool.superscript, 'on');\r\n                    }\r\n                    this.execCommand(command, false, null);\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                case 'superscript':\r\n                    if (util.hasClass(context.tool.subscript, 'on')) {\r\n                        this.execCommand('subscript', false, null);\r\n                        util.removeClass(context.tool.subscript, 'on');\r\n                    }\r\n                    this.execCommand(command, false, null);\r\n                    util.toggleClass(target, 'on');\r\n                    break;\r\n                default : // 'bold', 'underline', 'italic', 'strike'\r\n                    this.execCommand(command, false, target.getAttribute('data-value'));\r\n                    util.toggleClass(target, 'on');\r\n            }\r\n\r\n            this.focus();\r\n        },\r\n\r\n        /**\r\n         * @description Remove format of the currently selected range (IE, Edge not working)\r\n         */\r\n        removeFormat: function () {\r\n            this.nodeChange(util.createElement('REMOVENODE'));\r\n        },\r\n\r\n        /**\r\n         * @description This method implements indentation to selected range.\r\n         * Setted \"margin-left\" to \"25px\" in the top \"P\" tag of the parameter node.\r\n         * @param command {String} - Separator (\"indent\" or \"outdent\")\r\n         */\r\n        indent: function (command) {\r\n            const rangeLines = this.getSelectedFormatElements();\r\n            let p, margin;\r\n\r\n            for (let i = 0, len = rangeLines.length; i < len; i++) {\r\n                p = rangeLines[i];\r\n                margin = /\\d+/.test(p.style.marginLeft) ? p.style.marginLeft.match(/\\d+/)[0] * 1 : 0;\r\n\r\n                if ('indent' === command) {\r\n                    margin += 25;\r\n                } else {\r\n                    margin -= 25;\r\n                }\r\n    \r\n                p.style.marginLeft = (margin < 0 ? 0 : margin) + 'px';\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Add or remove the class name of \"body\" so that the code block is visible\r\n         */\r\n        toggleDisplayBlocks: function () {\r\n            util.toggleClass(context.element.wysiwyg, 'sun-editor-show-block');\r\n        },\r\n\r\n        /**\r\n         * @description Changes to code view or wysiwyg view\r\n         */\r\n        toggleCodeView: function () {\r\n            const wysiwygActive = this._variable.wysiwygActive;\r\n            const disButtons = this.codeViewDisabledButtons;\r\n            for (let i = 0, len = disButtons.length; i < len; i++) {\r\n                disButtons[i].disabled = wysiwygActive;\r\n            }\r\n\r\n            if (!wysiwygActive) {\r\n                const code_html = context.element.code.value.trim();\r\n                context.element.wysiwyg.innerHTML = code_html.length > 0 ? util.convertContentsForEditor(code_html) : '<p>' + util.zeroWidthSpace + '</p>';\r\n                context.element.wysiwyg.scrollTop = 0;\r\n                context.element.code.style.display = 'none';\r\n                context.element.wysiwyg.style.display = 'block';\r\n                if (context.option.height === 'auto') context.element.code.style.height = '0px';\r\n                this._variable.wysiwygActive = true;\r\n                this.focus();\r\n            }\r\n            else {\r\n                context.element.code.value = util.convertHTMLForCodeView(context.element.wysiwyg.innerHTML.trim());\r\n                context.element.wysiwyg.style.display = 'none';\r\n                context.element.code.style.display = 'block';\r\n                if (context.option.height === 'auto') context.element.code.style.height = context.element.code.scrollHeight > 0 ? (context.element.code.scrollHeight + 'px') : 'auto';\r\n                this._variable.wysiwygActive = false;\r\n                context.element.code.focus();\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Changes to full screen or default screen\r\n         * @param {Element} element - full screen button\r\n         */\r\n        toggleFullScreen: function (element) {\r\n            if (!this._variable.isFullScreen) {\r\n                this._variable.isFullScreen = true;\r\n\r\n                context.element.topArea.style.position = 'fixed';\r\n                context.element.topArea.style.top = '0';\r\n                context.element.topArea.style.left = '0';\r\n                context.element.topArea.style.width = '100%';\r\n                context.element.topArea.style.height = '100%';\r\n                context.element.topArea.style.zIndex = '2147483647';\r\n\r\n                this._variable._bodyOverflow = _d.body.style.overflow;\r\n                _d.body.style.overflow = 'hidden';\r\n\r\n                this._variable._editorAreaOriginCssText = context.element.editorArea.style.cssText;\r\n                this._variable._wysiwygOriginCssText = context.element.wysiwyg.style.cssText;\r\n                this._variable._codeOriginCssText = context.element.code.style.cssText;\r\n\r\n                context.element.editorArea.style.cssText = context.element.toolbar.style.cssText = context.element.wysiwyg.style.cssText = context.element.code.style.cssText = '';\r\n                context.element.toolbar.style.width = context.element.wysiwyg.style.height = context.element.code.style.height = '100%';\r\n                context.element.toolbar.style.position = 'relative';\r\n\r\n                this._variable.innerHeight_fullScreen = (_w.innerHeight - context.element.toolbar.offsetHeight);\r\n                context.element.editorArea.style.height = this._variable.innerHeight_fullScreen + 'px';\r\n\r\n                util.removeClass(element.firstElementChild, 'icon-expansion');\r\n                util.addClass(element.firstElementChild, 'icon-reduction');\r\n            }\r\n            else {\r\n                this._variable.isFullScreen = false;\r\n\r\n                context.element.code.style.cssText = this._variable._codeOriginCssText;\r\n                context.element.wysiwyg.style.cssText = this._variable._wysiwygOriginCssText;\r\n                context.element.toolbar.style.cssText = '';\r\n                context.element.editorArea.style.cssText = this._variable._editorAreaOriginCssText;\r\n                context.element.topArea.style.cssText = this._variable._originCssText;\r\n                _d.body.style.overflow = this._variable._bodyOverflow;\r\n\r\n                if (context.option.stickyToolbar > -1) {\r\n                    util.removeClass(context.element.toolbar, 'sun-editor-sticky');\r\n                    event.onScroll_window();\r\n                }\r\n\r\n                util.removeClass(element.firstElementChild, 'icon-reduction');\r\n                util.addClass(element.firstElementChild, 'icon-expansion');\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Open the preview window or open the print window\r\n         * @param {String} mode - 'preview' or 'print'\r\n         */\r\n        openWindowContents: function (mode) {\r\n            const isPrint = mode === 'print';\r\n            const windowObject = _w.open('', '_blank');\r\n            windowObject.mimeType = 'text/html';\r\n            windowObject.document.write('' +\r\n                '<!doctype html><html>' +\r\n                '<head>' +\r\n                '<meta charset=\"utf-8\" />' +\r\n                '<meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">' +\r\n                '<title>' + (isPrint ? lang.toolbar.print : lang.toolbar.preview) + '</title>' +\r\n                '<link rel=\"stylesheet\" type=\"text/css\" href=\"' + util.getIncludePath(['suneditor-contents', 'suneditor'], 'css') + '\">' +\r\n                '</head>' +\r\n                '<body>' +\r\n                '<div class=\"sun-editor-editable\" style=\"width:' + context.element.wysiwyg.offsetWidth + 'px; margin:auto;\">' +\r\n                this.getContents() + '</div>' +\r\n                (isPrint ? '<script>_w.print();</script>' : '') + '</body>' +\r\n                '</html>');\r\n        },\r\n\r\n        /**\r\n         * @description Gets the current contents\r\n         * @returns {Object}\r\n         */\r\n        getContents: function () {\r\n            let contents = '';\r\n\r\n            if (context.element.wysiwyg.innerText.trim().length === 0) return contents;\r\n\r\n            if (editor._variable.wysiwygActive) {\r\n                contents = context.element.wysiwyg.innerHTML;\r\n            } else {\r\n                contents = util.convertContentsForEditor(context.element.code.value);\r\n            }\r\n\r\n            const renderHTML = util.createElement('DIV');\r\n            renderHTML.innerHTML = contents;\r\n\r\n            const figcaptions = util.getListChildren(renderHTML, function (current) {\r\n                return /FIGCAPTION/i.test(current.nodeName);\r\n            });\r\n\r\n            for (let i = 0, len = figcaptions.length; i < len; i++) {\r\n                figcaptions[i].outerHTML = figcaptions[i].outerHTML.replace(/(?!^<figcaption\\s+)(contenteditable=\"([a-z]+|\\s*)\")\\s*(?=[^>]*>)/i, '');\r\n            }\r\n\r\n            return renderHTML.innerHTML;\r\n        }\r\n    };\r\n\r\n    /**\r\n     * @description event function\r\n     */\r\n    const event = {\r\n        _shortcutKeyCode: {\r\n            66: ['bold', 'B'],\r\n            83: ['strikethrough', 'STRIKE'],\r\n            85: ['underline', 'U'],\r\n            73: ['italic', 'I'],\r\n            89: ['redo'],\r\n            90: ['undo'],\r\n            219: ['outdent'],\r\n            221: ['indent']\r\n        },\r\n\r\n        _directionKeyKeyCode: new RegExp('^(?:8|13|32|46|33|34|35|36|37|38|39|40|98|100|102|104)$'),\r\n\r\n        _changeButtonClassTagCheck: new RegExp('^(?:B|U|I|STRIKE|SUB|SUP)$'),\r\n\r\n        _findButtonEffectTag: function () {\r\n            const commandMap = editor.commandMap;\r\n            const classOnCheck = this._changeButtonClassTagCheck;\r\n            const commandMapNodes = [];\r\n            const currentNodes = [];\r\n\r\n            let findFormat = true, findFont = true, findSize = true, findA = true;\r\n            let findB = true, findI = true, findU = true, findS = true;\r\n            let cssText = '', nodeName = '';\r\n\r\n            for (let selectionParent = editor.getSelectionNode(); !util.isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\r\n                if (!selectionParent) break;\r\n                if (selectionParent.nodeType !== 1) continue;\r\n                nodeName = selectionParent.nodeName.toUpperCase();\r\n                currentNodes.push(nodeName);\r\n\r\n                /** Format */\r\n                if (findFormat && util.isFormatElement(selectionParent)) {\r\n                    commandMapNodes.push('FORMAT');\r\n                    util.changeTxt(commandMap.FORMAT, nodeName);\r\n                    findFormat = false;\r\n                    continue;\r\n                }\r\n\r\n                /** Font */\r\n                if (findFont && (selectionParent.style.fontFamily.length > 0 || (selectionParent.face && selectionParent.face.length > 0))) {\r\n                    commandMapNodes.push('FONT');\r\n                    const selectFont = (selectionParent.style.fontFamily || selectionParent.face || lang.toolbar.font).replace(/[\"']/g,'');\r\n                    util.changeTxt(commandMap.FONT, selectFont);\r\n                    findFont = false;\r\n                }\r\n\r\n                /** A */\r\n                if (findA && /^A$/.test(nodeName) && selectionParent.getAttribute('data-image-link') === null) {\r\n                    if (!context.link || editor.controllerArray[0] !== context.link.linkBtn) {\r\n                        editor.callPlugin('link', function () {\r\n                            editor.plugins.link.call_controller_linkButton.call(editor, selectionParent);\r\n                        });\r\n                    }\r\n                    findA = false;\r\n                } else if (findA && context.link && editor.controllerArray[0] === context.link.linkBtn) {\r\n                    editor.controllersOff();\r\n                }\r\n\r\n                /** SPAN */\r\n                if (findSize && /^SPAN$/.test(nodeName)) {\r\n                    /** font size */\r\n                    if (selectionParent.style.fontSize.length > 0) {\r\n                        commandMapNodes.push('SIZE');\r\n                        util.changeTxt(commandMap.SIZE, selectionParent.style.fontSize.match(/\\d+/)[0]);\r\n                        findSize = false;\r\n                    }\r\n                }\r\n\r\n                /** command map */\r\n                cssText = selectionParent.style.cssText;\r\n                if (findB && /font\\-weight\\s*:\\s*(?:\\d+|bold|bolder)(?:;|\\s|)/.test(cssText)) {\r\n                    commandMapNodes.push('B');\r\n                    findB = false;\r\n                }\r\n                if (findI && /font\\-style\\s*:\\s*(?:italic|oblique)(?:;|\\s)/.test(cssText)) {\r\n                    commandMapNodes.push('I');\r\n                    findI = false;\r\n                }\r\n                if (findU && /text\\-decoration(?:\\-line)?\\s*:\\s*underline(?:;|\\s|)/.test(cssText)) {\r\n                    commandMapNodes.push('U');\r\n                    findU = false;\r\n                }\r\n                if (findS && /text\\-decoration(?:\\-line)?\\s*:\\s*line-through(?:;|\\s|)/.test(cssText)) {\r\n                    commandMapNodes.push('STRIKE');\r\n                    findS = false;\r\n                }\r\n\r\n                commandMapNodes.push((/^STRONG$/.test(nodeName) ? 'B' : /^EM$/.test(nodeName) ? 'I' : nodeName));\r\n            }\r\n\r\n            /** A Tag edit controller off */\r\n            if (findA) editor.controllersOff();\r\n\r\n            /** toggle class on */\r\n            for (let i = 0; i < commandMapNodes.length; i++) {\r\n                nodeName = commandMapNodes[i];\r\n                if (classOnCheck.test(nodeName)) {\r\n                    util.addClass(commandMap[nodeName], 'on');\r\n                }\r\n            }\r\n\r\n            /** remove class, display text */\r\n            for (let key in commandMap) {\r\n                if (commandMapNodes.indexOf(key) > -1) continue;\r\n                if (/^FONT/i.test(key)) {\r\n                    util.changeTxt(commandMap[key], lang.toolbar.font);\r\n                }\r\n                else if (/^SIZE$/i.test(key)) {\r\n                    util.changeTxt(commandMap[key], lang.toolbar.fontSize);\r\n                }\r\n                else {\r\n                    util.removeClass(commandMap[key], 'on');\r\n                }\r\n            }\r\n\r\n            /** save current nodes */\r\n            editor._variable.currentNodes = currentNodes.reverse();\r\n\r\n            /**  Displays the current node structure to resizingBar */\r\n            if (context.option.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\r\n        },\r\n\r\n        _cancelCaptionEdit: function () {\r\n            this.setAttribute('contenteditable', false);\r\n            this.removeEventListener('blur', event._cancelCaptionEdit);\r\n        },\r\n\r\n        onClick_toolbar: function (e) {\r\n            e.preventDefault();\r\n            e.stopPropagation();\r\n\r\n            let target = e.target;\r\n            let display = target.getAttribute('data-display');\r\n            let command = target.getAttribute('data-command');\r\n            let className = target.className;\r\n\r\n            while (!command && !/editor_tool/.test(className) && !/sun-editor-id-toolbar/.test(className)) {\r\n                target = target.parentNode;\r\n                command = target.getAttribute('data-command');\r\n                display = target.getAttribute('data-display');\r\n                className = target.className;\r\n            }\r\n\r\n            if (!command && !display) return;\r\n            if (target.disabled) return;\r\n            \r\n            /** Dialog, Submenu */\r\n            if (display) {\r\n                if (/submenu/.test(display) && (target.nextElementSibling === null || target !== editor.submenuActiveButton)) {\r\n                    editor.submenuOff();\r\n                    editor.callPlugin(command, function () {\r\n                        editor.submenuOn(target);\r\n                    });\r\n                    return;\r\n                }\r\n                else if (/dialog/.test(display)) {\r\n                    editor.callPlugin(command, function () {\r\n                        editor.plugins.dialog.open.call(editor, command, false);\r\n                    });\r\n                }\r\n\r\n                editor.submenuOff();\r\n                return;\r\n            }\r\n\r\n            editor.submenuOff();\r\n\r\n            /** default command */\r\n            if (command) {\r\n                editor.focus();\r\n                editor.commandHandler(target, command);\r\n            }\r\n        },\r\n\r\n        onClick_wysiwyg: function (e) {\r\n            e.stopPropagation();\r\n            const targetElement = e.target;\r\n            editor.submenuOff();\r\n\r\n            if (/^IMG$/i.test(targetElement.nodeName)) {\r\n                e.preventDefault();\r\n                editor.callPlugin('image', function () {\r\n                    const size = editor.plugins.resizing.call_controller_resize.call(editor, targetElement, 'image');\r\n                    editor.plugins.image.onModifyMode.call(editor, targetElement, size);\r\n                    \r\n                    if (!util.getParentElement(targetElement, '.sun-editor-id-image-container')) {\r\n                        editor.plugins.image.openModify.call(editor, true);\r\n                        editor.plugins.image.update_image.call(editor);\r\n                        editor.controllersOff();\r\n                    }\r\n                });\r\n\r\n                return;\r\n            }\r\n\r\n            if (/sun-editor-id-iframe-inner-resizing-cover/i.test(targetElement.className)) {\r\n                e.preventDefault();\r\n                editor.callPlugin('video', function () {\r\n                    const iframe = util.getChildElement(targetElement.parentNode, 'iframe');\r\n                    const size = editor.plugins.resizing.call_controller_resize.call(editor, iframe, 'video');\r\n                    editor.plugins.video.onModifyMode.call(editor, iframe, size);\r\n                });\r\n\r\n                return;\r\n            }\r\n\r\n            editor._setEditorRange();\r\n            event._findButtonEffectTag();\r\n\r\n            if (editor._isBalloon) {\r\n                if (editor.getRange().collapsed) {\r\n                    event._hideToolbar();\r\n                } else {\r\n                    event._showToolbarBalloon();\r\n                    return;\r\n                }\r\n            }\r\n\r\n            const figcaption = util.getParentElement(targetElement, 'FIGCAPTION');\r\n            if (figcaption && figcaption.getAttribute('contenteditable') !== 'ture') {\r\n                e.preventDefault();\r\n                figcaption.setAttribute('contenteditable', true);\r\n                figcaption.focus();\r\n\r\n                if (editor._isInline && !editor._inlineToolbarAttr.isShow) {\r\n                    event._showToolbarInline();\r\n\r\n                    const hideToolbar = function () {\r\n                        event._hideToolbar();\r\n                        _d.removeEventListener('click', hideToolbar);\r\n                    };\r\n\r\n                    _d.addEventListener('click', hideToolbar);\r\n                }\r\n            } else {\r\n                const td = util.getParentElement(targetElement, util.isCell);\r\n                if (td) {\r\n                    if (editor.controllerArray.length === 0) {\r\n                        editor.callPlugin('table', editor.plugins.table.call_controller_tableEdit.bind(editor, td));\r\n                    }\r\n                }\r\n            }\r\n\r\n            if (userFunction.onClick) userFunction.onClick(e);\r\n        },\r\n\r\n        _showToolbarBalloon: function () {\r\n            const range = editor.getRange();\r\n            const padding = 20;\r\n            const toolbar = context.element.toolbar;\r\n            const selection = _w.getSelection();\r\n\r\n            let isDirTop;\r\n            if (selection.focusNode === selection.anchorNode) {\r\n                isDirTop = selection.focusOffset < selection.anchorOffset;\r\n            } else {\r\n                const childNodes = util.getListChildNodes(range.commonAncestorContainer);\r\n                isDirTop = util.getArrayIndex(childNodes, selection.focusNode) < util.getArrayIndex(childNodes, selection.anchorNode)\r\n            }\r\n\r\n            let rects = range.getClientRects();\r\n            rects = rects[isDirTop ? 0 : rects.length - 1];\r\n            \r\n            toolbar.style.display = 'block';\r\n\r\n            const toolbarWidth = toolbar.offsetWidth;\r\n            const toolbarHeight = toolbar.offsetHeight;\r\n\r\n            let l = (isDirTop ? rects.left : rects.right) - context.element.topArea.offsetLeft + (_w.scrollX || _d.documentElement.scrollLeft) - toolbarWidth / 2;\r\n            let t = (isDirTop ? rects.top - toolbarHeight - 11 : rects.bottom + 11) - context.element.topArea.offsetTop + (_w.scrollY || _d.documentElement.scrollTop);\r\n\r\n            const overRight = l + toolbarWidth - context.element.topArea.offsetWidth;\r\n            \r\n            toolbar.style.left = (l < 0 ? padding : overRight < 0 ? l : l - overRight - padding) + 'px';\r\n            toolbar.style.top = (t) + 'px';\r\n\r\n            if (isDirTop) {\r\n                util.removeClass(context.element._arrow, 'arrow-up');\r\n                util.addClass(context.element._arrow, 'arrow-down');\r\n                context.element._arrow.style.top = (toolbarHeight) + 'px';\r\n            } else {\r\n                util.removeClass(context.element._arrow, 'arrow-down');\r\n                util.addClass(context.element._arrow, 'arrow-up');\r\n                context.element._arrow.style.top = '-11px';\r\n            }\r\n\r\n            const arrow_width = context.element._arrow.offsetWidth;\r\n            const arrow_left = (toolbarWidth / 2 + (l < 0 ? l - arrow_width : overRight < 0 ? 0 : overRight + arrow_width));\r\n            const arrow_point_width = arrow_width / 2;\r\n            context.element._arrow.style.left = (arrow_left < arrow_point_width ? arrow_point_width : arrow_left + arrow_point_width >= toolbarWidth ? arrow_left - arrow_point_width : arrow_left) + 'px';\r\n        },\r\n\r\n        _showToolbarInline: function () {\r\n            const toolbar = context.element.toolbar;\r\n            toolbar.style.display = 'block';\r\n            editor._inlineToolbarAttr.width = toolbar.style.width = context.option.toolbarWidth;\r\n            editor._inlineToolbarAttr.top = toolbar.style.top = (-1 - toolbar.offsetHeight) + 'px';\r\n            event.onScroll_window();\r\n            editor._inlineToolbarAttr.isShow = true;\r\n        },\r\n\r\n        _hideToolbar: function () {\r\n            context.element.toolbar.style.display = 'none';\r\n            editor._inlineToolbarAttr.isShow = false;\r\n        },\r\n\r\n        onKeyDown_wysiwyg: function (e) {\r\n            const keyCode = e.keyCode;\r\n            const shift = e.shiftKey;\r\n            const ctrl = e.ctrlKey || e.metaKey;\r\n            const alt = e.altKey;\r\n            e.stopPropagation();\r\n\r\n            if (editor._isBalloon) {\r\n                event._hideToolbar();\r\n            }\r\n\r\n            function shortcutCommand(keyCode) {\r\n                const key = event._shortcutKeyCode[keyCode];\r\n                if (!key) return false;\r\n\r\n                editor.commandHandler(util.getFormatElement(editor.getSelectionNode()), key[0]);\r\n                util.toggleClass(editor.commandMap[key[1]], 'on');\r\n\r\n                return true;\r\n            }\r\n\r\n            /** Shortcuts */\r\n            if (ctrl && !/^(?:16|17|18)$/.test(keyCode)) {\r\n                if (!(!shift && keyCode === 83) && shortcutCommand(keyCode)) {\r\n                    e.preventDefault();\r\n                    return;\r\n                }\r\n            }\r\n\r\n            /** default key action */\r\n            const selectionNode = editor.getSelectionNode();\r\n            switch (keyCode) {\r\n                case 8: /**backspace key*/\r\n                    if (util.isFormatElement(selectionNode) && util.isWysiwygDiv(selectionNode.parentNode) && selectionNode.previousSibling === null) {\r\n                        e.preventDefault();\r\n                        e.stopPropagation();\r\n                        selectionNode.innerHTML = util.zeroWidthSpace;\r\n                        return false;\r\n                    }\r\n                    \r\n                    break;\r\n                case 9:\r\n                    /**tab key*/\r\n                    e.preventDefault();\r\n                    if (ctrl || alt) break;\r\n\r\n                    editor.controllersOff();\r\n\r\n                    let currentNode = selectionNode || editor.getSelectionNode();\r\n                    while (!util.isCell(currentNode) && !util.isWysiwygDiv(currentNode)) {\r\n                        currentNode = currentNode.parentNode;\r\n                    }\r\n\r\n                    if (currentNode && util.isCell(currentNode)) {\r\n                        const table = util.getParentElement(currentNode, 'table');\r\n                        const cells = util.getListChildren(table, util.isCell);\r\n                        let idx = shift ? util.prevIdx(cells, currentNode) : util.nextIdx(cells, currentNode);\r\n\r\n                        if (idx === cells.length && !shift) idx = 0;\r\n                        if (idx === -1 && shift) idx = cells.length - 1;\r\n\r\n                        const moveCell = cells[idx];\r\n                        if (!moveCell) return false;\r\n\r\n                        editor.setRange(moveCell, 0, moveCell, 0);\r\n\r\n                        break;\r\n                    }\r\n\r\n                    /** format Tag */\r\n                    const lines = editor.getSelectedFormatElements();\r\n\r\n                    if (!shift) {\r\n                        const tabText = util.createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\r\n                        if (lines.length === 1) {\r\n                            editor.insertNode(tabText);\r\n                            editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize);\r\n                        } else {\r\n                            for (let i = 0, len = lines.length; i < len; i++) {\r\n                                lines[i].insertBefore(tabText.cloneNode(false), lines[i].firstChild);\r\n                            }\r\n                        }\r\n                    } else {\r\n                        for (let i = 0, len = lines.length, child; i < len; i++) {\r\n                            child = lines[i].firstChild;\r\n                            if (/^\\s{1,4}$/.test(child.textContent)) {\r\n                                util.removeItem(child);\r\n                            } else if (/^\\s{1,4}/.test(child.textContent)) {\r\n                                child.textContent = child.textContent.replace(/^\\s{1,4}/, '');\r\n                            }\r\n                        }\r\n                    }\r\n\r\n                    break;\r\n            }\r\n\r\n            if (userFunction.onKeyDown) userFunction.onKeyDown(e);\r\n        },\r\n\r\n        onKeyUp_wysiwyg: function (e) {\r\n            editor._setEditorRange();\r\n            editor.controllersOff();\r\n            const selectionNode = editor.getSelectionNode();\r\n\r\n            if (editor._isBalloon && !editor.getRange().collapsed) {\r\n                event._showToolbarBalloon();\r\n                return;\r\n            }\r\n\r\n            /** when format tag deleted */\r\n            if (e.keyCode === 8 && util.isWysiwygDiv(selectionNode) && context.element.wysiwyg.textContent.length === 0) {\r\n                e.preventDefault();\r\n                e.stopPropagation();\r\n\r\n                const oFormatTag = util.createElement(util.isFormatElement(editor._variable.currentNodes[0]) ? editor._variable.currentNodes[0] : 'P');\r\n                oFormatTag.innerHTML = util.zeroWidthSpace;\r\n\r\n                selectionNode.appendChild(oFormatTag);\r\n                editor.setSelectionNode(oFormatTag);\r\n                editor.setRange(oFormatTag, 0, oFormatTag, 0);\r\n                return;\r\n            }\r\n\r\n            if ((util.isWysiwygDiv(selectionNode.parentElement) || util.isRangeFormatElement(selectionNode.parentElement)) && selectionNode.nodeType === 3) {\r\n                editor.execCommand('formatBlock', false, util.isWysiwygDiv(selectionNode.parentElement) ? 'P' : 'DIV');\r\n                editor._setEditorRange();\r\n                event._findButtonEffectTag();\r\n                return;\r\n            }\r\n\r\n            if (event._directionKeyKeyCode.test(e.keyCode)) {\r\n                event._findButtonEffectTag();\r\n            }\r\n\r\n            if (userFunction.onKeyUp) userFunction.onKeyUp(e);\r\n        },\r\n\r\n        onScroll_wysiwyg: function (e) {\r\n            editor.controllersOff();\r\n            if (userFunction.onScroll) userFunction.onScroll(e);\r\n        },\r\n\r\n        onDrop_wysiwyg: function (e) {\r\n            const files = e.dataTransfer.files;\r\n\r\n            if (files.length > 0) {\r\n                e.stopPropagation();\r\n                e.preventDefault();\r\n                \r\n                editor.focus();\r\n    \r\n                editor.callPlugin('image', function () {\r\n                    context.image.imgInputFile.files = files;\r\n                    editor.plugins.image.onRender_imgInput.call(editor);\r\n                    context.image.imgInputFile.files = null;\r\n                });\r\n            }\r\n\r\n            if (userFunction.onDrop) userFunction.onDrop(e);\r\n        },\r\n\r\n        onMouseDown_resizingBar: function (e) {\r\n            e.stopPropagation();\r\n\r\n            editor._variable.resizeClientY = e.clientY;\r\n            context.element.resizeBackground.style.display = 'block';\r\n\r\n            function closureFunc() {\r\n                context.element.resizeBackground.style.display = 'none';\r\n                _d.removeEventListener('mousemove', event._resize_editor);\r\n                _d.removeEventListener('mouseup', closureFunc);\r\n            }\r\n\r\n            _d.addEventListener('mousemove', event._resize_editor);\r\n            _d.addEventListener('mouseup', closureFunc);\r\n        },\r\n\r\n        _resize_editor: function (e) {\r\n            const resizeInterval = context.element.editorArea.offsetHeight + (e.clientY - editor._variable.resizeClientY);\r\n            context.element.wysiwyg.style.height = context.element.code.style.height = (resizeInterval < editor._variable.minResizingSize ? editor._variable.minResizingSize : resizeInterval) + 'px';\r\n            editor._variable.resizeClientY = e.clientY;\r\n        },\r\n\r\n        onResize_window: function () {\r\n            if (context.element.toolbar.offsetWidth === 0) return;\r\n\r\n            if (editor._variable.isFullScreen) {\r\n                editor._variable.innerHeight_fullScreen += (_w.innerHeight - context.element.toolbar.offsetHeight) - editor._variable.innerHeight_fullScreen;\r\n                context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\r\n            }\r\n            else if (editor._variable._sticky) {\r\n                context.element.toolbar.style.width = (context.element.topArea.offsetWidth - 2) + 'px';\r\n                event.onScroll_window();\r\n            }\r\n\r\n            editor.controllersOff();\r\n        },\r\n\r\n        onScroll_window: function () {\r\n            if (editor._variable.isFullScreen || context.element.toolbar.offsetWidth === 0) return;\r\n\r\n            const element = context.element;\r\n            const editorHeight = element.editorArea.offsetHeight;\r\n            const editorTop = element.topArea.offsetTop - (editor._isInline ? element.toolbar.offsetHeight : 0);\r\n            const y = (this.scrollY || _d.documentElement.scrollTop) + context.option.stickyToolbar;\r\n            \r\n            if (y < editorTop) {\r\n                event._offStickyToolbar(element);\r\n            }\r\n            else if (y + editor._variable.minResizingSize >= editorHeight + editorTop) {\r\n                if (!editor._variable._sticky) event._onStickyToolbar(element);\r\n                element.toolbar.style.top = (editorHeight + editorTop + context.option.stickyToolbar -y - editor._variable.minResizingSize) + 'px';\r\n            }\r\n            else if (y >= editorTop) {\r\n                event._onStickyToolbar(element);\r\n            }\r\n        },\r\n\r\n        _onStickyToolbar: function (element) {\r\n            if (!editor._isInline) {\r\n                element._stickyDummy.style.height = element.toolbar.offsetHeight + 'px';\r\n                element._stickyDummy.style.display = 'block';\r\n            }\r\n\r\n            element.toolbar.style.top = context.option.stickyToolbar + 'px';\r\n            element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : element.toolbar.offsetWidth + 'px';\r\n            util.addClass(element.toolbar, 'sun-editor-sticky');\r\n            editor._variable._sticky = true;\r\n        },\r\n\r\n        _offStickyToolbar: function (element) {\r\n            element._stickyDummy.style.display = 'none';\r\n            element.toolbar.style.top = editor._isInline ? editor._inlineToolbarAttr.top : '';\r\n            element.toolbar.style.width = editor._isInline ? editor._inlineToolbarAttr.width : '';\r\n            element.editorArea.style.marginTop = '';\r\n            util.removeClass(element.toolbar, 'sun-editor-sticky');\r\n            editor._variable._sticky = false;\r\n        },\r\n\r\n        _codeViewAutoScroll: function () {\r\n            context.element.code.style.height = context.element.code.scrollHeight + 'px';\r\n        },\r\n\r\n        onPaste_wysiwyg: function (e) {\r\n            if (!e.clipboardData.getData) return true;\r\n\r\n            const cleanData = util.cleanHTML(e.clipboardData.getData('text/html'));\r\n            \r\n            if (cleanData) {\r\n                editor.execCommand('insertHTML', false, cleanData);\r\n                e.stopPropagation();\r\n                e.preventDefault();\r\n            }\r\n        }\r\n    };\r\n\r\n    /** add event listeners */\r\n    /** toolbar event */\r\n    context.element.toolbar.addEventListener('click', event.onClick_toolbar, false);\r\n    context.element.toolbar.addEventListener('mousedown', function (e) { e.preventDefault(); }, false);\r\n    /** editor area */\r\n    context.element.relative.addEventListener('click', editor.focus.bind(editor), false);\r\n    context.element.wysiwyg.addEventListener('click', event.onClick_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('keydown', event.onKeyDown_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('keyup', event.onKeyUp_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('drop', event.onDrop_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('paste', event.onPaste_wysiwyg, false);\r\n    \r\n    /** code view area auto line */\r\n    if (context.option.height === 'auto') context.element.code.addEventListener('keyup', event._codeViewAutoScroll, false);\r\n\r\n    /** resizingBar */\r\n    if (context.element.resizingBar) {\r\n        if (/\\d+/.test(context.option.height)) {\r\n            context.element.resizingBar.addEventListener('mousedown', event.onMouseDown_resizingBar, false);\r\n        } else {\r\n            util.addClass(context.element.resizingBar, 'none-resize');\r\n        }\r\n    }\r\n\r\n    /** inlineToolbar */\r\n    if (editor._isInline) {\r\n        context.element.wysiwyg.addEventListener('focus', event._showToolbarInline, false);\r\n    }\r\n\r\n    if (editor._isInline || editor._isBalloon) {\r\n        context.element.wysiwyg.addEventListener('blur', event._hideToolbar, false);\r\n    }\r\n    \r\n    /** window event */\r\n    _w.addEventListener('resize', event.onResize_window, false);\r\n    if (context.option.stickyToolbar > -1) _w.addEventListener('scroll', event.onScroll_window, false);\r\n\r\n    /** add plugin to plugins object */\r\n    if (plugins) {\r\n        Object.keys(plugins).map(function(key) {\r\n            let plugin = plugins[key];\r\n            editor.plugins[plugin.name] = util.copyObj(plugin);\r\n        });\r\n    }\r\n\r\n    /** User function */\r\n    const userFunction = {\r\n        /**\r\n         * @description Event functions\r\n         * @param {Object} event - Event Object\r\n         */\r\n        onScroll: null,\r\n        onClick: null,\r\n        onKeyDown: null,\r\n        onKeyUp: null,\r\n        onDrop: null,\r\n\r\n        /**\r\n         * @description Called when the image is uploaded or the uploaded image is deleted\r\n         * @param {Element} targetImgElement - Current img element\r\n         * @param {Number} index - Uploaded index\r\n         * @param {Boolean} isDelete - Whether or not it was called after the delete operation\r\n         */\r\n        onImageUpload: null,\r\n\r\n        /**\r\n         * @description Open a notice area\r\n         * @param {String} message - Notice message\r\n         */\r\n        noticeOpen: function (message) {\r\n            editor.addModule([notice]);\r\n            notice.open.call(editor, message);\r\n        },\r\n\r\n        /**\r\n         * @description Close a notice area\r\n         */\r\n        noticeClose: function () {\r\n            editor.addModule([notice]);\r\n            notice.close.call(editor);\r\n        },\r\n\r\n        /**\r\n         * @description Copying the contents of the editor to the original textarea\r\n         */\r\n        save: function () {\r\n            context.element.originElement.value = editor.getContents();\r\n        },\r\n\r\n        /**\r\n         * @description Gets the suneditor's context object. Contains settings, plugins, and cached element objects\r\n         * @returns {Object}\r\n         */\r\n        getContext: function () {\r\n            return context;\r\n        },\r\n\r\n        /**\r\n         * @description Gets the contents of the suneditor\r\n         * @returns {String}\r\n         */\r\n        getContents: function () {\r\n            return editor.getContents();\r\n        },\r\n\r\n        /**\r\n         * @description Gets uploaded images informations\r\n         * @returns {Array}\r\n         */\r\n        getImagesInfo: function () {\r\n            return editor._variable._imagesInfo;\r\n        },\r\n\r\n        /**\r\n         * @description Inserts an HTML element or HTML string or plain string at the current cursor position\r\n         * @param {Element|String} html - HTML Element or HTML string or plain string\r\n         */\r\n        insertHTML: function (html) {\r\n            if (!html.nodeType || html.nodeType !== 1) {\r\n                const template = util.createElement('template');\r\n                template.innerHTML = html;\r\n                html = template.firstChild || template.content.firstChild;\r\n            }\r\n\r\n            editor.insertNode(html);\r\n            editor.focus();\r\n        },\r\n\r\n        /**\r\n         * @description Change the contents of the suneditor\r\n         * @param {String} contents - Contents to Input\r\n         */\r\n        setContents: function (contents) {\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.wysiwyg.innerHTML = util.convertContentsForEditor(contents);\r\n            } else {\r\n                context.element.code.value = contents;\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Add contents to the suneditor\r\n         * @param {String} contents - Contents to Input\r\n         */\r\n        appendContents: function (contents) {\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.wysiwyg.innerHTML += util.convertContentsForEditor(contents);\r\n            } else {\r\n                context.element.code.value += contents;\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Disable the suneditor\r\n         */\r\n        disabled: function () {\r\n            context.tool.cover.style.display = 'block';\r\n            context.element.wysiwyg.setAttribute('contenteditable', false);\r\n            context.element.code.setAttribute('disabled', 'disabled');\r\n        },\r\n\r\n        /**\r\n         * @description Enabled the suneditor\r\n         */\r\n        enabled: function () {\r\n            context.tool.cover.style.display = 'none';\r\n            context.element.wysiwyg.setAttribute('contenteditable', true);\r\n            context.element.code.removeAttribute('disabled');\r\n        },\r\n\r\n        /**\r\n         * @description Show the suneditor\r\n         */\r\n        show: function () {\r\n            const topAreaStyle = context.element.topArea.style;\r\n            if (topAreaStyle.display === 'none') topAreaStyle.display = context.option.display;\r\n        },\r\n\r\n        /**\r\n         * @description Hide the suneditor\r\n         */\r\n        hide: function () {\r\n            context.element.topArea.style.display = 'none';\r\n        },\r\n\r\n        /**\r\n         * @description Destroy the suneditor\r\n         */\r\n        destroy: function () {\r\n            /** remove window event listeners */\r\n            _w.removeEventListener('resize', event.onResize_window);\r\n            _w.removeEventListener('scroll', event.onScroll_window);\r\n            \r\n            /** remove element */\r\n            util.removeItem(context.element.topArea);\r\n\r\n            this.onScroll = null;\r\n            this.onClick = null;\r\n            this.onKeyDown = null;\r\n            this.onKeyUp = null;\r\n            this.onDrop = null;\r\n            this.save = null;\r\n            this.onImageUpload = null;\r\n            this.noticeOpen = null;\r\n            this.noticeClose = null;\r\n            this.getContext = null;\r\n            this.getContents = null;\r\n            this.getImagesInfo = null;\r\n            this.insertHTML = null;\r\n            this.setContents = null;\r\n            this.appendContents = null;\r\n            this.disabled = null;\r\n            this.enabled = null;\r\n            this.show = null;\r\n            this.hide = null;\r\n            this.destroy = null;\r\n\r\n            context = null;\r\n            plugins = null;\r\n            lang = null;\r\n        }\r\n    };\r\n\r\n    return userFunction;\r\n};\r\n\r\nexport default core;"],"mappings":"AAAA;AAAA;AAAA;AAAA;;;;;;;AAOA;AACA;AACA;AACA;AAEA;;;;;;;;AAOA;AACA;AACA;AACA;;;;;AAIA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;AAGA;AACA;AACA;;;;AAIA;AACA;AACA;;;;AAIA;AACA;AACA;;;;AAIA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AAYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AACA;AAWA;;;;;;;;;;;;;AAaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAjBA;AACA;AAmBA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;;;;;;;AAOA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;;;;AAQA;AACA;AACA;AACA;AACA;AACA;AAEA;AAGA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAXA;AAaA;AACA;AACA;AACA;AAGA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;AAQA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AAWA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;AAcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAJA;AAMA;AACA;AACA;;;;;;;;AAQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;AAWA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;;;;;;AAWA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AAAA;AACA;AACA;AAhDA;AACA;AAkDA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AAAA;AACA;AAaA;AACA;AACA;;;;AAIA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA9iDA;AAijDA;;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AARA;AAWA;AAEA;AAEA;AACA;AACA;AACA;AACA;AAEA;AAAA;AAAA;AAAA;AACA;AAAA;AAAA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AAEA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAAA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AADA;AAEA;AACA;AAAA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAEA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AA9DA;AACA;AAgEA;AACA;AAEA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAjjBA;AAojBA;AACA;AAAA;AACA;AAAA;AACA;AAAA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAAA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;AAAA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AADA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AAAA;AAEA;AACA;AACA;AADA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AA9KA;AAiLA;AACA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/lib/core.js\n"); /***/ }),