diff --git a/README.md b/README.md index d31643ea2..906e0bfba 100644 --- a/README.md +++ b/README.md @@ -91,7 +91,7 @@ suneditor.create('sample', { }); ``` -### 2. Load plugins +### 2. Load only what you want ```javascript import 'suneditor/dist/css/suneditor.min.css' import suneditor from 'suneditor' @@ -131,7 +131,7 @@ import suneditor from 'suneditor' import plugins from 'suneditor/src/plugins' suneditor.create('sample', { - modules: plugins, + plugins: plugins, buttonList: [ ['undo', 'redo'], ['font', 'fontSize', 'formatBlock'], @@ -146,6 +146,20 @@ suneditor.create('sample', { ['preview', 'print'] ] }) + +// You can also load what you want +suneditor.create('sample', { + plugins: [ + plugins.font + plugins.fontSize, + plugins.formatBlock + ], + buttonList: [ + ['bold', 'underline', 'italic', 'strike', 'subscript', 'superscript'], + ['font', 'fontSize', 'formatBlock'], + ['removeFormat'] + ] +}) ``` ### 4. Plugins can be used directly in the button list diff --git a/bower.json b/bower.json index 48dbb0296..e71f7588f 100644 --- a/bower.json +++ b/bower.json @@ -1,6 +1,6 @@ { "name": "suneditor", - "version": "2.0.10", + "version": "2.0.11", "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 a9c7eb32e..61e9bb669 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";__webpack_exports__.a={name:"fontColor",add:function(_this,targetElement){let listDiv=eval(this.setSubmenu());listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickUp.bind(_this)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(){const e=document.createElement("DIV");e.className="layer_editor",e.style.display="none";const t=["#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 i='
",e.innerHTML=i,e},pickUp:function(e){if(e.preventDefault(),e.stopPropagation(),!/^BUTTON$/i.test(e.target.tagName))return!1;this.focus();const t=document.createElement("SPAN");t.style.color=e.target.getAttribute("data-value"),this.wrapRangeToTag(t,["color"]),this.submenuOff()}}},"1kvd":function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_exports__.a={name:"dialog",add:function(_this){const context=_this.context;context.dialog={_resizeClientX:0,_resizeClientY:0,_resize_plugin:"",_resize_w:0,_resize_h:0,_resize_direction:""};let dialog_div=document.createElement("DIV");dialog_div.className="sun-editor-id-dialogBox sun-editor-common";let dialog_back=document.createElement("DIV");dialog_back.className="modal-dialog-background sun-editor-id-dialog-back",dialog_back.style.display="none";let dialog_area=document.createElement("DIV");dialog_area.className="modal-dialog sun-editor-id-dialog-modal",dialog_area.style.display="none",dialog_div.appendChild(dialog_back),dialog_div.appendChild(dialog_area),context.dialog.modalArea=dialog_div,context.dialog.back=dialog_back,context.dialog.modal=dialog_area;let resize_div_container=eval(this.setController_resize());context.dialog.resizeContainer=resize_div_container,context.dialog.resizeDiv=resize_div_container.getElementsByClassName("modal-resize")[0],context.dialog.resizeDot=resize_div_container.getElementsByClassName("resize-dot")[0],context.dialog.resizeDisplay=resize_div_container.getElementsByClassName("resize-display")[0];let resize_button=eval(this.setController_button(_this.lang));context.dialog.resizeButton=resize_button;let resize_handles=resize_div_container.getElementsByClassName("sun-editor-name-resize-handle");context.dialog.modal.addEventListener("click",this.onClick_dialog.bind(_this)),context.element.topArea.getElementsByClassName("sun-editor-container")[0].appendChild(dialog_div),resize_handles[0].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[1].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[2].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[3].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[4].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[5].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[6].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_handles[7].addEventListener("mousedown",this.onMouseDown_resize_handle.bind(_this)),resize_button.addEventListener("click",this.onClick_resizeButton.bind(_this)),context.element.relative.appendChild(resize_div_container),context.element.relative.appendChild(resize_button),dialog_div=null,dialog_back=null,dialog_area=null,resize_div_container=null,resize_button=null,resize_handles=null},onClick_dialog:function(e){e.stopPropagation(),(/modal-dialog/.test(e.target.className)||/close/.test(e.target.getAttribute("data-command")))&&this.plugins.dialog.closeDialog.call(this)},openDialog:function(e,t,i){if(this.modalForm)return!1;this.context.dialog.updateModal=i,this.context.dialog.modalArea.style.position="full"===t?"fixed":"absolute",this.context.dialog.kind=e,this.modalForm=this.context[e].modal;const n=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",n&&n.focus()},closeDialog: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)},setController_resize:function(){const e=document.createElement("DIV");return e.className="modal-resize-container",e.style.display="none",e.innerHTML='
',e},setController_button:function(e){const t=document.createElement("DIV");return t.className="resize-btn",t.style.display="none",t.innerHTML='
',t},call_controller_resize:function(e,t){this.context.dialog._resize_plugin=t,e.setAttribute("unselectable","on"),e.contentEditable=!1;const i=this.context.dialog.resizeContainer,n=this.context.dialog.resizeDiv,o=e.offsetWidth,l=e.offsetHeight,s=e.offsetTop-this.context.element.wysiwyg.scrollTop,a=e.offsetLeft;return i.style.top=s+"px",i.style.left=a+"px",i.style.width=o+"px",i.style.height=l+"px",n.style.top="0px",n.style.left="0px",n.style.width=o+"px",n.style.height=l+"px",this.context.dialog.resizeButton.style.top=l+s+"px",this.context.dialog.resizeButton.style.left=a+"px",this.util.changeTxt(this.context.dialog.resizeDisplay,o+" x "+l),this.context.dialog.resizeContainer.style.display="block",this.context.dialog.resizeButton.style.display="block",this.context.dialog.resizeDot.style.display="block",this.context.dialog._resize_w=o,this.context.dialog._resize_h=l,this.controllerArray=[this.context.dialog.resizeContainer,this.context.dialog.resizeButton],{w:o,h:l,t:s,l:a}},cancel_controller_resize:function(){this.context[this.context.dialog._resize_plugin]._resize_element.style.width=this.context.dialog._resize_w+"px",this.context[this.context.dialog._resize_plugin]._resize_element.style.height=this.context.dialog._resize_h+"px",this.context.element.resizeBackground.style.display="none",this.context.dialog.resizeContainer.style.display="none",this.context.dialog.resizeButton.style.display="none",this.plugins[this.context.dialog._resize_plugin].init.call(this)},onClick_resizeButton:function(e){e.stopPropagation();const t=e.target.getAttribute("data-command")||e.target.parentNode.getAttribute("data-command");t&&(e.preventDefault(),/^\d+$/.test(t)?this.plugins[this.context.dialog._resize_plugin].setSize.call(this,t+"%",""):/update/.test(t)?this.plugins[this.context.dialog._resize_plugin].openModify.call(this):/delete/.test(t)&&this.plugins[this.context.dialog._resize_plugin].destroy.call(this),this.submenuOff(),this.focus())},onMouseDown_resize_handle:function(e){const t=this.context.dialog._resize_direction=e.target.classList[0];e.stopPropagation(),e.preventDefault(),this.context.dialog.resizeDot.style.display="none",this.context.dialog._resizeClientX=e.clientX,this.context.dialog._resizeClientY=e.clientY,this.context.element.resizeBackground.style.display="block",this.context.dialog.resizeButton.style.display="none",this.context.dialog.resizeDiv.style.float=/l/.test(t)?"right":/r/.test(t)?"left":"none";const i=this.plugins.dialog.resizing_element.bind(this),n=function(){this.plugins.dialog.cancel_controller_resize.call(this),document.removeEventListener("mousemove",i),document.removeEventListener("mouseup",n)}.bind(this);document.addEventListener("mousemove",i),document.addEventListener("mouseup",n)},resizing_element:function(e){const t=this.context.dialog._resize_direction,i=e.clientX,n=e.clientY,o=this.context[this.context.dialog._resize_plugin];let l=o._element_w,s=o._element_h;const a=o._element_w+(/r/.test(t)?i-this.context.dialog._resizeClientX:this.context.dialog._resizeClientX-i),r=o._element_h+(/b/.test(t)?n-this.context.dialog._resizeClientY:this.context.dialog._resizeClientY-n),d=o._element_h/o._element_w*a;/t/.test(t)&&(this.context.dialog.resizeDiv.style.top=o._element_h-(/h/.test(t)?r:d)+"px"),/l/.test(t)&&(this.context.dialog.resizeDiv.style.left=o._element_w-a+"px"),/r|l/.test(t)&&(this.context.dialog.resizeDiv.style.width=a+"px",l=a),/^(?:t|b)[^h]$/.test(t)?(this.context.dialog.resizeDiv.style.height=d+"px",s=d):/^(?:t|b)h$/.test(t)&&(this.context.dialog.resizeDiv.style.height=r+"px",s=r),this.context.dialog._resize_w=l,this.context.dialog._resize_h=s,this.util.changeTxt(this.context.dialog.resizeDisplay,Math.round(l)+" x "+Math.round(s))}}},"3FqI":function(e,t,i){},"50IV":function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_exports__.a={name:"font",add:function(_this,targetElement){let listDiv=eval(this.setSubmenu(_this.context.user));listDiv.getElementsByClassName("list_family")[0].addEventListener("click",this.pickup.bind(_this)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=document.createElement("DIV");let i,n,o,l;t.className="layer_editor",t.style.display="none";let s=e.font?e.font:["Arial","Comic Sans MS","Courier New,Courier","Georgia","tahoma","Trebuchet MS,Helvetica","Verdana"],a='
",a+="
",t.innerHTML=a,t},pickup:function(e){if(!/^BUTTON$/i.test(e.target.tagName))return!1;e.preventDefault(),e.stopPropagation();const t=e.target;this.focus(),this.util.changeTxt(this.context.tool.font,t.getAttribute("data-txt"));const i=document.createElement("SPAN");i.style.fontFamily=t.getAttribute("data-value"),this.wrapRangeToTag(i,["font-family"]),this.submenuOff()}}},KKur:function(module,__webpack_exports__,__webpack_require__){"use strict";var _modules_dialog__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("1kvd");__webpack_exports__.a={name:"image",add:function(_this){_this._addModule(_modules_dialog__WEBPACK_IMPORTED_MODULE_0__.a);const context=_this.context;context.image={_linkElement:null,_element:null,_resize_element:null,_element_w:1,_element_h:1,_element_l:0,_element_t:0,_origin_w:context.user.imageSize,_origin_h:0,_altText:"",_imageCaption:null,_linkValue:"",_align:"none",_captionChecked:!1,_proportionChecked:!0,_onCaption:!1,_floatClassRegExp:"float\\-[a-z]+"};let image_dialog=eval(this.setDialog(_this.context.user,_this.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.caption=image_dialog.querySelector("#suneditor_image_check_caption"),context.image.proportion=image_dialog.querySelector("#suneditor_image_check_proportion"),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=_this.context.user.imageSize,context.image.modal.getElementsByClassName("sun-editor-tab-button")[0].addEventListener("click",this.openTab.bind(_this)),context.image.modal.getElementsByClassName("btn-primary")[0].addEventListener("click",this.submit.bind(_this)),context.image.imageX.addEventListener("change",this.setInputSize.bind(_this,"x")),context.image.imageY.addEventListener("change",this.setInputSize.bind(_this,"y")),image_dialog.getElementsByClassName("sun-editor-id-image-revert-button")[0].addEventListener("click",this.sizeRevert.bind(_this)),context.dialog.modal.appendChild(image_dialog),image_dialog=null},setDialog:function(e,t){const i=document.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="init"===e?document.getElementsByClassName("sun-editor-id-tab-link")[0]:e.target;if(!/^BUTTON$/i.test(t.tagName))return!1;const i=t.getAttribute("data-tab-link");let n,o,l;for(o=document.getElementsByClassName("sun-editor-id-tab-content"),n=0;n0){const t=this.context.user.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:result.length;e0){const n=document.createElement("A");return n.href=/^https?:\/\//.test(t)?t:"http://"+t,n.target=i?"_blank":"",n.setAttribute("data-image-link","image"),n.addEventListener("click",function(e){e.preventDefault()}),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.caption.checked,this.context.image._proportionChecked=this.context.image.proportion.checked;try{this.context.dialog.updateModal?this.plugins.image.update_image.call(this):(this.plugins.image.onRender_imgInput.call(this),this.plugins.image.onRender_imgUrl.call(this))}finally{this.plugins.dialog.closeDialog.call(this),this.closeLoading()}return!1},create_caption:function(){const e=document.createElement("FIGCAPTION");return e.innerHTML="

"+this.lang.dialogBox.imageBox.caption+"

",e.addEventListener("click",this.plugins.image.toggle_caption_contenteditable.bind(this,!0)),e},set_cover:function(e){const t=document.createElement("FIGURE");return t.className="sun-editor-image-cover",t.appendChild(e),t},set_container:function(e){const t=document.createElement("DIV");return t.className="sun-editor-id-image-container",t.setAttribute("contenteditable",!1),t.appendChild(e),t},create_image:function(e,t,i,n,o,l){if(l)return void(this.context.image._element.src=e);let s=document.createElement("IMG");s.src=e,s.style.width=n,s.setAttribute("data-align",o),s.setAttribute("data-proportion",this.context.image._proportionChecked),s.alt=this.context.image._altText,s=this.plugins.image.onRender_link(s,t,i);const a=this.plugins.image.set_cover.call(this,s),r=this.plugins.image.set_container.call(this,a);this.context.image._captionChecked&&(this.context.image._imageCaption=this.plugins.image.create_caption.call(this),this.context.image._imageCaption.setAttribute("contenteditable",!1),a.appendChild(this.context.image._imageCaption)),"center"!==o&&(r.style.display="inline-block",this.util.removeClass(r,this.context.image._floatClassRegExp),this.util.addClass(r,"float-"+o)),this.insertNode(r,this.util.getFormatElement(this.getSelectionNode())),this.appendP(r)},update_image:function(){const e=this.context.image,t=e._linkValue;let i=this.util.getParentElement(e._element,".sun-editor-image-cover"),n=this.util.getParentElement(e._element,".sun-editor-id-image-container"),o=!1;if(null===i&&(o=!0,i=this.plugins.image.set_cover.call(this,e._element.cloneNode(!0))),null===n?(o=!0,n=this.plugins.image.set_container.call(this,i.cloneNode(!0))):o&&(n.innerHTML="",n.appendChild(i)),this.plugins.image.onRender_imgInput.call(this),e._element.src=e.imgUrlFile.value,e._element.alt=e._altText,e._element.setAttribute("data-proportion",e._proportionChecked),e._element.style.width=e.imageX.value+"px",e._element.style.height=e.imageY.value+"px",e._captionChecked?null===e._imageCaption&&(e._imageCaption=this.plugins.image.create_caption.call(this),i.appendChild(e._imageCaption)):e._imageCaption&&this.util.removeItem(e._imageCaption),"center"!==e._align?(n.style.display="inline-block",this.util.removeClass(n,this.context.image._floatClassRegExp),this.util.addClass(n,"float-"+e._align)):(n.style.display="",this.util.removeClass(n,this.context.image._floatClassRegExp),this.util.addClass(n,"float-none")),e._element.setAttribute("data-align",e._align),t.trim().length>0)if(null!==e._linkElement)e._linkElement.href=t,e._linkElement.target=e.imgLinkNewWindowCheck.checked?"_blank":"",e._element.setAttribute("data-image-link",t);else{let n=this.plugins.image.onRender_link(e._element.cloneNode(!0),t,this.context.image.imgLinkNewWindowCheck.checked);i.removeChild(e._element),i.insertBefore(n,e._imageCaption)}else if(null!==e._linkElement){const t=e._element;t.setAttribute("data-image-link","");let n=t.cloneNode(!0);i.removeChild(e._linkElement),i.insertBefore(n,e._imageCaption)}if(o){const t=this.util.getFormatElement(e._element);t.parentNode.insertBefore(n,t),this.util.removeItem(t)}},toggle_caption_contenteditable:function(e,t){this.context.image._onCaption=e,this.context.image._imageCaption.setAttribute("contenteditable",e),this.context.image._imageCaption.focus()},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=i._resize_element=e,i._imageCaption=i._linkElement?i._linkElement.nextSibling:e.nextSibling,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(){const e=this.context.image;e.imgUrlFile.value=e._element.src,e.altText.value=e._element.alt,e.imgLink.value=null===e._linkElement?"":e._linkElement.href,e.imgLinkNewWindowCheck.checked=e._linkElement&&"_blank"===e._linkElement.target,e.modal.querySelector("#suneditor_image_radio_"+(e._element.getAttribute("data-align")||"none")).checked=!0,e._captionChecked=e.caption.checked=!!e._imageCaption,e.proportion.checked=e._proportionChecked="true"===e._element.getAttribute("data-proportion"),e.imageX.value=e._element.offsetWidth,e.imageY.value=e._element.offsetHeight,e.imageY.disabled=!1,e.proportion.disabled=!1,this.plugins.dialog.openDialog.call(this,"image",null,!0)},setSize:function(e,t){this.context.image._resize_element.style.width=e,this.context.image._resize_element.style.height=t},destroy:function(){const e=this.util.getParentElement(this.context.image._element,".sun-editor-id-image-container")||this.context.image._element;this.util.removeItem(e),this.plugins.image.init.call(this)},init:function(){this.context.image.imgInputFile.value="",this.context.image.imgUrlFile.value="",this.context.image.altText.value="",this.context.image.imgLink.value="",this.context.image.imgLinkNewWindowCheck.checked=!1,this.context.image.modal.querySelector("#suneditor_image_radio_none").checked=!0,this.context.image.caption.checked=!1,this.context.image.proportion.checked=!1,this.context.image.imageX.value=this.context.user.imageSize,this.context.image.imageY.value="",this.context.image.imageY.disabled=!0,this.context.image.proportion.disabled=!0,this.context.image._element=null,this.plugins.image.openTab.call(this,"init")}}},MIhV:function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_exports__.a={name:"fontSize",add:function(_this,targetElement){let listDiv=eval(this.setSubmenu(_this.context.user));listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickup.bind(_this)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=document.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;this.focus(),this.util.changeTxt(this.context.tool.fontSize,e.target.getAttribute("data-value"));const t=document.createElement("SPAN");t.style.fontSize=e.target.getAttribute("data-value")+"px",this.wrapRangeToTag(t,["font-size"]),this.submenuOff()}}},Rp48:function(module,__webpack_exports__,__webpack_require__){"use strict";var _modules_dialog__WEBPACK_IMPORTED_MODULE_0__=__webpack_require__("1kvd");__webpack_exports__.a={name:"link",add:function(_this){_this._addModule(_modules_dialog__WEBPACK_IMPORTED_MODULE_0__.a);const context=_this.context;context.link={};let link_dialog=eval(this.setDialog(_this.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_LinkBtn(_this.lang));context.link.linkBtn=link_button,context.link._linkAnchor=null,link_dialog.getElementsByClassName("btn-primary")[0].addEventListener("click",this.submit.bind(_this)),link_button.addEventListener("click",this.onClick_linkBtn.bind(_this)),context.dialog.modal.appendChild(link_dialog),context.element.relative.appendChild(link_button),link_dialog=null,link_button=null},setDialog:function(e){const t=document.createElement("DIV");return t.className="modal-content sun-editor-id-dialog-link",t.style.display="none",t.innerHTML='",t},setController_LinkBtn:function(e){const t=document.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();try{(function(){if(0===this.context.link.focusElement.value.trim().length)return!1;const e=/^https?:\/\//.test(this.context.link.focusElement.value)?this.context.link.focusElement.value:"http://"+this.context.link.focusElement.value,t=this.context.link.linkAnchorText||this.context.dialog.document.getElementById("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=document.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=""}).call(this)}finally{this.plugins.dialog.closeDialog.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,t.style.left=e.offsetLeft+"px",t.style.top=e.offsetTop+e.offsetHeight-this.context.element.wysiwyg.scrollTop+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.openDialog.call(this,"link",null,!0)):(this.util.removeItem(this.context.link._linkAnchor),this.context.link._linkAnchor=null,this.focus()),this.context.link.linkBtn.style.display="none")},init:function(){this.context.link.linkBtn.style.display="none",this.context.link._linkAnchor=null,this.context.link.focusElement.value="",this.context.link.linkAnchorText.value="",this.context.link.linkNewWindowCheck.checked=!1}}},VquE:function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_exports__.a={name:"table",add:function(_this,targetElement){const context=_this.context;let listDiv=eval(this.setSubmenu()),tablePicker=listDiv.getElementsByClassName("sun-editor-id-table-picker")[0];context.submenu.tableHighlight=listDiv.getElementsByClassName("sun-editor-id-table-highlighted")[0],context.submenu.tableUnHighlight=listDiv.getElementsByClassName("sun-editor-id-table-unhighlighted")[0],context.submenu.tableDisplay=listDiv.getElementsByClassName("sun-editor-table-display")[0],context.submenu._tableXY=[],tablePicker.addEventListener("mousemove",this.onMouseMove_tablePicker.bind(_this)),tablePicker.addEventListener("click",this.appendTable.bind(_this)),targetElement.parentNode.appendChild(listDiv),listDiv=null,tablePicker=null},setSubmenu:function(){const e=document.createElement("DIV");return e.className="table-content",e.style.display="none",e.innerHTML='
1 x 1
',e},appendTable:function(){const e=document.createElement("TABLE");let t=this.context.submenu._tableXY[0],i=this.context.submenu._tableXY[1],n="";for(;i>0;){n+="";let e=t;for(;e>0;)n+="



",--e;n+="",--i}n+="",e.innerHTML=n,this.insertNode(e,this.util.getFormatElement(this.getSelectionNode())),this.appendP(e),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.submenu.tableHighlight.style.width=t+"em",this.context.submenu.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.submenu.tableUnHighlight.style.width=n+"em",this.context.submenu.tableUnHighlight.style.height=o+"em",this.util.changeTxt(this.context.submenu.tableDisplay,t+" x "+i),this.context.submenu._tableXY=[t,i]},reset_table_picker:function(){if(!this.context.submenu.tableHighlight)return;const e=this.context.submenu.tableHighlight.style,t=this.context.submenu.tableUnHighlight.style;e.width="1em",e.height="1em",t.width="5em",t.height="5em",this.util.changeTxt(this.context.submenu.tableDisplay,"1 x 1"),this.submenuOff()}}},WRt5:function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_exports__.a={name:"align",add:function(_this,targetElement){let listDiv=eval(this.setSubmenu(_this.lang));listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickup.bind(_this)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=document.createElement("DIV");return t.className="layer_editor layer_align",t.style.display="none",t.innerHTML='
",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),this.submenuOff()}}},WUQj:function(e,t,i){},WzUB:function(module,__webpack_exports__,__webpack_require__){"use strict";__webpack_exports__.a={name:"formatBlock",add:function(_this,targetElement){let listDiv=eval(this.setSubmenu(_this.lang));listDiv.getElementsByTagName("UL")[0].addEventListener("click",this.pickUp.bind(_this)),targetElement.parentNode.appendChild(listDiv),listDiv=null},setSubmenu:function(e){const t=document.createElement("DIV");return t.className="layer_editor layer_size",t.style.display="none",t.innerHTML='
",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-value"),t=t.parentNode;this.focus(),this.util.changeTxt(this.commandMap.FORMAT,i),this.execCommand("formatBlock",!1,i),this.submenuOff()}}},XJR1:function(e,t,i){"use strict";i.r(t);i("3FqI"),i("WUQj");var n=i("WRt5"),o=i("50IV"),l=i("MIhV"),s=i("0A7J"),a=i("s0fJ"),r=i("g4XY"),d=i("gMuy"),c=i("VquE"),u=i("WzUB"),m=i("Rp48"),g=i("KKur"),h=i("hlhS"),p={align:n.a,font:o.a,fontSize:l.a,fontColor:s.a,hiliteColor:a.a,horizontalRule:r.a,list:d.a,table:c.a,formatBlock:u.a,link:m.a,image:g.a,video:h.a};var _={util:{getXMLHttpRequest:function(){if(!window.ActiveXObject)return window.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},getIncludePath:function(e,t){let i="";const n="js"===t?"script":"link",o="js"===t?"src":"href";let l="(?:";for(let t=0,i=e.length;t"+i+"

";return 0===n.length&&(n="



"),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\\d)$/i.test(e.tagName))},getFormatElement:function(e){if(!e)return null;if(!e||this.isWysiwygDiv(e))e=context.element.wysiwyg.firstChild;else for(;!this.isWysiwygDiv(e.parentNode);)e=e.parentNode;return e},getArrayIndex:function(e,t){let i=-1;for(let n=0,o=e.length;n0){for(let t=0;t0?e.getRangeAt(0):this._createDefaultRange(),this._variable.range=t,t.collapsed?this._variable.selectionNode=t.commonAncestorContainer:this._variable.selectionNode=e.extentNode||e.anchorNode},_createDefaultRange:function(){const t=document.createRange();return t.setStart(e.element.wysiwyg.firstChild,0),t.setEnd(e.element.wysiwyg.firstChild,0),t},setRange:function(e,t,i,n){const o=document.createRange();o.setStart(e,t),o.setEnd(i,n);const l=window.getSelection();l.rangeCount>0&&l.removeAllRanges(),this._variable.range=o,l.addRange(o)},getRange:function(){return this._variable.range||this._createDefaultRange()},getSelectionNode:function(){return this._variable.selectionNode?this._variable.selectionNode:e.element.wysiwyg.firstChild},isEdgePoint:function(e,t){return 0===t||t===e.nodeValue.length},showLoading:function(){e.element.loading.style.display="block"},closeLoading:function(){e.element.loading.style.display="none"},appendP:function(e){const i=document.createElement("P");return i.innerHTML="",(e=t.getFormatElement(e)).parentNode.insertBefore(i,e.nextElementSibling),i},insertNode:function(e,t){const i=this.getRange();let n=null;if(t)n=t.parentNode,t=t.nextSibling;else{const e=i.startContainer,o=i.startOffset,l=i.endContainer,s=i.endOffset,a=i.commonAncestorContainer;if(n=e,3===e.nodeType&&(n=e.parentNode),i.collapsed)3===a.nodeType?t=a.splitText(s):(null!==n.lastChild&&/^BR$/i.test(n.lastChild.nodeName)&&n.removeChild(n.lastChild),t=null);else if(e===l){let i=e;this.isEdgePoint(l,s)||(t=l.splitText(s)),this.isEdgePoint(e,o)||(i=e.splitText(o)),n.removeChild(i)}else for(this.removeNode(),t=l;1!==t.nodeType;)t=t.parentNode}try{n.insertBefore(e,t)}catch(t){n.appendChild(e)}},removeNode:function(){const e=this.getRange();if(e.deleteContents)return void e.deleteContents();const i=e.startContainer,n=e.startOffset,o=e.endContainer,l=e.endOffset,s=e.commonAncestorContainer;let a=null,r=null;const d=t.getListChildNodes(s);let c=t.getArrayIndex(d,i),u=t.getArrayIndex(d,o);for(let e=c+1,t=i;e>=0;e--)d[e]===t.parentNode&&d[e].firstChild===t&&0===n&&(c=e,t=t.parentNode);for(let e=u-1,t=o;e>c;e--)d[e]===t.parentNode&&1===d[e].nodeType&&(d.splice(e,1),t=t.parentNode,--u);for(let e=c;e<=u;e++){const s=d[e];0===s.length||3===s.nodeType&&void 0===s.data?t.removeItem(s):s!==i?s!==o?t.removeItem(s):(r=1===o.nodeType?document.createTextNode(o.textContent):document.createTextNode(o.substringData(l,o.length-l))).length>0?o.data=r.data:t.removeItem(o):(a=1===i.nodeType?document.createTextNode(i.textContent):document.createTextNode(i.substringData(0,n))).length>0?i.data=a.data:t.removeItem(i)}},wrapRangeToTag:function(e,i){const n=this.getRange(),o=n.startContainer,l=n.startOffset,s=n.endContainer,a=n.endOffset,r=n.commonAncestorContainer;let d,c,u={},m={};if(i){c="(?:;|^|\\s)(?:"+i[0];for(let e=1;e0?o.data=e.data:o.data=o.substringData(0,l),t.data.length>0&&o.parentNode.insertBefore(t,d.nextSibling)}u.container=d,u.offset=1,m.container=d,m.offset=1}else{const i=function(t){if(3===t.nodeType)return!0;let i="";return c&&t.style.cssText.length>0&&(i=t.style.cssText.replace(c,"").trim()),(t.nodeName!==e.nodeName||i.length>0)&&(t.style.cssText.length>0&&(t.style.cssText=i),!0)};if(t.hasClass(r,"sun-editor-id-wysiwyg")){const n=t.getListChildren(r,function(e){return t.isFormatElement(e)});let c=t.getParentElement(o,"P"),g=t.getParentElement(s,"P");for(let e=0,t=n.length;e0){for(c=d=r.pop();r.length>0;)d=r.pop(),c.appendChild(d);i.appendChild(c),o=d}else o=i}if(g||h!==u){if(!g||n(h)){const e=h.cloneNode(!1);o.appendChild(e),1===h.nodeType&&(p=e)}e(h,p)}else{const e=document.createTextNode(u.substringData(0,m)),t=document.createTextNode(u.substringData(m,u.length-m));for(e.data.length>0&&o.appendChild(e),d=o,r=[];d!==a&&null!==d;)n(d)&&1===d.nodeType&&r.push(d.cloneNode(!1)),d=d.parentNode;for(c=d=r.pop()||o;r.length>0;)d=r.pop(),c.appendChild(d);c!==o?(i.appendChild(c),o=d):o=i,a.appendChild(i),u=t,m=0,g=!0,o.appendChild(u)}}}(e,a),e.parentNode.insertBefore(a,e),t.removeItem(e),{container:u,offset:m}},_wrapLineNodesEnd:function(e,i,n,o,l){const s=e,a=document.createElement("P");let r,d,c,u=o,m=l,g=!1;return function e(t,o){const l=t.childNodes;for(let t=l.length-1;0<=t;t--){const h=l[t];let p=o;if(g){if(1===h.nodeType){e(h,h);continue}for(d=h,r=[];null!==d.parentNode&&d!==s&&d!==i;)n(d)&&1===d.nodeType&&r.push(d.cloneNode(!1)),d=d.parentNode;if(r.length>0){for(c=d=r.pop();r.length>0;)d=r.pop(),c.insertBefore(d,c.firstChild);i.insertBefore(c,i.firstChild),o=d}else o=i}if(g||h!==u){if(!g||n(h)){const e=h.cloneNode(!1);o.insertBefore(e,o.firstChild),1===h.nodeType&&(p=e)}e(h,p)}else{const e=document.createTextNode(u.substringData(m,u.length-m)),t=document.createTextNode(u.substringData(0,m));for(e.data.length>0&&o.insertBefore(e,o.firstChild),d=o,r=[];d!==a&&null!==d;)n(d)&&1===d.nodeType&&r.push(d.cloneNode(!1)),d=d.parentNode;for(c=d=r.pop()||o;r.length>0;)d=r.pop(),c.insertBefore(d,c.firstChild);c!==o?(i.insertBefore(c,i.firstChild),o=d):o=i,a.insertBefore(i,a.firstChild),u=t,m=t.data.length,g=!0,o.insertBefore(u,o.firstChild)}}}(e,a),e.parentNode.insertBefore(a,e),t.removeItem(e),{container:u,offset:m}},indent:function(e,i){const n=t.getParentElement(e,"P");if(!n)return;let o=/\d+/.test(n.style.marginLeft)?1*n.style.marginLeft.match(/\d+/)[0]:0;"indent"===i?o+=25:o-=25,n.style.marginLeft=(o<0?0:o)+"px"},toggleDisplayBlocks:function(){t.toggleClass(e.element.wysiwyg,"sun-editor-show-block")},toggleCodeView:function(){if(this._variable.wysiwygActive)e.element.code.value=e.element.wysiwyg.innerHTML.trim().replace(/<\/p>(?=[^\n])/gi,"

\n"),e.element.wysiwyg.style.display="none",e.element.code.style.display="block",this._variable.wysiwygActive=!1;else{const t={"&":"&"," ":" ",""":"'","<":"<",">":">"},i=e.element.code.value.replace(/&[a-z]+;/g,function(e){return"string"==typeof t[e]?t[e]:e});e.element.wysiwyg.innerHTML=i.trim().length>0?i:"



",e.element.wysiwyg.scrollTop=0,e.element.code.style.display="none",e.element.wysiwyg.style.display="block",this._variable.wysiwygActive=!0}},toggleFullScreen:function(i){this._variable.isFullScreen?(e.element.topArea.style.cssText=this._variable.originCssText,e.element.editorArea.style.height=this._variable.editorHeight+"px",t.removeClass(i.firstElementChild,"icon-reduction"),t.addClass(i.firstElementChild,"icon-expansion")):(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.innerHeight_fullScreen=window.innerHeight-e.tool.bar.offsetHeight,e.element.editorArea.style.height=this._variable.innerHeight_fullScreen+"px",t.removeClass(i.firstElementChild,"icon-expansion"),t.addClass(i.firstElementChild,"icon-reduction")),this._variable.isFullScreen=!this._variable.isFullScreen},openWindowContents:function(i){const n="print"===i,l=window.open("","_blank");l.mimeType="text/html",l.document.write(''+(n?o.toolbar.print:o.toolbar.preview)+''+e.element.wysiwyg.innerHTML+""+(n?"' : '') + '');\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._variable.selectionNode; !util.isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\n if (selectionParent.nodeType !== 1) continue;\n nodeName = selectionParent.nodeName.toUpperCase();\n currentNodes.push(nodeName);\n /** Format */\n\n if (findFormat && util.isFormatElement(selectionParent)) {\n commandMapNodes.push('FORMAT');\n util.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.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.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.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.changeTxt(commandMap[key], lang.toolbar.font);\n } else if (/^SIZE$/i.test(key)) {\n util.changeTxt(commandMap[key], lang.toolbar.fontSize);\n } else {\n util.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 resizebar */\n\n if (context.user.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\n },\n resize_window: function () {\n if (editor._variable.isFullScreen) {\n editor._variable.innerHeight_fullScreen += window.innerHeight - context.tool.bar.offsetHeight - editor._variable.innerHeight_fullScreen;\n context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\n }\n },\n onMouseDown_toolbar: function (e) {\n e.preventDefault();\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 /** 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.openDialog.call(editor, command, target.getAttribute('data-option'), 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\n switch (command) {\n case 'codeView':\n editor.toggleCodeView();\n util.toggleClass(target, 'on');\n break;\n\n case 'fullScreen':\n editor.toggleFullScreen(target);\n util.toggleClass(target, 'on');\n break;\n\n case 'indent':\n case 'outdent':\n editor.indent(editor._variable.selectionNode, command);\n break;\n\n case 'redo':\n case 'undo':\n case 'removeFormat':\n editor.execCommand(command, false, null);\n break;\n\n case 'preview':\n case 'print':\n editor.openWindowContents(command);\n break;\n\n case 'showBlocks':\n editor.toggleDisplayBlocks();\n util.toggleClass(target, 'on');\n break;\n\n case 'subscript':\n if (util.hasClass(context.tool.superscript, 'on')) {\n editor.execCommand('superscript', false, null);\n util.removeClass(context.tool.superscript, 'on');\n }\n\n editor.execCommand(command, false, null);\n util.toggleClass(target, 'on');\n break;\n\n case 'superscript':\n if (util.hasClass(context.tool.subscript, 'on')) {\n editor.execCommand('subscript', false, null);\n util.removeClass(context.tool.subscript, 'on');\n }\n\n editor.execCommand(command, false, null);\n util.toggleClass(target, 'on');\n break;\n\n default:\n editor.execCommand(command, false, target.getAttribute('data-value'));\n util.toggleClass(target, 'on');\n }\n }\n },\n onMouseUp_wysiwyg: function (e) {\n e.stopPropagation();\n const targetElement = e.target;\n editor.submenuOff();\n\n if (/^HTML$/i.test(targetElement.nodeName)) {\n e.preventDefault();\n editor.focus();\n return;\n }\n\n if (/^IMG$/i.test(targetElement.nodeName)) {\n e.preventDefault();\n editor.callPlugin('image', function () {\n const size = editor.plugins.dialog.call_controller_resize.call(editor, targetElement, 'image');\n editor.plugins.image.onModifyMode.call(editor, targetElement, size);\n });\n return;\n }\n\n editor._setEditorRange();\n\n event._findButtonEffectTag();\n },\n onKeyDown_wysiwyg: function (e) {\n // editor._setSelectionNode();\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 function shortcutCommand(keyCode) {\n const key = event._shortcutKeyCode[keyCode];\n if (!key) return false;\n editor.execCommand(key[0], false, null);\n util.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 switch (keyCode) {\n case 8:\n /**backspace key*/\n if (util.isFormatElement(editor._variable.selectionNode) && editor._variable.selectionNode.previousSibling === null) {\n e.preventDefault();\n e.stopPropagation();\n editor._variable.selectionNode.innerHTML = '';\n return false;\n }\n\n break;\n\n case 9:\n /**tab key*/\n e.preventDefault();\n if (ctrl || alt) break;\n let currentNode = editor._variable.selectionNode || editor.getSelectionNode();\n\n while (!/^TD$/i.test(currentNode.tagName) && !util.isWysiwygDiv(currentNode)) {\n currentNode = currentNode.parentNode;\n }\n\n if (currentNode && /^TD$/i.test(currentNode.tagName)) {\n const table = util.getParentElement(currentNode, 'table');\n const cells = util.getListChildren(table, util.isCell);\n let idx = shift ? util.prevIdx(cells, currentNode) : util.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 /** if P Tag */\n\n\n if (shift) break;\n const tabText = document.createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\n editor.insertNode(tabText);\n editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize);\n break;\n }\n },\n onKeyUp_wysiwyg: function (e) {\n editor._setEditorRange();\n /** when format tag deleted */\n\n\n if (e.keyCode === 8 && util.isWysiwygDiv(editor._variable.selectionNode)) {\n e.preventDefault();\n e.stopPropagation();\n const oFormatTag = document.createElement(editor._variable.currentNodes[0]);\n oFormatTag.innerHTML = '';\n\n editor._variable.selectionNode.appendChild(oFormatTag);\n\n editor._variable.selectionNode = oFormatTag;\n editor.setRange(oFormatTag, 0, oFormatTag, 0);\n }\n\n if (event._directionKeyKeyCode.test(e.keyCode)) {\n event._findButtonEffectTag();\n }\n },\n onScroll_wysiwyg: function () {\n editor.controllersOff();\n },\n onDrop_wysiwyg: function (e) {\n const files = e.dataTransfer.files;\n if (files.length === 0) return true;\n e.stopPropagation();\n e.preventDefault();\n\n if (editor.plugins.image) {\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 onMouseDown_resizeBar: 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 document.removeEventListener('mousemove', event.resize_editor);\n document.removeEventListener('mouseup', closureFunc);\n }\n\n document.addEventListener('mousemove', event.resize_editor);\n document.addEventListener('mouseup', closureFunc);\n },\n resize_editor: function (e) {\n const resizeInterval = e.clientY - editor._variable.resizeClientY;\n context.element.editorArea.style.height = context.element.editorArea.offsetHeight + resizeInterval + 'px';\n editor._variable.editorHeight = context.element.editorArea.offsetHeight + resizeInterval;\n editor._variable.resizeClientY = e.clientY;\n }\n };\n /** add event listeners */\n\n /** tool bar event */\n\n context.tool.bar.addEventListener('click', event.onClick_toolbar, false);\n context.tool.bar.addEventListener('mousedown', event.onMouseDown_toolbar, false);\n /** editor area */\n\n context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\n context.element.wysiwyg.addEventListener('mouseup', event.onMouseUp_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 /** resize bar */\n\n context.element.resizebar.addEventListener('mousedown', event.onMouseDown_resizeBar, false);\n /** window resize event */\n\n window.addEventListener('resize', event.resize_window, false);\n /** add plugin to plugins object */\n\n if (plugins) {\n let pluginsValues = Object.values(plugins);\n\n for (let i = 0, len = pluginsValues.length, plugin; i < len; i++) {\n plugin = pluginsValues[i];\n editor.plugins[plugin.name] = util.copyObj(plugin);\n }\n\n pluginsValues = null;\n }\n /** User function */\n\n\n return {\n /**\r\n * @description Copying the contents of the editor to the original textarea\r\n */\n save: function () {\n if (editor._variable.wysiwygActive) {\n context.element.originElement.value = context.element.wysiwyg.innerHTML;\n } else {\n context.element.originElement.value = context.element.code.value;\n }\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 getContent: function () {\n let content = '';\n if (context.element.wysiwyg.innerText.trim().length === 0) return content;\n\n if (editor._variable.wysiwygActive) {\n content = context.element.wysiwyg.innerHTML;\n } else {\n content = context.element.code.value;\n }\n\n return content;\n },\n\n /**\r\n * @description Change the contents of the suneditor\r\n * @param {String} content - Content to Input\r\n */\n setContent: function (content) {\n const innerHTML = util.convertContentForEditor(content);\n\n if (editor._variable.wysiwygActive) {\n context.element.wysiwyg.innerHTML = innerHTML;\n } else {\n context.element.code.value = innerHTML;\n }\n },\n\n /**\r\n * @description Add content to the suneditor\r\n * @param {String} content - to Input\r\n */\n appendContent: function (content) {\n if (editor._variable.wysiwygActive) {\n const oP = document.createElement('P');\n oP.innerHTML = content;\n context.element.wysiwyg.appendChild(oP);\n } else {\n context.element.code.value += oP.outerHTML;\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 },\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 },\n\n /**\r\n * @description Show the suneditor\r\n */\n show: function () {\n const topAreaStyle = context.element.topArea.style;\n topAreaStyle.cssText = editor._variable.originCssText;\n if (topAreaStyle.display === 'none') topAreaStyle.display = 'block';\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 window.removeEventListener('resize', event.resize_window);\n /** remove element */\n\n context.element.topArea.parentNode.removeChild(context.element.topArea);\n this.save = null;\n this.getContext = null;\n this.getContent = null;\n this.setContent = null;\n this.appendContent = null;\n this.disabled = null;\n this.enabled = null;\n this.show = null;\n this.hide = null;\n this.destroy = null;\n }\n };\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":["'use strict';\r\n\r\n/**\r\n * @description SunEditor core closure\r\n * @param context\r\n * @param util\r\n * @param modules\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, util, modules, plugins, lang) {\r\n    /**\r\n     * @description Practical editor function\r\n     * This function is 'this' used by other plugins\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 modules\r\n         */\r\n        modules: {},\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 util function\r\n         */\r\n        util: util,\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 controllers array (image resize area, link modified button)\r\n         */\r\n        controllerArray: [],\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 {Element} originCssText - Remembered the CSS of the editor before full screen (Used when returning to original size again)\r\n         * @property {Number} editorHeight - The height value entered by the user or the height value of the \"textarea\" when the suneditor is created\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            originCssText: context.element.topArea.style.cssText,\r\n            editorHeight: context.user.height,\r\n            currentNodes: []\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.callModule.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 {Object} moduleObj - module object (dialog)\r\n         */\r\n        _addModule: function (moduleObj) {\r\n            const moduleName = moduleObj.name;\r\n            if (!this.plugins[moduleName]) {\r\n                this.plugins[moduleName] = this.util.copyObj(moduleObj);\r\n                this.plugins[moduleName].add(this);\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            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\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            if (context.image && context.image._onCaption === true) {\r\n                this.plugins.image.toggle_caption_contenteditable.call(editor, false);\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\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\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            document.execCommand(command, showDefaultUI, 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._variable.selectionNode, '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 = window.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._variable.selectionNode = range.commonAncestorContainer;\r\n            } else {\r\n                this._variable.selectionNode = 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 = document.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 = document.createRange();\r\n            range.setStart(startCon, startOff);\r\n            range.setEnd(endCon, endOff);\r\n\r\n            const selection = window.getSelection();\r\n\r\n            if (selection.rangeCount > 0) {\r\n                selection.removeAllRanges();\r\n            }\r\n\r\n            this._variable.range = range;\r\n            selection.addRange(range);\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 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 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 P tag to current line next\r\n         * @param {Element} element - Insert as siblings of that element\r\n         * @returns {Element}\r\n         */\r\n        appendP: function (element) {\r\n            const oP = document.createElement('P');\r\n            oP.innerHTML = '&#65279';\r\n\r\n            element = util.getFormatElement(element);\r\n            element.parentNode.insertBefore(oP, element.nextElementSibling);\r\n\r\n            return oP;\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                /** Select within the same 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 && /^BR$/i.test(parentNode.lastChild.nodeName)) {\r\n                            parentNode.removeChild(parentNode.lastChild);\r\n                        }\r\n                        rightNode = null;\r\n                    }\r\n                }\r\n                /** Select multiple nodes */\r\n                else {\r\n                    const isSameContainer = startCon === endCon;\r\n\r\n                    if (isSameContainer) {\r\n                        let removeNode = startCon;\r\n                        if (!this.isEdgePoint(endCon, endOff)) rightNode = endCon.splitText(endOff);\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                        rightNode = endCon;\r\n\r\n                        while (rightNode.nodeType !== 1) {\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            // this.setRange(oNode, 0, oNode, 0);\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 = document.createTextNode(startCon.textContent);\r\n                    } else {\r\n                        beforeNode = document.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 = document.createTextNode(endCon.textContent);\r\n                    } else {\r\n                        afterNode = document.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 Copies the node of the argument value and wraps all selected text.\r\n         * 1. When there is the same 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 same node, removed only Css attribute values received as arguments\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']...])\r\n         */\r\n        wrapRangeToTag: function (appendNode, checkCSSPropertyArray) {\r\n            const range = this.getRange();\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 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, 'gi');\r\n            }\r\n\r\n            /** one node */\r\n            if (startCon === endCon) {\r\n                newNode = appendNode.cloneNode(false);\r\n\r\n                /** No node selected */\r\n                if (startOff === endOff) {\r\n                    newNode.innerHTML = '&#65279';\r\n                    if (util.isFormatElement(startCon)) {\r\n                        startCon.appendChild(newNode);\r\n                    } else {\r\n                        startCon.splitText(startOff);\r\n                        startCon.parentNode.insertBefore(newNode, startCon.nextSibling);\r\n                    }\r\n                }\r\n                /** Select within the same node */\r\n                else {\r\n                    const isElement = startCon.nodeType === 1;\r\n                    if (isElement) {\r\n                        newNode.innerHTML = startCon.outerHTML;\r\n                        startCon.parentNode.appendChild(newNode);\r\n                        util.removeItem(startCon);\r\n                    } else {\r\n                        const beforeNode = document.createTextNode(startCon.substringData(0, startOff));\r\n                        const afterNode = document.createTextNode(startCon.substringData(endOff, (startCon.length - endOff)));\r\n\r\n                        newNode.innerText = startCon.substringData(startOff, (endOff - startOff));\r\n                        startCon.parentNode.insertBefore(newNode, startCon.nextSibling);\r\n\r\n                        if (beforeNode.data.length > 0) {\r\n                            startCon.data = beforeNode.data;\r\n                        } else {\r\n                            startCon.data = startCon.substringData(0, startOff);\r\n                        }\r\n\r\n                        if (afterNode.data.length > 0) {\r\n                            startCon.parentNode.insertBefore(afterNode, newNode.nextSibling);\r\n                        }\r\n                    }\r\n                }\r\n\r\n                start.container = newNode;\r\n                start.offset = 1;\r\n                end.container = newNode;\r\n                end.offset = 1;\r\n            }\r\n            /** multiple nodes */\r\n            else {\r\n                /** tag check function*/\r\n                const checkCss = function (vNode) {\r\n                    if (vNode.nodeType === 3) 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 (vNode.nodeName !== appendNode.nodeName || style.length > 0) {\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.hasClass(commonCon, 'sun-editor-id-wysiwyg')) {\r\n                    newNode = appendNode.cloneNode(false);\r\n                    const newRange = this._wrapLineNodesPart(commonCon, newNode, checkCss, startCon, startOff, endCon, endOff);\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 = util.getListChildren(commonCon, function (current) {\r\n                        return util.isFormatElement(current);\r\n                    });\r\n\r\n                    let startLine = util.getParentElement(startCon, 'P');\r\n                    let endLine = util.getParentElement(endCon, 'P');\r\n\r\n                    for (let i = 0, len = lineNodes.length; i < len; i++) {\r\n                        if (startLine === lineNodes[i]) {\r\n                            startLine = i;\r\n                            continue;\r\n                        }\r\n                        if (endLine === lineNodes[i]) {\r\n                            endLine = i;\r\n                            break;\r\n                        }\r\n                    }\r\n\r\n                    // startCon\r\n                    newNode = appendNode.cloneNode(false);\r\n                    start = this._wrapLineNodesStart(lineNodes[startLine], newNode, checkCss, startCon, startOff);\r\n                    // mid\r\n                    for (let i = startLine + 1; i < endLine; i++) {\r\n                        newNode = appendNode.cloneNode(false);\r\n                        this._wrapLineNodes(lineNodes[i], newNode, checkCss);\r\n                    }\r\n                    // endCon\r\n                    newNode = appendNode.cloneNode(false);\r\n                    end = this._wrapLineNodesEnd(lineNodes[endLine], newNode, checkCss, endCon, endOff);\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         * @returns {{startContainer: *, startOffset: *, endContainer: *, endOffset: *}}\r\n         * @private\r\n         */\r\n        _wrapLineNodesPart: function (element, newInnerNode, validation, startCon, startOff, endCon, endOff) {\r\n            const el = element;\r\n            const removeNodeList = [];\r\n\r\n            let startContainer = startCon;\r\n            let startOffset = startOff;\r\n            let endContainer = endCon;\r\n            let endOffset = endOff;\r\n            let prevNode, afterNode;\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\r\n                    if (validation(child)) {\r\n                        let cloneNode;\r\n\r\n                        if (child === startContainer) {\r\n                            prevNode = document.createTextNode(startContainer.substringData(0, startOffset));\r\n                            child = document.createTextNode(startContainer.substringData(startOffset, (startContainer.length - startOffset)));\r\n                            startOffset = 0;\r\n                            startContainer = cloneNode = child.cloneNode(false);\r\n                        }\r\n                        else if (child === endContainer) {\r\n                            afterNode = document.createTextNode(endContainer.substringData(endOffset, (endContainer.length - endOffset)));\r\n                            child = document.createTextNode(endContainer.substringData(0, endOffset));\r\n                            endOffset = child.length;\r\n                            endContainer = cloneNode = child.cloneNode(false);\r\n                        }\r\n                        else {\r\n                            cloneNode = child.cloneNode(false);\r\n                        }\r\n\r\n                        removeNodeList.push(child);\r\n                        node.appendChild(cloneNode);\r\n                        if (child.nodeType === 1) coverNode = cloneNode;\r\n                    }\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(el, newInnerNode);\r\n\r\n            el.innerHTML = '';\r\n            el.appendChild(prevNode);\r\n            el.appendChild(newInnerNode);\r\n            el.appendChild(afterNode);\r\n\r\n            return {\r\n                startContainer: startContainer,\r\n                startOffset: startOffset,\r\n                endContainer: endContainer,\r\n                endOffset: 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         * @private\r\n         */\r\n        _wrapLineNodes: function (element, newInnerNode, validation) {\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) coverNode = cloneNode;\r\n                    }\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, newInnerNode);\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         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _wrapLineNodesStart: function (element, newInnerNode, validation, startCon, startOff) {\r\n            const el = element;\r\n            const pNode = document.createElement('P');\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) {\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                            appendNode = newNode = pCurrent.pop();\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                            }\r\n                            newInnerNode.appendChild(appendNode);\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 = document.createTextNode(container.substringData(0, offset));\r\n                        const textNode = document.createTextNode(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 (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                        appendNode = newNode = pCurrent.pop() || node;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                        }\r\n\r\n                        if (appendNode !== node) {\r\n                            newInnerNode.appendChild(appendNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\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) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            element.parentNode.insertBefore(pNode, element);\r\n            util.removeItem(element);\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         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _wrapLineNodesEnd: function (element, newInnerNode, validation, endCon, endOff) {\r\n            const el = element;\r\n            const pNode = document.createElement('P');\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) {\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                            appendNode = newNode = pCurrent.pop();\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.insertBefore(newNode, appendNode.firstChild);\r\n                            }\r\n                            newInnerNode.insertBefore(appendNode, 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 = document.createTextNode(container.substringData(offset, (container.length - offset)));\r\n                        const textNode = document.createTextNode(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                        appendNode = newNode = pCurrent.pop() || node;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.insertBefore(newNode, appendNode.firstChild);\r\n                        }\r\n\r\n                        if (appendNode !== node) {\r\n                            newInnerNode.insertBefore(appendNode, newInnerNode.firstChild);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\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) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            element.parentNode.insertBefore(pNode, element);\r\n            util.removeItem(element);\r\n\r\n            return {\r\n                container: container,\r\n                offset: offset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description This function implements indentation.\r\n         * Set \"margin-left\" to \"25px\" in the top \"P\" tag of the parameter node.\r\n         * @param element {Element} - The element to indent (editor.getSelectionNode())\r\n         * @param command {String} - Separator (\"indent\" or \"outdent\")\r\n         */\r\n        indent: function (element, command) {\r\n            const p = util.getParentElement(element, 'P');\r\n            if (!p) return;\r\n\r\n            let 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         * @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            if (!this._variable.wysiwygActive) {\r\n                const ec = {'&amp;': '&', '&nbsp;': '\\u00A0', \"&quot;\": \"'\", '&lt;': '<', '&gt;': '>'};\r\n                const code_html = context.element.code.value.replace(/&[a-z]+;/g, function (m) {\r\n                    return (typeof ec[m] === 'string') ? ec[m] : m;\r\n                });\r\n                context.element.wysiwyg.innerHTML = code_html.trim().length > 0 ? code_html : '<p>&#65279</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                this._variable.wysiwygActive = true;\r\n            }\r\n            else {\r\n                context.element.code.value = context.element.wysiwyg.innerHTML.trim().replace(/<\\/p>(?=[^\\n])/gi, '<\\/p>\\n');\r\n                context.element.wysiwyg.style.display = 'none';\r\n                context.element.code.style.display = 'block';\r\n                this._variable.wysiwygActive = false;\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                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.innerHeight_fullScreen = (window.innerHeight - context.tool.bar.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                context.element.topArea.style.cssText = this._variable.originCssText;\r\n                context.element.editorArea.style.height = this._variable.editorHeight + 'px';\r\n\r\n                util.removeClass(element.firstElementChild, 'icon-reduction');\r\n                util.addClass(element.firstElementChild, 'icon-expansion');\r\n            }\r\n\r\n            this._variable.isFullScreen = !this._variable.isFullScreen;\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 = window.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 class=\"sun-editor-editable\">' + context.element.wysiwyg.innerHTML + '</body>' +\r\n                '</body>' + (isPrint ? '<script>window.print();</script>' : '') + '</html>');\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._variable.selectionNode; !util.isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\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 resizebar */\r\n            if (context.user.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\r\n        },\r\n\r\n        resize_window: function () {\r\n            if (editor._variable.isFullScreen) {\r\n                editor._variable.innerHeight_fullScreen += (window.innerHeight - context.tool.bar.offsetHeight) - editor._variable.innerHeight_fullScreen;\r\n                context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\r\n            }\r\n        },\r\n\r\n        onMouseDown_toolbar: function (e) {\r\n            e.preventDefault();\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            \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.openDialog.call(editor, command, target.getAttribute('data-option'), 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\r\n                switch (command) {\r\n                    case 'codeView':\r\n                        editor.toggleCodeView();\r\n                        util.toggleClass(target, 'on');\r\n                        break;\r\n                    case 'fullScreen':\r\n                        editor.toggleFullScreen(target);\r\n                        util.toggleClass(target, 'on');\r\n                        break;\r\n                    case 'indent':\r\n                    case 'outdent':\r\n                        editor.indent(editor._variable.selectionNode, command);\r\n                        break;\r\n                    case 'redo':\r\n                    case 'undo':\r\n                    case 'removeFormat':\r\n                        editor.execCommand(command, false, null);\r\n                        break;\r\n                    case 'preview':\r\n                    case 'print':\r\n                        editor.openWindowContents(command);\r\n                        break;\r\n                    case 'showBlocks':\r\n                        editor.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                            editor.execCommand('superscript', false, null);\r\n                            util.removeClass(context.tool.superscript, 'on');\r\n                        }\r\n                        editor.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                            editor.execCommand('subscript', false, null);\r\n                            util.removeClass(context.tool.subscript, 'on');\r\n                        }\r\n                        editor.execCommand(command, false, null);\r\n                        util.toggleClass(target, 'on');\r\n                        break;\r\n                    default :\r\n                        editor.execCommand(command, false, target.getAttribute('data-value'));\r\n                        util.toggleClass(target, 'on');\r\n                }\r\n            }\r\n        },\r\n\r\n        onMouseUp_wysiwyg: function (e) {\r\n            e.stopPropagation();\r\n\r\n            const targetElement = e.target;\r\n            editor.submenuOff();\r\n\r\n            if (/^HTML$/i.test(targetElement.nodeName)) {\r\n                e.preventDefault();\r\n                editor.focus();\r\n                return;\r\n            }\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.dialog.call_controller_resize.call(editor, targetElement, 'image');\r\n                    editor.plugins.image.onModifyMode.call(editor, targetElement, size);\r\n                });\r\n                return;\r\n            }\r\n\r\n            editor._setEditorRange();\r\n            event._findButtonEffectTag();\r\n        },\r\n\r\n        onKeyDown_wysiwyg: function (e) {\r\n            // editor._setSelectionNode();\r\n\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            function shortcutCommand(keyCode) {\r\n                const key = event._shortcutKeyCode[keyCode];\r\n                if (!key) return false;\r\n\r\n                editor.execCommand(key[0], false, null);\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            switch (keyCode) {\r\n                case 8: /**backspace key*/\r\n                    if (util.isFormatElement(editor._variable.selectionNode) && editor._variable.selectionNode.previousSibling === null) {\r\n                        e.preventDefault();\r\n                        e.stopPropagation();\r\n                        editor._variable.selectionNode.innerHTML = '&#65279';\r\n                        return false;\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                    let currentNode = editor._variable.selectionNode || editor.getSelectionNode();\r\n                    while (!/^TD$/i.test(currentNode.tagName) && !util.isWysiwygDiv(currentNode)) {\r\n                        currentNode = currentNode.parentNode;\r\n                    }\r\n\r\n                    if (currentNode && /^TD$/i.test(currentNode.tagName)) {\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                    /** if P Tag */\r\n                    if (shift) break;\r\n\r\n                    const tabText = document.createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\r\n                    editor.insertNode(tabText);\r\n                    editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize)\r\n\r\n                    break;\r\n            }\r\n        },\r\n\r\n        onKeyUp_wysiwyg: function (e) {\r\n            editor._setEditorRange();\r\n\r\n            /** when format tag deleted */\r\n            if (e.keyCode === 8 && util.isWysiwygDiv(editor._variable.selectionNode)) {\r\n                e.preventDefault();\r\n                e.stopPropagation();\r\n\r\n                const oFormatTag = document.createElement(editor._variable.currentNodes[0]);\r\n                oFormatTag.innerHTML = '&#65279';\r\n\r\n                editor._variable.selectionNode.appendChild(oFormatTag);\r\n                editor._variable.selectionNode = oFormatTag;\r\n                editor.setRange(oFormatTag, 0, oFormatTag, 0);\r\n            }\r\n\r\n            if (event._directionKeyKeyCode.test(e.keyCode)) {\r\n                event._findButtonEffectTag();\r\n            }\r\n        },\r\n\r\n        onScroll_wysiwyg: function () {\r\n            editor.controllersOff();\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) return true;\r\n\r\n            e.stopPropagation();\r\n            e.preventDefault();\r\n            \r\n            if (editor.plugins.image) {\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\r\n        onMouseDown_resizeBar: 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                document.removeEventListener('mousemove', event.resize_editor);\r\n                document.removeEventListener('mouseup', closureFunc);\r\n            }\r\n\r\n            document.addEventListener('mousemove', event.resize_editor);\r\n            document.addEventListener('mouseup', closureFunc);\r\n        },\r\n\r\n        resize_editor: function (e) {\r\n            const resizeInterval = (e.clientY - editor._variable.resizeClientY);\r\n\r\n            context.element.editorArea.style.height = (context.element.editorArea.offsetHeight + resizeInterval) + 'px';\r\n            editor._variable.editorHeight = (context.element.editorArea.offsetHeight + resizeInterval);\r\n\r\n            editor._variable.resizeClientY = e.clientY;\r\n        }\r\n    };\r\n\r\n    /** add event listeners */\r\n    /** tool bar event */\r\n    context.tool.bar.addEventListener('click', event.onClick_toolbar, false);\r\n    context.tool.bar.addEventListener('mousedown', event.onMouseDown_toolbar, false);\r\n    /** editor area */\r\n    context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('mouseup', event.onMouseUp_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    /** resize bar */\r\n    context.element.resizebar.addEventListener('mousedown', event.onMouseDown_resizeBar, false);\r\n    /** window resize event */\r\n    window.addEventListener('resize', event.resize_window, false);\r\n\r\n    /** add plugin to plugins object */\r\n    if (plugins) {\r\n        let pluginsValues = Object.values(plugins);\r\n        for (let i = 0, len = pluginsValues.length, plugin; i < len; i++) {\r\n            plugin = pluginsValues[i];\r\n            editor.plugins[plugin.name] = util.copyObj(plugin);\r\n        }\r\n         pluginsValues = null;\r\n    }\r\n\r\n    /** User function */\r\n    return {\r\n        /**\r\n         * @description Copying the contents of the editor to the original textarea\r\n         */\r\n        save: function () {\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.originElement.value = context.element.wysiwyg.innerHTML;\r\n            } else {\r\n                context.element.originElement.value = context.element.code.value;\r\n            }\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        getContent: function () {\r\n            let content = '';\r\n\r\n            if (context.element.wysiwyg.innerText.trim().length === 0) return content;\r\n\r\n            if (editor._variable.wysiwygActive) {\r\n                content = context.element.wysiwyg.innerHTML;\r\n            } else {\r\n                content = context.element.code.value;\r\n            }\r\n            return content;\r\n        },\r\n\r\n        /**\r\n         * @description Change the contents of the suneditor\r\n         * @param {String} content - Content to Input\r\n         */\r\n        setContent: function (content) {\r\n            const innerHTML = util.convertContentForEditor(content);\r\n\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.wysiwyg.innerHTML = innerHTML;\r\n            } else {\r\n                context.element.code.value = innerHTML;\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Add content to the suneditor\r\n         * @param {String} content - to Input\r\n         */\r\n        appendContent: function (content) {\r\n            if (editor._variable.wysiwygActive) {\r\n                const oP = document.createElement('P');\r\n                oP.innerHTML = content;\r\n                context.element.wysiwyg.appendChild(oP);\r\n            } else {\r\n                context.element.code.value += oP.outerHTML;\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        },\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        },\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            topAreaStyle.cssText = editor._variable.originCssText;\r\n            if (topAreaStyle.display === 'none') topAreaStyle.display = 'block';\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            window.removeEventListener('resize', event.resize_window);\r\n            \r\n            /** remove element */\r\n            context.element.topArea.parentNode.removeChild(context.element.topArea);\r\n\r\n            this.save = null;\r\n            this.getContext = null;\r\n            this.getContent = null;\r\n            this.setContent = null;\r\n            this.appendContent = 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    };\r\n};\r\n\r\nexport default core;"],"mappings":"AAAA;AAAA;AAEA;;;;;;;;;;AASA;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;;;;;;;;;;;;AAYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AACA;AAWA;;;;;;;;;;;;;;AAcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAVA;AACA;AAYA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;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;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;AAOA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AAEA;AACA;AAEA;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;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;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;;;;;;;AAOA;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;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AAWA;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;AA5CA;AA8CA;AACA;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;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AAYA;AACA;AACA;AAEA;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;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAJA;AAMA;AACA;AACA;;;;;;;AAOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;AAUA;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;AAAA;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;AAAA;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;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;;;;;AAUA;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;AAAA;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;AAAA;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;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;AAMA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AAUA;AApgCA;AAugCA;;;;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;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;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;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;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;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;AACA;AACA;AA5CA;AA8CA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AAEA;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;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AADA;AAEA;AACA;AACA;AAEA;AA1CA;AA4CA;AAEA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AACA;AAxYA;AA2YA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AAAA;AAEA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AADA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AAEA;AACA;AAAA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AArHA;AAuHA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/lib/core.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n\n/**\r\n * @description SunEditor core closure\r\n * @param context\r\n * @param util\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, util, plugins, lang) {\n /**\r\n * @description Practical editor function\r\n * This function is 'this' used by other plugins\r\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 util function\r\n */\n util: util,\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 controllers array (image resize area, link modified button)\r\n */\n controllerArray: [],\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 {Element} originCssText - Remembered the CSS of the editor before full screen (Used when returning to original size again)\r\n * @property {Number} editorHeight - The height value entered by the user or the height value of the \"textarea\" when the suneditor is created\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 originCssText: context.element.topArea.style.cssText,\n editorHeight: context.user.height,\n currentNodes: []\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.callModule.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 {Object} moduleObj - module object (dialog)\r\n */\n _addModule: function (moduleObj) {\n const moduleName = moduleObj.name;\n\n if (!this.plugins[moduleName]) {\n this.plugins[moduleName] = this.util.copyObj(moduleObj);\n this.plugins[moduleName].add(this);\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 this.submenu = element.nextElementSibling;\n this.submenu.style.display = 'block';\n util.addClass(element, 'on');\n this.submenuActiveButton = element;\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.removeClass(this.submenuActiveButton, 'on');\n this.submenuActiveButton = null;\n }\n\n if (context.image && context.image._onCaption === true) {\n this.plugins.image.toggle_caption_contenteditable.call(editor, false);\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\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\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 document.execCommand(command, showDefaultUI, 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.getParentElement(this._variable.selectionNode, '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 = window.getSelection();\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._variable.selectionNode = range.commonAncestorContainer;\n } else {\n this._variable.selectionNode = 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 = document.createRange();\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 = document.createRange();\n range.setStart(startCon, startOff);\n range.setEnd(endCon, endOff);\n const selection = window.getSelection();\n\n if (selection.rangeCount > 0) {\n selection.removeAllRanges();\n }\n\n this._variable.range = range;\n selection.addRange(range);\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 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 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 P tag to current line next\r\n * @param {Element} element - Insert as siblings of that element\r\n * @returns {Element}\r\n */\n appendP: function (element) {\n const oP = document.createElement('P');\n oP.innerHTML = '';\n element = util.getFormatElement(element);\n element.parentNode.insertBefore(oP, element.nextElementSibling);\n return oP;\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 /** Select within the same node */\n\n\n if (range.collapsed) {\n if (commonCon.nodeType === 3) {\n rightNode = commonCon.splitText(endOff);\n } else {\n if (parentNode.lastChild !== null && /^BR$/i.test(parentNode.lastChild.nodeName)) {\n parentNode.removeChild(parentNode.lastChild);\n }\n\n rightNode = null;\n }\n }\n /** Select multiple nodes */\n else {\n const isSameContainer = startCon === endCon;\n\n if (isSameContainer) {\n let removeNode = startCon;\n if (!this.isEdgePoint(endCon, endOff)) rightNode = endCon.splitText(endOff);\n if (!this.isEdgePoint(startCon, startOff)) removeNode = startCon.splitText(startOff);\n parentNode.removeChild(removeNode);\n } else {\n this.removeNode();\n rightNode = endCon;\n\n while (rightNode.nodeType !== 1) {\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 } // this.setRange(oNode, 0, oNode, 0);\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.getListChildNodes(commonCon);\n let startIndex = util.getArrayIndex(childNodes, startCon);\n let endIndex = util.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.removeItem(item);\n continue;\n }\n\n if (item === startCon) {\n if (startCon.nodeType === 1) {\n beforeNode = document.createTextNode(startCon.textContent);\n } else {\n beforeNode = document.createTextNode(startCon.substringData(0, startOff));\n }\n\n if (beforeNode.length > 0) {\n startCon.data = beforeNode.data;\n } else {\n util.removeItem(startCon);\n }\n\n continue;\n }\n\n if (item === endCon) {\n if (endCon.nodeType === 1) {\n afterNode = document.createTextNode(endCon.textContent);\n } else {\n afterNode = document.createTextNode(endCon.substringData(endOff, endCon.length - endOff));\n }\n\n if (afterNode.length > 0) {\n endCon.data = afterNode.data;\n } else {\n util.removeItem(endCon);\n }\n\n continue;\n }\n\n util.removeItem(item);\n }\n },\n\n /**\r\n * @description Copies the node of the argument value and wraps all selected text.\r\n * 1. When there is the same 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 same node, removed only Css attribute values received as arguments\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']...])\r\n */\n wrapRangeToTag: function (appendNode, checkCSSPropertyArray) {\n const range = this.getRange();\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 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, 'gi');\n }\n /** one node */\n\n\n if (startCon === endCon) {\n newNode = appendNode.cloneNode(false);\n /** No node selected */\n\n if (startOff === endOff) {\n newNode.innerHTML = '';\n\n if (util.isFormatElement(startCon)) {\n startCon.appendChild(newNode);\n } else {\n startCon.splitText(startOff);\n startCon.parentNode.insertBefore(newNode, startCon.nextSibling);\n }\n }\n /** Select within the same node */\n else {\n const isElement = startCon.nodeType === 1;\n\n if (isElement) {\n newNode.innerHTML = startCon.outerHTML;\n startCon.parentNode.appendChild(newNode);\n util.removeItem(startCon);\n } else {\n const beforeNode = document.createTextNode(startCon.substringData(0, startOff));\n const afterNode = document.createTextNode(startCon.substringData(endOff, startCon.length - endOff));\n newNode.innerText = startCon.substringData(startOff, endOff - startOff);\n startCon.parentNode.insertBefore(newNode, startCon.nextSibling);\n\n if (beforeNode.data.length > 0) {\n startCon.data = beforeNode.data;\n } else {\n startCon.data = startCon.substringData(0, startOff);\n }\n\n if (afterNode.data.length > 0) {\n startCon.parentNode.insertBefore(afterNode, newNode.nextSibling);\n }\n }\n }\n\n start.container = newNode;\n start.offset = 1;\n end.container = newNode;\n end.offset = 1;\n }\n /** multiple nodes */\n else {\n /** tag check function*/\n const checkCss = function (vNode) {\n if (vNode.nodeType === 3) 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 (vNode.nodeName !== appendNode.nodeName || style.length > 0) {\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.hasClass(commonCon, 'sun-editor-id-wysiwyg')) {\n newNode = appendNode.cloneNode(false);\n\n const newRange = this._wrapLineNodesPart(commonCon, newNode, checkCss, startCon, startOff, endCon, endOff);\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 = util.getListChildren(commonCon, function (current) {\n return util.isFormatElement(current);\n });\n let startLine = util.getParentElement(startCon, 'P');\n let endLine = util.getParentElement(endCon, 'P');\n\n for (let i = 0, len = lineNodes.length; i < len; i++) {\n if (startLine === lineNodes[i]) {\n startLine = i;\n continue;\n }\n\n if (endLine === lineNodes[i]) {\n endLine = i;\n break;\n }\n } // startCon\n\n\n newNode = appendNode.cloneNode(false);\n start = this._wrapLineNodesStart(lineNodes[startLine], newNode, checkCss, startCon, startOff); // mid\n\n for (let i = startLine + 1; i < endLine; i++) {\n newNode = appendNode.cloneNode(false);\n\n this._wrapLineNodes(lineNodes[i], newNode, checkCss);\n } // endCon\n\n\n newNode = appendNode.cloneNode(false);\n end = this._wrapLineNodesEnd(lineNodes[endLine], newNode, checkCss, endCon, endOff);\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 * @returns {{startContainer: *, startOffset: *, endContainer: *, endOffset: *}}\r\n * @private\r\n */\n _wrapLineNodesPart: function (element, newInnerNode, validation, startCon, startOff, endCon, endOff) {\n const el = element;\n const removeNodeList = [];\n let startContainer = startCon;\n let startOffset = startOff;\n let endContainer = endCon;\n let endOffset = endOff;\n let prevNode, afterNode;\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\n if (validation(child)) {\n let cloneNode;\n\n if (child === startContainer) {\n prevNode = document.createTextNode(startContainer.substringData(0, startOffset));\n child = document.createTextNode(startContainer.substringData(startOffset, startContainer.length - startOffset));\n startOffset = 0;\n startContainer = cloneNode = child.cloneNode(false);\n } else if (child === endContainer) {\n afterNode = document.createTextNode(endContainer.substringData(endOffset, endContainer.length - endOffset));\n child = document.createTextNode(endContainer.substringData(0, endOffset));\n endOffset = child.length;\n endContainer = cloneNode = child.cloneNode(false);\n } else {\n cloneNode = child.cloneNode(false);\n }\n\n removeNodeList.push(child);\n node.appendChild(cloneNode);\n if (child.nodeType === 1) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(el, newInnerNode);\n\n el.innerHTML = '';\n el.appendChild(prevNode);\n el.appendChild(newInnerNode);\n el.appendChild(afterNode);\n return {\n startContainer: startContainer,\n startOffset: startOffset,\n endContainer: endContainer,\n endOffset: 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 * @private\r\n */\n _wrapLineNodes: function (element, newInnerNode, validation) {\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) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(element, newInnerNode);\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 * @returns {{container: *, offset: *}}\r\n * @private\r\n */\n _wrapLineNodesStart: function (element, newInnerNode, validation, startCon, startOff) {\n const el = element;\n const pNode = document.createElement('P');\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) {\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 appendNode = newNode = pCurrent.pop();\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n }\n\n newInnerNode.appendChild(appendNode);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n } // startContainer\n\n\n if (!passNode && child === container) {\n const prevNode = document.createTextNode(container.substringData(0, offset));\n const textNode = document.createTextNode(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 (validation(newNode) && newNode.nodeType === 1) {\n pCurrent.push(newNode.cloneNode(false));\n }\n\n newNode = newNode.parentNode;\n }\n\n appendNode = newNode = pCurrent.pop() || node;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.appendChild(newNode);\n }\n\n if (appendNode !== node) {\n newInnerNode.appendChild(appendNode);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n\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) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(element, pNode);\n\n element.parentNode.insertBefore(pNode, element);\n util.removeItem(element);\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 * @returns {{container: *, offset: *}}\r\n * @private\r\n */\n _wrapLineNodesEnd: function (element, newInnerNode, validation, endCon, endOff) {\n const el = element;\n const pNode = document.createElement('P');\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) {\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 appendNode = newNode = pCurrent.pop();\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.insertBefore(newNode, appendNode.firstChild);\n }\n\n newInnerNode.insertBefore(appendNode, newInnerNode.firstChild);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n } // endContainer\n\n\n if (!passNode && child === container) {\n const afterNode = document.createTextNode(container.substringData(offset, container.length - offset));\n const textNode = document.createTextNode(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 appendNode = newNode = pCurrent.pop() || node;\n\n while (pCurrent.length > 0) {\n newNode = pCurrent.pop();\n appendNode.insertBefore(newNode, appendNode.firstChild);\n }\n\n if (appendNode !== node) {\n newInnerNode.insertBefore(appendNode, newInnerNode.firstChild);\n node = newNode;\n } else {\n node = newInnerNode;\n }\n\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) coverNode = cloneNode;\n }\n\n recursionFunc(child, coverNode);\n }\n })(element, pNode);\n\n element.parentNode.insertBefore(pNode, element);\n util.removeItem(element);\n return {\n container: container,\n offset: offset\n };\n },\n\n /**\r\n * @description This function implements indentation.\r\n * Set \"margin-left\" to \"25px\" in the top \"P\" tag of the parameter node.\r\n * @param element {Element} - The element to indent (editor.getSelectionNode())\r\n * @param command {String} - Separator (\"indent\" or \"outdent\")\r\n */\n indent: function (element, command) {\n const p = util.getParentElement(element, 'P');\n if (!p) return;\n let 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 /**\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.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 if (!this._variable.wysiwygActive) {\n const ec = {\n '&': '&',\n ' ': '\\u00A0',\n \""\": \"'\",\n '<': '<',\n '>': '>'\n };\n const code_html = context.element.code.value.replace(/&[a-z]+;/g, function (m) {\n return typeof ec[m] === 'string' ? ec[m] : m;\n });\n context.element.wysiwyg.innerHTML = code_html.trim().length > 0 ? code_html : '



';\n context.element.wysiwyg.scrollTop = 0;\n context.element.code.style.display = 'none';\n context.element.wysiwyg.style.display = 'block';\n this._variable.wysiwygActive = true;\n } else {\n context.element.code.value = context.element.wysiwyg.innerHTML.trim().replace(/<\\/p>(?=[^\\n])/gi, '<\\/p>\\n');\n context.element.wysiwyg.style.display = 'none';\n context.element.code.style.display = 'block';\n this._variable.wysiwygActive = false;\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 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.innerHeight_fullScreen = window.innerHeight - context.tool.bar.offsetHeight;\n context.element.editorArea.style.height = this._variable.innerHeight_fullScreen + 'px';\n util.removeClass(element.firstElementChild, 'icon-expansion');\n util.addClass(element.firstElementChild, 'icon-reduction');\n } else {\n context.element.topArea.style.cssText = this._variable.originCssText;\n context.element.editorArea.style.height = this._variable.editorHeight + 'px';\n util.removeClass(element.firstElementChild, 'icon-reduction');\n util.addClass(element.firstElementChild, 'icon-expansion');\n }\n\n this._variable.isFullScreen = !this._variable.isFullScreen;\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 const windowObject = window.open('', '_blank');\n windowObject.mimeType = 'text/html';\n windowObject.document.write('' + '' + '' + '' + '' + '' + (isPrint ? lang.toolbar.print : lang.toolbar.preview) + '' + '' + '' + '' + context.element.wysiwyg.innerHTML + '' + '' + (isPrint ? '' : '') + '');\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._variable.selectionNode; !util.isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\n if (selectionParent.nodeType !== 1) continue;\n nodeName = selectionParent.nodeName.toUpperCase();\n currentNodes.push(nodeName);\n /** Format */\n\n if (findFormat && util.isFormatElement(selectionParent)) {\n commandMapNodes.push('FORMAT');\n util.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.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.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.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.changeTxt(commandMap[key], lang.toolbar.font);\n } else if (/^SIZE$/i.test(key)) {\n util.changeTxt(commandMap[key], lang.toolbar.fontSize);\n } else {\n util.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 resizebar */\n\n if (context.user.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\n },\n resize_window: function () {\n if (editor._variable.isFullScreen) {\n editor._variable.innerHeight_fullScreen += window.innerHeight - context.tool.bar.offsetHeight - editor._variable.innerHeight_fullScreen;\n context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\n }\n },\n onMouseDown_toolbar: function (e) {\n e.preventDefault();\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 /** 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.openDialog.call(editor, command, target.getAttribute('data-option'), 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\n switch (command) {\n case 'codeView':\n editor.toggleCodeView();\n util.toggleClass(target, 'on');\n break;\n\n case 'fullScreen':\n editor.toggleFullScreen(target);\n util.toggleClass(target, 'on');\n break;\n\n case 'indent':\n case 'outdent':\n editor.indent(editor._variable.selectionNode, command);\n break;\n\n case 'redo':\n case 'undo':\n case 'removeFormat':\n editor.execCommand(command, false, null);\n break;\n\n case 'preview':\n case 'print':\n editor.openWindowContents(command);\n break;\n\n case 'showBlocks':\n editor.toggleDisplayBlocks();\n util.toggleClass(target, 'on');\n break;\n\n case 'subscript':\n if (util.hasClass(context.tool.superscript, 'on')) {\n editor.execCommand('superscript', false, null);\n util.removeClass(context.tool.superscript, 'on');\n }\n\n editor.execCommand(command, false, null);\n util.toggleClass(target, 'on');\n break;\n\n case 'superscript':\n if (util.hasClass(context.tool.subscript, 'on')) {\n editor.execCommand('subscript', false, null);\n util.removeClass(context.tool.subscript, 'on');\n }\n\n editor.execCommand(command, false, null);\n util.toggleClass(target, 'on');\n break;\n\n default:\n editor.execCommand(command, false, target.getAttribute('data-value'));\n util.toggleClass(target, 'on');\n }\n }\n },\n onMouseUp_wysiwyg: function (e) {\n e.stopPropagation();\n const targetElement = e.target;\n editor.submenuOff();\n\n if (/^HTML$/i.test(targetElement.nodeName)) {\n e.preventDefault();\n editor.focus();\n return;\n }\n\n if (/^IMG$/i.test(targetElement.nodeName)) {\n e.preventDefault();\n editor.callPlugin('image', function () {\n const size = editor.plugins.dialog.call_controller_resize.call(editor, targetElement, 'image');\n editor.plugins.image.onModifyMode.call(editor, targetElement, size);\n });\n return;\n }\n\n editor._setEditorRange();\n\n event._findButtonEffectTag();\n },\n onKeyDown_wysiwyg: function (e) {\n // editor._setSelectionNode();\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 function shortcutCommand(keyCode) {\n const key = event._shortcutKeyCode[keyCode];\n if (!key) return false;\n editor.execCommand(key[0], false, null);\n util.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 switch (keyCode) {\n case 8:\n /**backspace key*/\n if (util.isFormatElement(editor._variable.selectionNode) && editor._variable.selectionNode.previousSibling === null) {\n e.preventDefault();\n e.stopPropagation();\n editor._variable.selectionNode.innerHTML = '';\n return false;\n }\n\n break;\n\n case 9:\n /**tab key*/\n e.preventDefault();\n if (ctrl || alt) break;\n let currentNode = editor._variable.selectionNode || editor.getSelectionNode();\n\n while (!/^TD$/i.test(currentNode.tagName) && !util.isWysiwygDiv(currentNode)) {\n currentNode = currentNode.parentNode;\n }\n\n if (currentNode && /^TD$/i.test(currentNode.tagName)) {\n const table = util.getParentElement(currentNode, 'table');\n const cells = util.getListChildren(table, util.isCell);\n let idx = shift ? util.prevIdx(cells, currentNode) : util.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 /** if P Tag */\n\n\n if (shift) break;\n const tabText = document.createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\n editor.insertNode(tabText);\n editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize);\n break;\n }\n },\n onKeyUp_wysiwyg: function (e) {\n editor._setEditorRange();\n /** when format tag deleted */\n\n\n if (e.keyCode === 8 && util.isWysiwygDiv(editor._variable.selectionNode)) {\n e.preventDefault();\n e.stopPropagation();\n const oFormatTag = document.createElement(editor._variable.currentNodes[0]);\n oFormatTag.innerHTML = '';\n\n editor._variable.selectionNode.appendChild(oFormatTag);\n\n editor._variable.selectionNode = oFormatTag;\n editor.setRange(oFormatTag, 0, oFormatTag, 0);\n }\n\n if (event._directionKeyKeyCode.test(e.keyCode)) {\n event._findButtonEffectTag();\n }\n },\n onScroll_wysiwyg: function () {\n editor.controllersOff();\n },\n onDrop_wysiwyg: function (e) {\n const files = e.dataTransfer.files;\n if (files.length === 0) return true;\n e.stopPropagation();\n e.preventDefault();\n\n if (editor.plugins.image) {\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 onMouseDown_resizeBar: 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 document.removeEventListener('mousemove', event.resize_editor);\n document.removeEventListener('mouseup', closureFunc);\n }\n\n document.addEventListener('mousemove', event.resize_editor);\n document.addEventListener('mouseup', closureFunc);\n },\n resize_editor: function (e) {\n const resizeInterval = e.clientY - editor._variable.resizeClientY;\n context.element.editorArea.style.height = context.element.editorArea.offsetHeight + resizeInterval + 'px';\n editor._variable.editorHeight = context.element.editorArea.offsetHeight + resizeInterval;\n editor._variable.resizeClientY = e.clientY;\n }\n };\n /** add event listeners */\n\n /** tool bar event */\n\n context.tool.bar.addEventListener('click', event.onClick_toolbar, false);\n context.tool.bar.addEventListener('mousedown', event.onMouseDown_toolbar, false);\n /** editor area */\n\n context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\n context.element.wysiwyg.addEventListener('mouseup', event.onMouseUp_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 /** resize bar */\n\n context.element.resizebar.addEventListener('mousedown', event.onMouseDown_resizeBar, false);\n /** window resize event */\n\n window.addEventListener('resize', event.resize_window, false);\n /** add plugin to plugins object */\n\n if (plugins) {\n let pluginsValues = Object.values(plugins);\n\n for (let i = 0, len = pluginsValues.length, plugin; i < len; i++) {\n plugin = pluginsValues[i];\n editor.plugins[plugin.name] = util.copyObj(plugin);\n }\n\n pluginsValues = null;\n }\n /** User function */\n\n\n return {\n /**\r\n * @description Copying the contents of the editor to the original textarea\r\n */\n save: function () {\n if (editor._variable.wysiwygActive) {\n context.element.originElement.value = context.element.wysiwyg.innerHTML;\n } else {\n context.element.originElement.value = context.element.code.value;\n }\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 getContent: function () {\n let content = '';\n if (context.element.wysiwyg.innerText.trim().length === 0) return content;\n\n if (editor._variable.wysiwygActive) {\n content = context.element.wysiwyg.innerHTML;\n } else {\n content = context.element.code.value;\n }\n\n return content;\n },\n\n /**\r\n * @description Change the contents of the suneditor\r\n * @param {String} content - Content to Input\r\n */\n setContent: function (content) {\n const innerHTML = util.convertContentForEditor(content);\n\n if (editor._variable.wysiwygActive) {\n context.element.wysiwyg.innerHTML = innerHTML;\n } else {\n context.element.code.value = innerHTML;\n }\n },\n\n /**\r\n * @description Add content to the suneditor\r\n * @param {String} content - to Input\r\n */\n appendContent: function (content) {\n if (editor._variable.wysiwygActive) {\n const oP = document.createElement('P');\n oP.innerHTML = content;\n context.element.wysiwyg.appendChild(oP);\n } else {\n context.element.code.value += oP.outerHTML;\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 },\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 },\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.user.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 window.removeEventListener('resize', event.resize_window);\n /** remove element */\n\n context.element.topArea.parentNode.removeChild(context.element.topArea);\n this.save = null;\n this.getContext = null;\n this.getContent = null;\n this.setContent = null;\n this.appendContent = null;\n this.disabled = null;\n this.enabled = null;\n this.show = null;\n this.hide = null;\n this.destroy = null;\n }\n };\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":["'use strict';\r\n\r\n/**\r\n * @description SunEditor core closure\r\n * @param context\r\n * @param util\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, util, plugins, lang) {\r\n    /**\r\n     * @description Practical editor function\r\n     * This function is 'this' used by other plugins\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 util function\r\n         */\r\n        util: util,\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 controllers array (image resize area, link modified button)\r\n         */\r\n        controllerArray: [],\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 {Element} originCssText - Remembered the CSS of the editor before full screen (Used when returning to original size again)\r\n         * @property {Number} editorHeight - The height value entered by the user or the height value of the \"textarea\" when the suneditor is created\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            originCssText: context.element.topArea.style.cssText,\r\n            editorHeight: context.user.height,\r\n            currentNodes: []\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.callModule.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 {Object} moduleObj - module object (dialog)\r\n         */\r\n        _addModule: function (moduleObj) {\r\n            const moduleName = moduleObj.name;\r\n            if (!this.plugins[moduleName]) {\r\n                this.plugins[moduleName] = this.util.copyObj(moduleObj);\r\n                this.plugins[moduleName].add(this);\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            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\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            if (context.image && context.image._onCaption === true) {\r\n                this.plugins.image.toggle_caption_contenteditable.call(editor, false);\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\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\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            document.execCommand(command, showDefaultUI, 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._variable.selectionNode, '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 = window.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._variable.selectionNode = range.commonAncestorContainer;\r\n            } else {\r\n                this._variable.selectionNode = 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 = document.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 = document.createRange();\r\n            range.setStart(startCon, startOff);\r\n            range.setEnd(endCon, endOff);\r\n\r\n            const selection = window.getSelection();\r\n\r\n            if (selection.rangeCount > 0) {\r\n                selection.removeAllRanges();\r\n            }\r\n\r\n            this._variable.range = range;\r\n            selection.addRange(range);\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 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 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 P tag to current line next\r\n         * @param {Element} element - Insert as siblings of that element\r\n         * @returns {Element}\r\n         */\r\n        appendP: function (element) {\r\n            const oP = document.createElement('P');\r\n            oP.innerHTML = '&#65279';\r\n\r\n            element = util.getFormatElement(element);\r\n            element.parentNode.insertBefore(oP, element.nextElementSibling);\r\n\r\n            return oP;\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                /** Select within the same 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 && /^BR$/i.test(parentNode.lastChild.nodeName)) {\r\n                            parentNode.removeChild(parentNode.lastChild);\r\n                        }\r\n                        rightNode = null;\r\n                    }\r\n                }\r\n                /** Select multiple nodes */\r\n                else {\r\n                    const isSameContainer = startCon === endCon;\r\n\r\n                    if (isSameContainer) {\r\n                        let removeNode = startCon;\r\n                        if (!this.isEdgePoint(endCon, endOff)) rightNode = endCon.splitText(endOff);\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                        rightNode = endCon;\r\n\r\n                        while (rightNode.nodeType !== 1) {\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            // this.setRange(oNode, 0, oNode, 0);\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 = document.createTextNode(startCon.textContent);\r\n                    } else {\r\n                        beforeNode = document.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 = document.createTextNode(endCon.textContent);\r\n                    } else {\r\n                        afterNode = document.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 Copies the node of the argument value and wraps all selected text.\r\n         * 1. When there is the same 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 same node, removed only Css attribute values received as arguments\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']...])\r\n         */\r\n        wrapRangeToTag: function (appendNode, checkCSSPropertyArray) {\r\n            const range = this.getRange();\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 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, 'gi');\r\n            }\r\n\r\n            /** one node */\r\n            if (startCon === endCon) {\r\n                newNode = appendNode.cloneNode(false);\r\n\r\n                /** No node selected */\r\n                if (startOff === endOff) {\r\n                    newNode.innerHTML = '&#65279';\r\n                    if (util.isFormatElement(startCon)) {\r\n                        startCon.appendChild(newNode);\r\n                    } else {\r\n                        startCon.splitText(startOff);\r\n                        startCon.parentNode.insertBefore(newNode, startCon.nextSibling);\r\n                    }\r\n                }\r\n                /** Select within the same node */\r\n                else {\r\n                    const isElement = startCon.nodeType === 1;\r\n                    if (isElement) {\r\n                        newNode.innerHTML = startCon.outerHTML;\r\n                        startCon.parentNode.appendChild(newNode);\r\n                        util.removeItem(startCon);\r\n                    } else {\r\n                        const beforeNode = document.createTextNode(startCon.substringData(0, startOff));\r\n                        const afterNode = document.createTextNode(startCon.substringData(endOff, (startCon.length - endOff)));\r\n\r\n                        newNode.innerText = startCon.substringData(startOff, (endOff - startOff));\r\n                        startCon.parentNode.insertBefore(newNode, startCon.nextSibling);\r\n\r\n                        if (beforeNode.data.length > 0) {\r\n                            startCon.data = beforeNode.data;\r\n                        } else {\r\n                            startCon.data = startCon.substringData(0, startOff);\r\n                        }\r\n\r\n                        if (afterNode.data.length > 0) {\r\n                            startCon.parentNode.insertBefore(afterNode, newNode.nextSibling);\r\n                        }\r\n                    }\r\n                }\r\n\r\n                start.container = newNode;\r\n                start.offset = 1;\r\n                end.container = newNode;\r\n                end.offset = 1;\r\n            }\r\n            /** multiple nodes */\r\n            else {\r\n                /** tag check function*/\r\n                const checkCss = function (vNode) {\r\n                    if (vNode.nodeType === 3) 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 (vNode.nodeName !== appendNode.nodeName || style.length > 0) {\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.hasClass(commonCon, 'sun-editor-id-wysiwyg')) {\r\n                    newNode = appendNode.cloneNode(false);\r\n                    const newRange = this._wrapLineNodesPart(commonCon, newNode, checkCss, startCon, startOff, endCon, endOff);\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 = util.getListChildren(commonCon, function (current) {\r\n                        return util.isFormatElement(current);\r\n                    });\r\n\r\n                    let startLine = util.getParentElement(startCon, 'P');\r\n                    let endLine = util.getParentElement(endCon, 'P');\r\n\r\n                    for (let i = 0, len = lineNodes.length; i < len; i++) {\r\n                        if (startLine === lineNodes[i]) {\r\n                            startLine = i;\r\n                            continue;\r\n                        }\r\n                        if (endLine === lineNodes[i]) {\r\n                            endLine = i;\r\n                            break;\r\n                        }\r\n                    }\r\n\r\n                    // startCon\r\n                    newNode = appendNode.cloneNode(false);\r\n                    start = this._wrapLineNodesStart(lineNodes[startLine], newNode, checkCss, startCon, startOff);\r\n                    // mid\r\n                    for (let i = startLine + 1; i < endLine; i++) {\r\n                        newNode = appendNode.cloneNode(false);\r\n                        this._wrapLineNodes(lineNodes[i], newNode, checkCss);\r\n                    }\r\n                    // endCon\r\n                    newNode = appendNode.cloneNode(false);\r\n                    end = this._wrapLineNodesEnd(lineNodes[endLine], newNode, checkCss, endCon, endOff);\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         * @returns {{startContainer: *, startOffset: *, endContainer: *, endOffset: *}}\r\n         * @private\r\n         */\r\n        _wrapLineNodesPart: function (element, newInnerNode, validation, startCon, startOff, endCon, endOff) {\r\n            const el = element;\r\n            const removeNodeList = [];\r\n\r\n            let startContainer = startCon;\r\n            let startOffset = startOff;\r\n            let endContainer = endCon;\r\n            let endOffset = endOff;\r\n            let prevNode, afterNode;\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\r\n                    if (validation(child)) {\r\n                        let cloneNode;\r\n\r\n                        if (child === startContainer) {\r\n                            prevNode = document.createTextNode(startContainer.substringData(0, startOffset));\r\n                            child = document.createTextNode(startContainer.substringData(startOffset, (startContainer.length - startOffset)));\r\n                            startOffset = 0;\r\n                            startContainer = cloneNode = child.cloneNode(false);\r\n                        }\r\n                        else if (child === endContainer) {\r\n                            afterNode = document.createTextNode(endContainer.substringData(endOffset, (endContainer.length - endOffset)));\r\n                            child = document.createTextNode(endContainer.substringData(0, endOffset));\r\n                            endOffset = child.length;\r\n                            endContainer = cloneNode = child.cloneNode(false);\r\n                        }\r\n                        else {\r\n                            cloneNode = child.cloneNode(false);\r\n                        }\r\n\r\n                        removeNodeList.push(child);\r\n                        node.appendChild(cloneNode);\r\n                        if (child.nodeType === 1) coverNode = cloneNode;\r\n                    }\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(el, newInnerNode);\r\n\r\n            el.innerHTML = '';\r\n            el.appendChild(prevNode);\r\n            el.appendChild(newInnerNode);\r\n            el.appendChild(afterNode);\r\n\r\n            return {\r\n                startContainer: startContainer,\r\n                startOffset: startOffset,\r\n                endContainer: endContainer,\r\n                endOffset: 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         * @private\r\n         */\r\n        _wrapLineNodes: function (element, newInnerNode, validation) {\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) coverNode = cloneNode;\r\n                    }\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, newInnerNode);\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         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _wrapLineNodesStart: function (element, newInnerNode, validation, startCon, startOff) {\r\n            const el = element;\r\n            const pNode = document.createElement('P');\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) {\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                            appendNode = newNode = pCurrent.pop();\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.appendChild(newNode);\r\n                            }\r\n                            newInnerNode.appendChild(appendNode);\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 = document.createTextNode(container.substringData(0, offset));\r\n                        const textNode = document.createTextNode(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 (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                        appendNode = newNode = pCurrent.pop() || node;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.appendChild(newNode);\r\n                        }\r\n\r\n                        if (appendNode !== node) {\r\n                            newInnerNode.appendChild(appendNode);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\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) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            element.parentNode.insertBefore(pNode, element);\r\n            util.removeItem(element);\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         * @returns {{container: *, offset: *}}\r\n         * @private\r\n         */\r\n        _wrapLineNodesEnd: function (element, newInnerNode, validation, endCon, endOff) {\r\n            const el = element;\r\n            const pNode = document.createElement('P');\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) {\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                            appendNode = newNode = pCurrent.pop();\r\n                            while (pCurrent.length > 0) {\r\n                                newNode = pCurrent.pop();\r\n                                appendNode.insertBefore(newNode, appendNode.firstChild);\r\n                            }\r\n                            newInnerNode.insertBefore(appendNode, 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 = document.createTextNode(container.substringData(offset, (container.length - offset)));\r\n                        const textNode = document.createTextNode(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                        appendNode = newNode = pCurrent.pop() || node;\r\n                        while (pCurrent.length > 0) {\r\n                            newNode = pCurrent.pop();\r\n                            appendNode.insertBefore(newNode, appendNode.firstChild);\r\n                        }\r\n\r\n                        if (appendNode !== node) {\r\n                            newInnerNode.insertBefore(appendNode, newInnerNode.firstChild);\r\n                            node = newNode;\r\n                        } else {\r\n                            node = newInnerNode;\r\n                        }\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) coverNode = cloneNode;\r\n                    }\r\n\r\n                    recursionFunc(child, coverNode);\r\n                }\r\n            })(element, pNode);\r\n\r\n            element.parentNode.insertBefore(pNode, element);\r\n            util.removeItem(element);\r\n\r\n            return {\r\n                container: container,\r\n                offset: offset\r\n            };\r\n        },\r\n\r\n        /**\r\n         * @description This function implements indentation.\r\n         * Set \"margin-left\" to \"25px\" in the top \"P\" tag of the parameter node.\r\n         * @param element {Element} - The element to indent (editor.getSelectionNode())\r\n         * @param command {String} - Separator (\"indent\" or \"outdent\")\r\n         */\r\n        indent: function (element, command) {\r\n            const p = util.getParentElement(element, 'P');\r\n            if (!p) return;\r\n\r\n            let 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         * @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            if (!this._variable.wysiwygActive) {\r\n                const ec = {'&amp;': '&', '&nbsp;': '\\u00A0', \"&quot;\": \"'\", '&lt;': '<', '&gt;': '>'};\r\n                const code_html = context.element.code.value.replace(/&[a-z]+;/g, function (m) {\r\n                    return (typeof ec[m] === 'string') ? ec[m] : m;\r\n                });\r\n                context.element.wysiwyg.innerHTML = code_html.trim().length > 0 ? code_html : '<p>&#65279</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                this._variable.wysiwygActive = true;\r\n            }\r\n            else {\r\n                context.element.code.value = context.element.wysiwyg.innerHTML.trim().replace(/<\\/p>(?=[^\\n])/gi, '<\\/p>\\n');\r\n                context.element.wysiwyg.style.display = 'none';\r\n                context.element.code.style.display = 'block';\r\n                this._variable.wysiwygActive = false;\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                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.innerHeight_fullScreen = (window.innerHeight - context.tool.bar.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                context.element.topArea.style.cssText = this._variable.originCssText;\r\n                context.element.editorArea.style.height = this._variable.editorHeight + 'px';\r\n\r\n                util.removeClass(element.firstElementChild, 'icon-reduction');\r\n                util.addClass(element.firstElementChild, 'icon-expansion');\r\n            }\r\n\r\n            this._variable.isFullScreen = !this._variable.isFullScreen;\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 = window.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 class=\"sun-editor-editable\">' + context.element.wysiwyg.innerHTML + '</body>' +\r\n                '</body>' + (isPrint ? '<script>window.print();</script>' : '') + '</html>');\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._variable.selectionNode; !util.isWysiwygDiv(selectionParent); selectionParent = selectionParent.parentNode) {\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 resizebar */\r\n            if (context.user.showPathLabel) context.element.navigation.textContent = editor._variable.currentNodes.join(' > ');\r\n        },\r\n\r\n        resize_window: function () {\r\n            if (editor._variable.isFullScreen) {\r\n                editor._variable.innerHeight_fullScreen += (window.innerHeight - context.tool.bar.offsetHeight) - editor._variable.innerHeight_fullScreen;\r\n                context.element.editorArea.style.height = editor._variable.innerHeight_fullScreen + 'px';\r\n            }\r\n        },\r\n\r\n        onMouseDown_toolbar: function (e) {\r\n            e.preventDefault();\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            \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.openDialog.call(editor, command, target.getAttribute('data-option'), 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\r\n                switch (command) {\r\n                    case 'codeView':\r\n                        editor.toggleCodeView();\r\n                        util.toggleClass(target, 'on');\r\n                        break;\r\n                    case 'fullScreen':\r\n                        editor.toggleFullScreen(target);\r\n                        util.toggleClass(target, 'on');\r\n                        break;\r\n                    case 'indent':\r\n                    case 'outdent':\r\n                        editor.indent(editor._variable.selectionNode, command);\r\n                        break;\r\n                    case 'redo':\r\n                    case 'undo':\r\n                    case 'removeFormat':\r\n                        editor.execCommand(command, false, null);\r\n                        break;\r\n                    case 'preview':\r\n                    case 'print':\r\n                        editor.openWindowContents(command);\r\n                        break;\r\n                    case 'showBlocks':\r\n                        editor.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                            editor.execCommand('superscript', false, null);\r\n                            util.removeClass(context.tool.superscript, 'on');\r\n                        }\r\n                        editor.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                            editor.execCommand('subscript', false, null);\r\n                            util.removeClass(context.tool.subscript, 'on');\r\n                        }\r\n                        editor.execCommand(command, false, null);\r\n                        util.toggleClass(target, 'on');\r\n                        break;\r\n                    default :\r\n                        editor.execCommand(command, false, target.getAttribute('data-value'));\r\n                        util.toggleClass(target, 'on');\r\n                }\r\n            }\r\n        },\r\n\r\n        onMouseUp_wysiwyg: function (e) {\r\n            e.stopPropagation();\r\n\r\n            const targetElement = e.target;\r\n            editor.submenuOff();\r\n\r\n            if (/^HTML$/i.test(targetElement.nodeName)) {\r\n                e.preventDefault();\r\n                editor.focus();\r\n                return;\r\n            }\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.dialog.call_controller_resize.call(editor, targetElement, 'image');\r\n                    editor.plugins.image.onModifyMode.call(editor, targetElement, size);\r\n                });\r\n                return;\r\n            }\r\n\r\n            editor._setEditorRange();\r\n            event._findButtonEffectTag();\r\n        },\r\n\r\n        onKeyDown_wysiwyg: function (e) {\r\n            // editor._setSelectionNode();\r\n\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            function shortcutCommand(keyCode) {\r\n                const key = event._shortcutKeyCode[keyCode];\r\n                if (!key) return false;\r\n\r\n                editor.execCommand(key[0], false, null);\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            switch (keyCode) {\r\n                case 8: /**backspace key*/\r\n                    if (util.isFormatElement(editor._variable.selectionNode) && editor._variable.selectionNode.previousSibling === null) {\r\n                        e.preventDefault();\r\n                        e.stopPropagation();\r\n                        editor._variable.selectionNode.innerHTML = '&#65279';\r\n                        return false;\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                    let currentNode = editor._variable.selectionNode || editor.getSelectionNode();\r\n                    while (!/^TD$/i.test(currentNode.tagName) && !util.isWysiwygDiv(currentNode)) {\r\n                        currentNode = currentNode.parentNode;\r\n                    }\r\n\r\n                    if (currentNode && /^TD$/i.test(currentNode.tagName)) {\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                    /** if P Tag */\r\n                    if (shift) break;\r\n\r\n                    const tabText = document.createTextNode(new Array(editor._variable.tabSize + 1).join('\\u00A0'));\r\n                    editor.insertNode(tabText);\r\n                    editor.setRange(tabText, editor._variable.tabSize, tabText, editor._variable.tabSize)\r\n\r\n                    break;\r\n            }\r\n        },\r\n\r\n        onKeyUp_wysiwyg: function (e) {\r\n            editor._setEditorRange();\r\n\r\n            /** when format tag deleted */\r\n            if (e.keyCode === 8 && util.isWysiwygDiv(editor._variable.selectionNode)) {\r\n                e.preventDefault();\r\n                e.stopPropagation();\r\n\r\n                const oFormatTag = document.createElement(editor._variable.currentNodes[0]);\r\n                oFormatTag.innerHTML = '&#65279';\r\n\r\n                editor._variable.selectionNode.appendChild(oFormatTag);\r\n                editor._variable.selectionNode = oFormatTag;\r\n                editor.setRange(oFormatTag, 0, oFormatTag, 0);\r\n            }\r\n\r\n            if (event._directionKeyKeyCode.test(e.keyCode)) {\r\n                event._findButtonEffectTag();\r\n            }\r\n        },\r\n\r\n        onScroll_wysiwyg: function () {\r\n            editor.controllersOff();\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) return true;\r\n\r\n            e.stopPropagation();\r\n            e.preventDefault();\r\n            \r\n            if (editor.plugins.image) {\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\r\n        onMouseDown_resizeBar: 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                document.removeEventListener('mousemove', event.resize_editor);\r\n                document.removeEventListener('mouseup', closureFunc);\r\n            }\r\n\r\n            document.addEventListener('mousemove', event.resize_editor);\r\n            document.addEventListener('mouseup', closureFunc);\r\n        },\r\n\r\n        resize_editor: function (e) {\r\n            const resizeInterval = (e.clientY - editor._variable.resizeClientY);\r\n\r\n            context.element.editorArea.style.height = (context.element.editorArea.offsetHeight + resizeInterval) + 'px';\r\n            editor._variable.editorHeight = (context.element.editorArea.offsetHeight + resizeInterval);\r\n\r\n            editor._variable.resizeClientY = e.clientY;\r\n        }\r\n    };\r\n\r\n    /** add event listeners */\r\n    /** tool bar event */\r\n    context.tool.bar.addEventListener('click', event.onClick_toolbar, false);\r\n    context.tool.bar.addEventListener('mousedown', event.onMouseDown_toolbar, false);\r\n    /** editor area */\r\n    context.element.wysiwyg.addEventListener('scroll', event.onScroll_wysiwyg, false);\r\n    context.element.wysiwyg.addEventListener('mouseup', event.onMouseUp_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    /** resize bar */\r\n    context.element.resizebar.addEventListener('mousedown', event.onMouseDown_resizeBar, false);\r\n    /** window resize event */\r\n    window.addEventListener('resize', event.resize_window, false);\r\n\r\n    /** add plugin to plugins object */\r\n    if (plugins) {\r\n        let pluginsValues = Object.values(plugins);\r\n        for (let i = 0, len = pluginsValues.length, plugin; i < len; i++) {\r\n            plugin = pluginsValues[i];\r\n            editor.plugins[plugin.name] = util.copyObj(plugin);\r\n        }\r\n         pluginsValues = null;\r\n    }\r\n\r\n    /** User function */\r\n    return {\r\n        /**\r\n         * @description Copying the contents of the editor to the original textarea\r\n         */\r\n        save: function () {\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.originElement.value = context.element.wysiwyg.innerHTML;\r\n            } else {\r\n                context.element.originElement.value = context.element.code.value;\r\n            }\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        getContent: function () {\r\n            let content = '';\r\n\r\n            if (context.element.wysiwyg.innerText.trim().length === 0) return content;\r\n\r\n            if (editor._variable.wysiwygActive) {\r\n                content = context.element.wysiwyg.innerHTML;\r\n            } else {\r\n                content = context.element.code.value;\r\n            }\r\n            return content;\r\n        },\r\n\r\n        /**\r\n         * @description Change the contents of the suneditor\r\n         * @param {String} content - Content to Input\r\n         */\r\n        setContent: function (content) {\r\n            const innerHTML = util.convertContentForEditor(content);\r\n\r\n            if (editor._variable.wysiwygActive) {\r\n                context.element.wysiwyg.innerHTML = innerHTML;\r\n            } else {\r\n                context.element.code.value = innerHTML;\r\n            }\r\n        },\r\n\r\n        /**\r\n         * @description Add content to the suneditor\r\n         * @param {String} content - to Input\r\n         */\r\n        appendContent: function (content) {\r\n            if (editor._variable.wysiwygActive) {\r\n                const oP = document.createElement('P');\r\n                oP.innerHTML = content;\r\n                context.element.wysiwyg.appendChild(oP);\r\n            } else {\r\n                context.element.code.value += oP.outerHTML;\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        },\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        },\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.user.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            window.removeEventListener('resize', event.resize_window);\r\n            \r\n            /** remove element */\r\n            context.element.topArea.parentNode.removeChild(context.element.topArea);\r\n\r\n            this.save = null;\r\n            this.getContext = null;\r\n            this.getContent = null;\r\n            this.setContent = null;\r\n            this.appendContent = 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    };\r\n};\r\n\r\nexport default core;"],"mappings":"AAAA;AAAA;AAEA;;;;;;;;;AAQA;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;;;;;;;;;;;;AAYA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AACA;AAWA;;;;;;;;;;;;;;AAcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAVA;AACA;AAYA;;;;;;AAMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;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;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;AAOA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;AAMA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;;;AAKA;AACA;AACA;AAEA;AACA;AAEA;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;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;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;;;;;;;AAOA;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;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AATA;AAWA;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;AA5CA;AA8CA;AACA;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;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AAYA;AACA;AACA;AAEA;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;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAJA;AAMA;AACA;AACA;;;;;;;AAOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;AAUA;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;AAAA;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;AAAA;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;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;;;;;AAUA;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;AAAA;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;AAAA;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;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAFA;AAIA;AACA;AACA;;;;;;AAMA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AAUA;AA//BA;AAkgCA;;;;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;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;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AACA;AAEA;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;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;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;AACA;AACA;AA5CA;AA8CA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AAEA;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;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAEA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AAEA;AACA;AACA;AADA;AAEA;AACA;AACA;AAEA;AA1CA;AA4CA;AAEA;AACA;AAEA;AACA;AACA;AADA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AAEA;AACA;AAEA;AAEA;AACA;AACA;AACA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA;AACA;AAEA;AACA;AAEA;AACA;AAxYA;AA2YA;AACA;AAAA;AACA;AAAA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AAAA;AAEA;AACA;AAAA;AACA;AACA;AAAA;AACA;AACA;AACA;AACA;AAAA;AACA;AAEA;AACA;AACA;AADA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;AAIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AAEA;AACA;AAAA;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AApHA;AAsHA;AACA;AACA","sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/lib/core.js\n"); /***/ }), @@ -166,7 +166,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n\n/**\r\n * @description util /***/ (function(module, __webpack_exports__, __webpack_require__) { "use strict"; -eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _lib_util__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./lib/util */ \"./src/lib/util.js\");\n/* harmony import */ var _lib_core__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./lib/core */ \"./src/lib/core.js\");\n/* harmony import */ var _lib_constructor__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./lib/constructor */ \"./src/lib/constructor.js\");\n/* harmony import */ var _lib_context__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./lib/context */ \"./src/lib/context.js\");\n/* harmony import */ var _lang_en__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./lang/en */ \"./src/lang/en.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\n\n\n/* harmony default export */ __webpack_exports__[\"default\"] = ({\n util: _lib_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"],\n core: _lib_core__WEBPACK_IMPORTED_MODULE_1__[\"default\"],\n _Constructor: _lib_constructor__WEBPACK_IMPORTED_MODULE_2__[\"default\"],\n _Context: _lib_context__WEBPACK_IMPORTED_MODULE_3__[\"default\"],\n _defaultLang: _lang_en__WEBPACK_IMPORTED_MODULE_4__[\"default\"],\n\n /**\r\n * @description Returns the create function with preset options.\r\n * If the options overlap, the options of the 'create' function take precedence.\r\n * @param {Json} options - Initialization options\r\n * @returns {function}\r\n */\n init: function (init_options) {\n const self = this;\n return {\n create: function (idOrElement, options) {\n return self.create(idOrElement, options, init_options);\n }\n };\n },\n\n /**\r\n * @description Create the suneditor\r\n * @param {String|Element} idOrElement - textarea Id or textarea element\r\n * @param {Json} options - user options\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 create: function (idOrElement, options, _init_options) {\n if (typeof options !== 'object') options = {};\n\n if (_init_options) {\n options = Object.assign(this.util.copyObj(_init_options), options);\n }\n\n const element = typeof idOrElement === 'string' ? document.getElementById(idOrElement) : idOrElement;\n\n if (!element) {\n if (typeof idOrElement === 'string') {\n throw Error('[SUNEDITOR.create.fail] The element for that id was not found (ID:\"' + idOrElement + '\")');\n }\n\n throw Error('[SUNEDITOR.create.fail] suneditor requires textarea\\'s element or id value');\n }\n\n const cons = this._Constructor.init(element, options, options.lang || this._defaultLang, options.plugins, this.util.convertContentForEditor);\n\n if (document.getElementById(cons.constructed._top.id)) {\n throw Error('[SUNEDITOR.create.fail] The ID of the suneditor you are trying to create already exists (ID:\"' + cons.constructed._top.id + '\")');\n }\n\n element.style.display = 'none';\n cons.constructed._top.style.display = 'block';\n /** Create to sibling node */\n\n if (typeof element.nextElementSibling === 'object') {\n element.parentNode.insertBefore(cons.constructed._top, element.nextElementSibling);\n } else {\n element.parentNode.appendChild(cons.constructed._top);\n }\n\n return this.core(this._Context(element, cons.constructed, cons.options), this.util, options.modules, cons.plugins, cons.options.lang);\n }\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvc3VuZWRpdG9yLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4vc3JjL3N1bmVkaXRvci5qcz9iNTliIl0sInNvdXJjZXNDb250ZW50IjpbIi8qXHJcbiAqIHd5c2l3eWcgd2ViIGVkaXRvclxyXG4gKlxyXG4gKiBzdW5lZGl0b3IuanNcclxuICogQ29weXJpZ2h0IDIwMTcgSmlIb25nIExlZS5cclxuICogTUlUIGxpY2Vuc2UuXHJcbiAqL1xyXG4ndXNlIHN0cmljdCc7XHJcblxyXG5pbXBvcnQgdXRpbCBmcm9tICcuL2xpYi91dGlsJztcclxuaW1wb3J0IGNvcmUgZnJvbSAnLi9saWIvY29yZSc7XHJcbmltcG9ydCBfQ29uc3RydWN0b3IgZnJvbSAnLi9saWIvY29uc3RydWN0b3InO1xyXG5pbXBvcnQgX0NvbnRleHQgZnJvbSAnLi9saWIvY29udGV4dCc7XHJcbmltcG9ydCBfZGVmYXVsdExhbmcgZnJvbSAnLi9sYW5nL2VuJ1xyXG5cclxuXHJcbmV4cG9ydCBkZWZhdWx0IHtcclxuICAgIHV0aWwsXHJcbiAgICBjb3JlLFxyXG4gICAgX0NvbnN0cnVjdG9yLFxyXG4gICAgX0NvbnRleHQsXHJcbiAgICBfZGVmYXVsdExhbmcsXHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBAZGVzY3JpcHRpb24gUmV0dXJucyB0aGUgY3JlYXRlIGZ1bmN0aW9uIHdpdGggcHJlc2V0IG9wdGlvbnMuXHJcbiAgICAgKiBJZiB0aGUgb3B0aW9ucyBvdmVybGFwLCB0aGUgb3B0aW9ucyBvZiB0aGUgJ2NyZWF0ZScgZnVuY3Rpb24gdGFrZSBwcmVjZWRlbmNlLlxyXG4gICAgICogQHBhcmFtIHtKc29ufSBvcHRpb25zIC0gSW5pdGlhbGl6YXRpb24gb3B0aW9uc1xyXG4gICAgICogQHJldHVybnMge2Z1bmN0aW9ufVxyXG4gICAgICovXHJcbiAgICBpbml0OiBmdW5jdGlvbiAoaW5pdF9vcHRpb25zKSB7XHJcbiAgICAgICAgY29uc3Qgc2VsZiA9IHRoaXM7XHJcblxyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIGNyZWF0ZTogZnVuY3Rpb24gKGlkT3JFbGVtZW50LCBvcHRpb25zKSB7XHJcbiAgICAgICAgICAgICAgICByZXR1cm4gc2VsZi5jcmVhdGUoaWRPckVsZW1lbnQsIG9wdGlvbnMsIGluaXRfb3B0aW9ucyk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICB9LFxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQGRlc2NyaXB0aW9uIENyZWF0ZSB0aGUgc3VuZWRpdG9yXHJcbiAgICAgKiBAcGFyYW0ge1N0cmluZ3xFbGVtZW50fSBpZE9yRWxlbWVudCAtIHRleHRhcmVhIElkIG9yIHRleHRhcmVhIGVsZW1lbnRcclxuICAgICAqIEBwYXJhbSB7SnNvbn0gb3B0aW9ucyAtIHVzZXIgb3B0aW9uc1xyXG4gICAgICogQHJldHVybnMge3tzYXZlOiBzYXZlLCBnZXRDb250ZXh0OiBnZXRDb250ZXh0LCBnZXRDb250ZW50OiBnZXRDb250ZW50LCBzZXRDb250ZW50OiBzZXRDb250ZW50LCBhcHBlbmRDb250ZW50OiBhcHBlbmRDb250ZW50LCBkaXNhYmxlZDogZGlzYWJsZWQsIGVuYWJsZWQ6IGVuYWJsZWQsIHNob3c6IHNob3csIGhpZGU6IGhpZGUsIGRlc3Ryb3k6IGRlc3Ryb3l9fVxyXG4gICAgICovXHJcbiAgICBjcmVhdGU6IGZ1bmN0aW9uIChpZE9yRWxlbWVudCwgb3B0aW9ucywgX2luaXRfb3B0aW9ucykge1xyXG4gICAgICAgIGlmICh0eXBlb2Ygb3B0aW9ucyAhPT0gJ29iamVjdCcpIG9wdGlvbnMgPSB7fTtcclxuICAgICAgICBpZiAoX2luaXRfb3B0aW9ucykge1xyXG4gICAgICAgICAgICBvcHRpb25zID0gT2JqZWN0LmFzc2lnbih0aGlzLnV0aWwuY29weU9iaihfaW5pdF9vcHRpb25zKSwgb3B0aW9ucyk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIFxyXG4gICAgICAgIGNvbnN0IGVsZW1lbnQgPSB0eXBlb2YgaWRPckVsZW1lbnQgPT09ICdzdHJpbmcnID8gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoaWRPckVsZW1lbnQpIDogaWRPckVsZW1lbnQ7XHJcblxyXG4gICAgICAgIGlmICghZWxlbWVudCkge1xyXG4gICAgICAgICAgICBpZiAodHlwZW9mIGlkT3JFbGVtZW50ID09PSAnc3RyaW5nJykge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgRXJyb3IoJ1tTVU5FRElUT1IuY3JlYXRlLmZhaWxdIFRoZSBlbGVtZW50IGZvciB0aGF0IGlkIHdhcyBub3QgZm91bmQgKElEOlwiJyArIGlkT3JFbGVtZW50ICsgJ1wiKScpO1xyXG4gICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICB0aHJvdyBFcnJvcignW1NVTkVESVRPUi5jcmVhdGUuZmFpbF0gc3VuZWRpdG9yIHJlcXVpcmVzIHRleHRhcmVhXFwncyBlbGVtZW50IG9yIGlkIHZhbHVlJyk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBjb25zID0gdGhpcy5fQ29uc3RydWN0b3IuaW5pdChlbGVtZW50LCBvcHRpb25zLCAob3B0aW9ucy5sYW5nIHx8ICB0aGlzLl9kZWZhdWx0TGFuZyksIG9wdGlvbnMucGx1Z2lucywgdGhpcy51dGlsLmNvbnZlcnRDb250ZW50Rm9yRWRpdG9yKTtcclxuXHJcbiAgICAgICAgaWYgKGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKGNvbnMuY29uc3RydWN0ZWQuX3RvcC5pZCkpIHtcclxuICAgICAgICAgICAgdGhyb3cgRXJyb3IoJ1tTVU5FRElUT1IuY3JlYXRlLmZhaWxdIFRoZSBJRCBvZiB0aGUgc3VuZWRpdG9yIHlvdSBhcmUgdHJ5aW5nIHRvIGNyZWF0ZSBhbHJlYWR5IGV4aXN0cyAoSUQ6XCInICsgY29ucy5jb25zdHJ1Y3RlZC5fdG9wLmlkICsgJ1wiKScpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgZWxlbWVudC5zdHlsZS5kaXNwbGF5ID0gJ25vbmUnO1xyXG4gICAgICAgIGNvbnMuY29uc3RydWN0ZWQuX3RvcC5zdHlsZS5kaXNwbGF5ID0gJ2Jsb2NrJztcclxuXHJcbiAgICAgICAgLyoqIENyZWF0ZSB0byBzaWJsaW5nIG5vZGUgKi9cclxuICAgICAgICBpZiAodHlwZW9mIGVsZW1lbnQubmV4dEVsZW1lbnRTaWJsaW5nID09PSAnb2JqZWN0Jykge1xyXG4gICAgICAgICAgICBlbGVtZW50LnBhcmVudE5vZGUuaW5zZXJ0QmVmb3JlKGNvbnMuY29uc3RydWN0ZWQuX3RvcCwgZWxlbWVudC5uZXh0RWxlbWVudFNpYmxpbmcpO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIGVsZW1lbnQucGFyZW50Tm9kZS5hcHBlbmRDaGlsZChjb25zLmNvbnN0cnVjdGVkLl90b3ApO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgcmV0dXJuIHRoaXMuY29yZSh0aGlzLl9Db250ZXh0KGVsZW1lbnQsIGNvbnMuY29uc3RydWN0ZWQsIGNvbnMub3B0aW9ucyksIHRoaXMudXRpbCwgb3B0aW9ucy5tb2R1bGVzLCBjb25zLnBsdWdpbnMsIGNvbnMub3B0aW9ucy5sYW5nKTtcclxuICAgIH1cclxufTtcclxuIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTs7Ozs7OztBQU9BO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBR0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7O0FBTUE7QUFDQTtBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBSEE7QUFLQTtBQUNBO0FBQ0E7Ozs7OztBQU1BO0FBQ0E7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBRUE7QUFDQTtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUE5REEiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/suneditor.js\n"); +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _lib_util__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./lib/util */ \"./src/lib/util.js\");\n/* harmony import */ var _lib_core__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./lib/core */ \"./src/lib/core.js\");\n/* harmony import */ var _lib_constructor__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./lib/constructor */ \"./src/lib/constructor.js\");\n/* harmony import */ var _lib_context__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./lib/context */ \"./src/lib/context.js\");\n/* harmony import */ var _lang_en__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./lang/en */ \"./src/lang/en.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\n\n\n/* harmony default export */ __webpack_exports__[\"default\"] = ({\n util: _lib_util__WEBPACK_IMPORTED_MODULE_0__[\"default\"],\n core: _lib_core__WEBPACK_IMPORTED_MODULE_1__[\"default\"],\n _Constructor: _lib_constructor__WEBPACK_IMPORTED_MODULE_2__[\"default\"],\n _Context: _lib_context__WEBPACK_IMPORTED_MODULE_3__[\"default\"],\n _defaultLang: _lang_en__WEBPACK_IMPORTED_MODULE_4__[\"default\"],\n\n /**\r\n * @description Returns the create function with preset options.\r\n * If the options overlap, the options of the 'create' function take precedence.\r\n * @param {Json} options - Initialization options\r\n * @returns {function}\r\n */\n init: function (init_options) {\n const self = this;\n return {\n create: function (idOrElement, options) {\n return self.create(idOrElement, options, init_options);\n }\n };\n },\n\n /**\r\n * @description Create the suneditor\r\n * @param {String|Element} idOrElement - textarea Id or textarea element\r\n * @param {Json} options - user options\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 create: function (idOrElement, options, _init_options) {\n if (typeof options !== 'object') options = {};\n\n if (_init_options) {\n options = Object.assign(this.util.copyObj(_init_options), options);\n }\n\n const element = typeof idOrElement === 'string' ? document.getElementById(idOrElement) : idOrElement;\n\n if (!element) {\n if (typeof idOrElement === 'string') {\n throw Error('[SUNEDITOR.create.fail] The element for that id was not found (ID:\"' + idOrElement + '\")');\n }\n\n throw Error('[SUNEDITOR.create.fail] suneditor requires textarea\\'s element or id value');\n }\n\n const cons = this._Constructor.init(element, options, options.lang || this._defaultLang, options.plugins, this.util.convertContentForEditor);\n\n if (document.getElementById(cons.constructed._top.id)) {\n throw Error('[SUNEDITOR.create.fail] The ID of the suneditor you are trying to create already exists (ID:\"' + cons.constructed._top.id + '\")');\n }\n\n element.style.display = 'none';\n cons.constructed._top.style.display = 'block';\n /** Create to sibling node */\n\n if (typeof element.nextElementSibling === 'object') {\n element.parentNode.insertBefore(cons.constructed._top, element.nextElementSibling);\n } else {\n element.parentNode.appendChild(cons.constructed._top);\n }\n\n return this.core(this._Context(element, cons.constructed, cons.options), this.util, cons.plugins, cons.options.lang);\n }\n});//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvc3VuZWRpdG9yLmpzLmpzIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vLy4vc3JjL3N1bmVkaXRvci5qcz9iNTliIl0sInNvdXJjZXNDb250ZW50IjpbIi8qXHJcbiAqIHd5c2l3eWcgd2ViIGVkaXRvclxyXG4gKlxyXG4gKiBzdW5lZGl0b3IuanNcclxuICogQ29weXJpZ2h0IDIwMTcgSmlIb25nIExlZS5cclxuICogTUlUIGxpY2Vuc2UuXHJcbiAqL1xyXG4ndXNlIHN0cmljdCc7XHJcblxyXG5pbXBvcnQgdXRpbCBmcm9tICcuL2xpYi91dGlsJztcclxuaW1wb3J0IGNvcmUgZnJvbSAnLi9saWIvY29yZSc7XHJcbmltcG9ydCBfQ29uc3RydWN0b3IgZnJvbSAnLi9saWIvY29uc3RydWN0b3InO1xyXG5pbXBvcnQgX0NvbnRleHQgZnJvbSAnLi9saWIvY29udGV4dCc7XHJcbmltcG9ydCBfZGVmYXVsdExhbmcgZnJvbSAnLi9sYW5nL2VuJ1xyXG5cclxuXHJcbmV4cG9ydCBkZWZhdWx0IHtcclxuICAgIHV0aWwsXHJcbiAgICBjb3JlLFxyXG4gICAgX0NvbnN0cnVjdG9yLFxyXG4gICAgX0NvbnRleHQsXHJcbiAgICBfZGVmYXVsdExhbmcsXHJcblxyXG4gICAgLyoqXHJcbiAgICAgKiBAZGVzY3JpcHRpb24gUmV0dXJucyB0aGUgY3JlYXRlIGZ1bmN0aW9uIHdpdGggcHJlc2V0IG9wdGlvbnMuXHJcbiAgICAgKiBJZiB0aGUgb3B0aW9ucyBvdmVybGFwLCB0aGUgb3B0aW9ucyBvZiB0aGUgJ2NyZWF0ZScgZnVuY3Rpb24gdGFrZSBwcmVjZWRlbmNlLlxyXG4gICAgICogQHBhcmFtIHtKc29ufSBvcHRpb25zIC0gSW5pdGlhbGl6YXRpb24gb3B0aW9uc1xyXG4gICAgICogQHJldHVybnMge2Z1bmN0aW9ufVxyXG4gICAgICovXHJcbiAgICBpbml0OiBmdW5jdGlvbiAoaW5pdF9vcHRpb25zKSB7XHJcbiAgICAgICAgY29uc3Qgc2VsZiA9IHRoaXM7XHJcblxyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIGNyZWF0ZTogZnVuY3Rpb24gKGlkT3JFbGVtZW50LCBvcHRpb25zKSB7XHJcbiAgICAgICAgICAgICAgICByZXR1cm4gc2VsZi5jcmVhdGUoaWRPckVsZW1lbnQsIG9wdGlvbnMsIGluaXRfb3B0aW9ucyk7XHJcbiAgICAgICAgICAgIH1cclxuICAgICAgICB9XHJcbiAgICB9LFxyXG5cclxuICAgIC8qKlxyXG4gICAgICogQGRlc2NyaXB0aW9uIENyZWF0ZSB0aGUgc3VuZWRpdG9yXHJcbiAgICAgKiBAcGFyYW0ge1N0cmluZ3xFbGVtZW50fSBpZE9yRWxlbWVudCAtIHRleHRhcmVhIElkIG9yIHRleHRhcmVhIGVsZW1lbnRcclxuICAgICAqIEBwYXJhbSB7SnNvbn0gb3B0aW9ucyAtIHVzZXIgb3B0aW9uc1xyXG4gICAgICogQHJldHVybnMge3tzYXZlOiBzYXZlLCBnZXRDb250ZXh0OiBnZXRDb250ZXh0LCBnZXRDb250ZW50OiBnZXRDb250ZW50LCBzZXRDb250ZW50OiBzZXRDb250ZW50LCBhcHBlbmRDb250ZW50OiBhcHBlbmRDb250ZW50LCBkaXNhYmxlZDogZGlzYWJsZWQsIGVuYWJsZWQ6IGVuYWJsZWQsIHNob3c6IHNob3csIGhpZGU6IGhpZGUsIGRlc3Ryb3k6IGRlc3Ryb3l9fVxyXG4gICAgICovXHJcbiAgICBjcmVhdGU6IGZ1bmN0aW9uIChpZE9yRWxlbWVudCwgb3B0aW9ucywgX2luaXRfb3B0aW9ucykge1xyXG4gICAgICAgIGlmICh0eXBlb2Ygb3B0aW9ucyAhPT0gJ29iamVjdCcpIG9wdGlvbnMgPSB7fTtcclxuICAgICAgICBpZiAoX2luaXRfb3B0aW9ucykge1xyXG4gICAgICAgICAgICBvcHRpb25zID0gT2JqZWN0LmFzc2lnbih0aGlzLnV0aWwuY29weU9iaihfaW5pdF9vcHRpb25zKSwgb3B0aW9ucyk7XHJcbiAgICAgICAgfVxyXG4gICAgICAgIFxyXG4gICAgICAgIGNvbnN0IGVsZW1lbnQgPSB0eXBlb2YgaWRPckVsZW1lbnQgPT09ICdzdHJpbmcnID8gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoaWRPckVsZW1lbnQpIDogaWRPckVsZW1lbnQ7XHJcblxyXG4gICAgICAgIGlmICghZWxlbWVudCkge1xyXG4gICAgICAgICAgICBpZiAodHlwZW9mIGlkT3JFbGVtZW50ID09PSAnc3RyaW5nJykge1xyXG4gICAgICAgICAgICAgICAgdGhyb3cgRXJyb3IoJ1tTVU5FRElUT1IuY3JlYXRlLmZhaWxdIFRoZSBlbGVtZW50IGZvciB0aGF0IGlkIHdhcyBub3QgZm91bmQgKElEOlwiJyArIGlkT3JFbGVtZW50ICsgJ1wiKScpO1xyXG4gICAgICAgICAgICB9XHJcblxyXG4gICAgICAgICAgICB0aHJvdyBFcnJvcignW1NVTkVESVRPUi5jcmVhdGUuZmFpbF0gc3VuZWRpdG9yIHJlcXVpcmVzIHRleHRhcmVhXFwncyBlbGVtZW50IG9yIGlkIHZhbHVlJyk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBjb25zdCBjb25zID0gdGhpcy5fQ29uc3RydWN0b3IuaW5pdChlbGVtZW50LCBvcHRpb25zLCAob3B0aW9ucy5sYW5nIHx8ICB0aGlzLl9kZWZhdWx0TGFuZyksIG9wdGlvbnMucGx1Z2lucywgdGhpcy51dGlsLmNvbnZlcnRDb250ZW50Rm9yRWRpdG9yKTtcclxuXHJcbiAgICAgICAgaWYgKGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKGNvbnMuY29uc3RydWN0ZWQuX3RvcC5pZCkpIHtcclxuICAgICAgICAgICAgdGhyb3cgRXJyb3IoJ1tTVU5FRElUT1IuY3JlYXRlLmZhaWxdIFRoZSBJRCBvZiB0aGUgc3VuZWRpdG9yIHlvdSBhcmUgdHJ5aW5nIHRvIGNyZWF0ZSBhbHJlYWR5IGV4aXN0cyAoSUQ6XCInICsgY29ucy5jb25zdHJ1Y3RlZC5fdG9wLmlkICsgJ1wiKScpO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgZWxlbWVudC5zdHlsZS5kaXNwbGF5ID0gJ25vbmUnO1xyXG4gICAgICAgIGNvbnMuY29uc3RydWN0ZWQuX3RvcC5zdHlsZS5kaXNwbGF5ID0gJ2Jsb2NrJztcclxuXHJcbiAgICAgICAgLyoqIENyZWF0ZSB0byBzaWJsaW5nIG5vZGUgKi9cclxuICAgICAgICBpZiAodHlwZW9mIGVsZW1lbnQubmV4dEVsZW1lbnRTaWJsaW5nID09PSAnb2JqZWN0Jykge1xyXG4gICAgICAgICAgICBlbGVtZW50LnBhcmVudE5vZGUuaW5zZXJ0QmVmb3JlKGNvbnMuY29uc3RydWN0ZWQuX3RvcCwgZWxlbWVudC5uZXh0RWxlbWVudFNpYmxpbmcpO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAgIGVsZW1lbnQucGFyZW50Tm9kZS5hcHBlbmRDaGlsZChjb25zLmNvbnN0cnVjdGVkLl90b3ApO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgcmV0dXJuIHRoaXMuY29yZSh0aGlzLl9Db250ZXh0KGVsZW1lbnQsIGNvbnMuY29uc3RydWN0ZWQsIGNvbnMub3B0aW9ucyksIHRoaXMudXRpbCwgY29ucy5wbHVnaW5zLCBjb25zLm9wdGlvbnMubGFuZyk7XHJcbiAgICB9XHJcbn07XHJcbiJdLCJtYXBwaW5ncyI6IkFBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7Ozs7Ozs7QUFPQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUdBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7Ozs7OztBQU1BO0FBQ0E7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUhBO0FBS0E7QUFDQTtBQUNBOzs7Ozs7QUFNQTtBQUNBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBO0FBQ0E7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBOURBIiwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/suneditor.js\n"); /***/ }), diff --git a/sample/html/getting-started.html b/sample/html/getting-started.html index 868c7d2af..d56daa06c 100644 --- a/sample/html/getting-started.html +++ b/sample/html/getting-started.html @@ -97,7 +97,7 @@

1. Default options

// insert options }); -

2. Load plugins

+

2. Load only what you want

import 'suneditor/dist/css/suneditor.min.css'
 import suneditor from 'suneditor'
 import {en, ko} from 'suneditor/src/lang'
@@ -134,7 +134,7 @@ 

3. Load all plugins

import plugins from 'suneditor/src/plugins' suneditor.create('sample', { - modules: plugins, + plugins: plugins, buttonList: [ ['undo', 'redo'], ['font', 'fontSize', 'formatBlock'], @@ -148,6 +148,20 @@

3. Load all plugins

['fullScreen', 'showBlocks', 'codeView'], ['preview', 'print'] ] +}) + +// You can also load what you want +suneditor.create('sample', { + plugins: [ + plugins.font + plugins.fontSize, + plugins.formatBlock + ], + buttonList: [ + ['bold', 'underline', 'italic', 'strike', 'subscript', 'superscript'], + ['font', 'fontSize', 'formatBlock'], + ['removeFormat'] + ] })

4. Plugins can be used directly in the button list

diff --git a/src/lib/context.js b/src/lib/context.js index fff8baba7..f131e307c 100644 --- a/src/lib/context.js +++ b/src/lib/context.js @@ -45,7 +45,8 @@ const _Context = function (element, cons, options) { font: options.font, fontSize: options.fontSize, height: options.height.match(/\d+/)[0], - showPathLabel: options.showPathLabel + showPathLabel: options.showPathLabel, + display: options.display }, dialog: {}, submenu: {} diff --git a/src/lib/core.js b/src/lib/core.js index c7ce8efe4..80451c124 100644 --- a/src/lib/core.js +++ b/src/lib/core.js @@ -4,12 +4,11 @@ * @description SunEditor core closure * @param context * @param util - * @param modules * @param plugins * @param lang * @returns {{save: save, getContext: getContext, getContent: getContent, setContent: setContent, appendContent: appendContent, disabled: disabled, enabled: enabled, show: show, hide: hide, destroy: destroy}} */ -const core = function (context, util, modules, plugins, lang) { +const core = function (context, util, plugins, lang) { /** * @description Practical editor function * This function is 'this' used by other plugins @@ -20,11 +19,6 @@ const core = function (context, util, modules, plugins, lang) { */ context: context, - /** - * @description loaded modules - */ - modules: {}, - /** * @description loaded plugins */ @@ -1555,8 +1549,7 @@ const core = function (context, util, modules, plugins, lang) { */ show: function () { const topAreaStyle = context.element.topArea.style; - topAreaStyle.cssText = editor._variable.originCssText; - if (topAreaStyle.display === 'none') topAreaStyle.display = 'block'; + if (topAreaStyle.display === 'none') topAreaStyle.display = context.user.display; }, /** diff --git a/src/suneditor.js b/src/suneditor.js index 5dc290be9..ae02ac957 100644 --- a/src/suneditor.js +++ b/src/suneditor.js @@ -75,6 +75,6 @@ export default { element.parentNode.appendChild(cons.constructed._top); } - return this.core(this._Context(element, cons.constructed, cons.options), this.util, options.modules, cons.plugins, cons.options.lang); + return this.core(this._Context(element, cons.constructed, cons.options), this.util, cons.plugins, cons.options.lang); } };