1 |
- {"version":3,"sources":["webpack:///./node_modules/_wangeditor@3.1.1@wangeditor/release/wangEditor.js","webpack:///./src/components/fuEditor/index.vue?77fc","webpack:///src/components/fuEditor/index.vue","webpack:///./src/components/fuEditor/index.vue?8a78","webpack:///./src/components/fuEditor/index.vue","webpack:///./src/components/fuEditor/index.vue?f47f"],"names":["global","factory","module","exports","this","polyfill","Object","assign","target","varArgs","TypeError","to","index","arguments","length","nextSource","nextKey","prototype","hasOwnProperty","call","Element","matches","matchesSelector","mozMatchesSelector","msMatchesSelector","oMatchesSelector","webkitMatchesSelector","s","document","ownerDocument","querySelectorAll","i","item","createElemByHTML","html","div","createElement","innerHTML","children","isDOMList","selector","HTMLCollection","NodeList","result","eventList","DomElement","nodeType","selectorResult","Array","replace","trim","indexOf","$","constructor","forEach","fn","elem","clone","deep","cloneList","push","cloneNode","get","first","last","on","type","types","split","addEventListener","e","off","removeEventListener","attr","key","val","getAttribute","setAttribute","addClass","className","arr","filter","join","removeClass","css","currentStyle","style","styleArr","resultArr","map","show","hide","childNodes","append","$children","child","appendChild","remove","parent","parentElement","removeChild","isContain","$child","contains","getSizeData","getBoundingClientRect","getNodeName","nodeName","find","text","value","focus","parentUntil","_currentElem","results","equal","$elem","insertBefore","$referenceNode","referenceNode","parentNode","insertAfter","lastChild","nextSibling","offAll","config","menus","fontNames","colors","emotions","title","content","alt","src","zIndex","debug","linkCheck","link","linkImgCheck","pasteFilterStyle","pasteIgnoreImg","pasteTextHandle","showLinkImg","linkImgCallback","url","uploadImgMaxSize","uploadImgShowBase64","uploadFileName","uploadImgParams","uploadImgHeaders","withCredentials","uploadImgTimeout","uploadImgHooks","before","xhr","editor","files","success","fail","error","timeout","qiniu","UA","_ua","navigator","userAgent","isWebkit","reg","test","isIE","window","objForEach","obj","arrForEach","fakeArr","getRandom","prefix","Math","random","toString","slice","replaceHtmlSymbol","isFunction","Bold","_active","onClick","isSeleEmpty","selection","isSelectionEmpty","createEmptyRange","cmd","do","collapseRange","restoreSelection","tryChangeActive","queryCommandState","replaceLang","str","langArgs","_emptyFn","DropList","menu","opt","_this","$container","$title","titleHtml","list","$list","elemHtml","$li","hideTimeoutId","setTimeout","_rendered","_show","Head","droplist","width","_command","FontSize","FontName","fontName","clearTimeout","$menuELem","menuHeight","height","showTimeoutId","$selectionElem","getSelectionContainerElem","$textElem","cmdValue","queryCommandValue","emptyFn","_isCreatedPanelMenus","Panel","Link","Italic","Redo","StrikeThrough","Underline","Undo","List","Justify","ForeColor","color","BackColor","Quote","Code","Emoticon","Table","Video","Image","imgMenuId","$body","$textContainerElem","$closeBtn","$tabTitleContainer","$tabContentContainer","tabs","tabTitleArr","tabContentArr","tab","tabIndex","tpl","$content","_index","stopPropagation","events","event","needToHide","$inputs","_hideOtherPanels","panel","$linkelem","createRangeByElem","_createPanel","getSelectionText","inputLinkId","inputTextId","btnOkId","btnDelId","delBtnDisplay","$link","$text","_insertLink","_delLink","$selectionELem","selectionText","checkResult","alert","$parent","$targetELem","$startElem","getSelectionStartElem","$endElem","getSelectionEndElem","$code","textId","btnId","_insertCode","_updateCode","$parentElem","tabConfig","emotData","emotType","faceHtml","$target","insertHtml","_insert","emotHtml","_createEditPanel","_createInsertPanel","btnInsertId","textRowNum","textColNum","rowNum","parseInt","colNum","r","c","_this2","addRowBtnId","addColBtnId","delRowBtnId","delColBtnId","delTableBtnId","_addRow","_addCol","_delRow","_delCol","_delTable","_getLocationData","$tr","$tds","tdLength","td","$tbody","$trs","trLength","tr","locationData","trData","$currentTr","tdData","newTr","tdIndex","$trParent","$currentTd","name","toLowerCase","newTd","$table","textValId","width30","width50","width100","delBtn","tabsConfig","$img","_selectedImg","uploadImg","upTriggerId","upFileId","linkUrlId","linkBtnId","$file","fileElem","click","fileList","$linkUrl","insertLinkImg","tabsConfigResult","uploadImgServer","customUploadImg","FileReader","MenuConstructors","Menus","getPasteText","clipboardData","originalEvent","pasteText","getData","getPasteHtml","filterStyle","ignoreImg","pasteHtml","docSplitHtml","getPasteImgs","txt","items","getAsFile","getChildrenJSON","curElem","elemResult","textContent","tag","attrData","attrList","attributes","attrListLength","attrs","Text","Command","API","_currentRange","Progress","_time","_isShow","_isRender","_timeoutId","$textContainer","$bar","bold","head","fontSize","italic","redo","strikeThrough","underline","undo","justify","foreColor","backColor","quote","code","emoticon","table","video","image","init","configMenus","menuKey","MenuConstructor","_addToToolbar","_bindEvent","$toolbarElem","getRange","changeActive","clear","initSelection","getJSON","_saveRangeRealTime","_enterKeyHandle","_clearHandle","_pasteHandle","_tabHandle","_imgHandle","_dragHandle","saveRange","insertEmptyP","$p","pHandle","codeHandle","selectionNodeName","parentNodeName","queryCommandSupported","_willBreakCode","preventDefault","_startOffset","startOffset","codeLength","keyCode","txtHtml","pasteTime","canDo","now","Date","flag","resetTime","ex","pasteFiles","img","$document","dataTransfer","_useStyleWithCSS","execCommand","_name","_execCommand","change","_insertHTML","range","insertNode","deleteContents","pasteHTML","_insertElem","_range","getSelection","rangeCount","getRangeAt","$containerElem","toStart","collapse","commonAncestorContainer","startContainer","endContainer","endOffset","removeAllRanges","addRange","setEnd","isContent","createRange","selectNodeContents","selectNode","progress","timeoutId","_hide","_typeof","Symbol","iterator","UploadImg","_alert","alertInfo","debugInfo","customAlert","Error","onload","callback","onerror","onabort","_this3","maxSize","maxSizeM","maxLength","uploadImgMaxLength","uploadImgParamsWithUrl","hooks","resultFiles","errInfo","file","size","bind","formdata","FormData","uploadImgServerArr","uploadImgServerHash","XMLHttpRequest","open","ontimeout","upload","onprogress","percent","progressBar","lengthComputable","loaded","total","onreadystatechange","readyState","status","responseText","JSON","parse","customInsert","errno","data","beforeResult","prevent","msg","setRequestHeader","send","reader","readAsDataURL","editorId","Editor","toolbarSelector","textSelector","id","customConfig","_initConfig","langConfig","lang","RegExp","_initDom","$toolbarSelector","config$$1","toolbarElemId","textElemId","compositionEnd","onfocus","onblur","isFocus","isChild","isToolbar","isMenu","_initCommand","_initSelectionAPI","_initUploadImg","_initMenus","_initText","newLine","$last","onChangeTimeoutId","beforeChangeHtml","onchangeTimeout","onchange","currentHtml","create","_offAllEvent","inlinecss","getElementsByTagName","wangEditor","render","_vm","_h","$createElement","_c","_self","staticClass","ref","staticRenderFns","info_","model","prop","props","String","default","isClear","Boolean","watch","mounted","seteditor","methods","console","log","insertImg","component"],"mappings":"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","file":"js/chunk-6cf4df16.9fbedacd.js","sourcesContent":["(function (global, factory) {\n\ttypeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n\ttypeof define === 'function' && define.amd ? define(factory) :\n\t(global.wangEditor = factory());\n}(this, (function () { 'use strict';\n\n/*\n poly-fill\n*/\n\nvar polyfill = function () {\n\n // Object.assign\n if (typeof Object.assign != 'function') {\n Object.assign = function (target, varArgs) {\n // .length of function is 2\n if (target == null) {\n // TypeError if undefined or null\n throw new TypeError('Cannot convert undefined or null to object');\n }\n\n var to = Object(target);\n\n for (var index = 1; index < arguments.length; index++) {\n var nextSource = arguments[index];\n\n if (nextSource != null) {\n // Skip over if undefined or null\n for (var nextKey in nextSource) {\n // Avoid bugs when hasOwnProperty is shadowed\n if (Object.prototype.hasOwnProperty.call(nextSource, nextKey)) {\n to[nextKey] = nextSource[nextKey];\n }\n }\n }\n }\n return to;\n };\n }\n\n // IE 中兼容 Element.prototype.matches\n if (!Element.prototype.matches) {\n Element.prototype.matches = Element.prototype.matchesSelector || Element.prototype.mozMatchesSelector || Element.prototype.msMatchesSelector || Element.prototype.oMatchesSelector || Element.prototype.webkitMatchesSelector || function (s) {\n var matches = (this.document || this.ownerDocument).querySelectorAll(s),\n i = matches.length;\n while (--i >= 0 && matches.item(i) !== this) {}\n return i > -1;\n };\n }\n};\n\n/*\n DOM 操作 API\n*/\n\n// 根据 html 代码片段创建 dom 对象\nfunction createElemByHTML(html) {\n var div = void 0;\n div = document.createElement('div');\n div.innerHTML = html;\n return div.children;\n}\n\n// 是否是 DOM List\nfunction isDOMList(selector) {\n if (!selector) {\n return false;\n }\n if (selector instanceof HTMLCollection || selector instanceof NodeList) {\n return true;\n }\n return false;\n}\n\n// 封装 document.querySelectorAll\nfunction querySelectorAll(selector) {\n var result = document.querySelectorAll(selector);\n if (isDOMList(result)) {\n return result;\n } else {\n return [result];\n }\n}\n\n// 记录所有的事件绑定\nvar eventList = [];\n\n// 创建构造函数\nfunction DomElement(selector) {\n if (!selector) {\n return;\n }\n\n // selector 本来就是 DomElement 对象,直接返回\n if (selector instanceof DomElement) {\n return selector;\n }\n\n this.selector = selector;\n var nodeType = selector.nodeType;\n\n // 根据 selector 得出的结果(如 DOM,DOM List)\n var selectorResult = [];\n if (nodeType === 9) {\n // document 节点\n selectorResult = [selector];\n } else if (nodeType === 1) {\n // 单个 DOM 节点\n selectorResult = [selector];\n } else if (isDOMList(selector) || selector instanceof Array) {\n // DOM List 或者数组\n selectorResult = selector;\n } else if (typeof selector === 'string') {\n // 字符串\n selector = selector.replace('/\\n/mg', '').trim();\n if (selector.indexOf('<') === 0) {\n // 如 <div>\n selectorResult = createElemByHTML(selector);\n } else {\n // 如 #id .class\n selectorResult = querySelectorAll(selector);\n }\n }\n\n var length = selectorResult.length;\n if (!length) {\n // 空数组\n return this;\n }\n\n // 加入 DOM 节点\n var i = void 0;\n for (i = 0; i < length; i++) {\n this[i] = selectorResult[i];\n }\n this.length = length;\n}\n\n// 修改原型\nDomElement.prototype = {\n constructor: DomElement,\n\n // 类数组,forEach\n forEach: function forEach(fn) {\n var i = void 0;\n for (i = 0; i < this.length; i++) {\n var elem = this[i];\n var result = fn.call(elem, elem, i);\n if (result === false) {\n break;\n }\n }\n return this;\n },\n\n // clone\n clone: function clone(deep) {\n var cloneList = [];\n this.forEach(function (elem) {\n cloneList.push(elem.cloneNode(!!deep));\n });\n return $(cloneList);\n },\n\n // 获取第几个元素\n get: function get(index) {\n var length = this.length;\n if (index >= length) {\n index = index % length;\n }\n return $(this[index]);\n },\n\n // 第一个\n first: function first() {\n return this.get(0);\n },\n\n // 最后一个\n last: function last() {\n var length = this.length;\n return this.get(length - 1);\n },\n\n // 绑定事件\n on: function on(type, selector, fn) {\n // selector 不为空,证明绑定事件要加代理\n if (!fn) {\n fn = selector;\n selector = null;\n }\n\n // type 是否有多个\n var types = [];\n types = type.split(/\\s+/);\n\n return this.forEach(function (elem) {\n types.forEach(function (type) {\n if (!type) {\n return;\n }\n\n // 记录下,方便后面解绑\n eventList.push({\n elem: elem,\n type: type,\n fn: fn\n });\n\n if (!selector) {\n // 无代理\n elem.addEventListener(type, fn);\n return;\n }\n\n // 有代理\n elem.addEventListener(type, function (e) {\n var target = e.target;\n if (target.matches(selector)) {\n fn.call(target, e);\n }\n });\n });\n });\n },\n\n // 取消事件绑定\n off: function off(type, fn) {\n return this.forEach(function (elem) {\n elem.removeEventListener(type, fn);\n });\n },\n\n // 获取/设置 属性\n attr: function attr(key, val) {\n if (val == null) {\n // 获取值\n return this[0].getAttribute(key);\n } else {\n // 设置值\n return this.forEach(function (elem) {\n elem.setAttribute(key, val);\n });\n }\n },\n\n // 添加 class\n addClass: function addClass(className) {\n if (!className) {\n return this;\n }\n return this.forEach(function (elem) {\n var arr = void 0;\n if (elem.className) {\n // 解析当前 className 转换为数组\n arr = elem.className.split(/\\s/);\n arr = arr.filter(function (item) {\n return !!item.trim();\n });\n // 添加 class\n if (arr.indexOf(className) < 0) {\n arr.push(className);\n }\n // 修改 elem.class\n elem.className = arr.join(' ');\n } else {\n elem.className = className;\n }\n });\n },\n\n // 删除 class\n removeClass: function removeClass(className) {\n if (!className) {\n return this;\n }\n return this.forEach(function (elem) {\n var arr = void 0;\n if (elem.className) {\n // 解析当前 className 转换为数组\n arr = elem.className.split(/\\s/);\n arr = arr.filter(function (item) {\n item = item.trim();\n // 删除 class\n if (!item || item === className) {\n return false;\n }\n return true;\n });\n // 修改 elem.class\n elem.className = arr.join(' ');\n }\n });\n },\n\n // 修改 css\n css: function css(key, val) {\n var currentStyle = key + ':' + val + ';';\n return this.forEach(function (elem) {\n var style = (elem.getAttribute('style') || '').trim();\n var styleArr = void 0,\n resultArr = [];\n if (style) {\n // 将 style 按照 ; 拆分为数组\n styleArr = style.split(';');\n styleArr.forEach(function (item) {\n // 对每项样式,按照 : 拆分为 key 和 value\n var arr = item.split(':').map(function (i) {\n return i.trim();\n });\n if (arr.length === 2) {\n resultArr.push(arr[0] + ':' + arr[1]);\n }\n });\n // 替换或者新增\n resultArr = resultArr.map(function (item) {\n if (item.indexOf(key) === 0) {\n return currentStyle;\n } else {\n return item;\n }\n });\n if (resultArr.indexOf(currentStyle) < 0) {\n resultArr.push(currentStyle);\n }\n // 结果\n elem.setAttribute('style', resultArr.join('; '));\n } else {\n // style 无值\n elem.setAttribute('style', currentStyle);\n }\n });\n },\n\n // 显示\n show: function show() {\n return this.css('display', 'block');\n },\n\n // 隐藏\n hide: function hide() {\n return this.css('display', 'none');\n },\n\n // 获取子节点\n children: function children() {\n var elem = this[0];\n if (!elem) {\n return null;\n }\n\n return $(elem.children);\n },\n\n // 获取子节点(包括文本节点)\n childNodes: function childNodes() {\n var elem = this[0];\n if (!elem) {\n return null;\n }\n\n return $(elem.childNodes);\n },\n\n // 增加子节点\n append: function append($children) {\n return this.forEach(function (elem) {\n $children.forEach(function (child) {\n elem.appendChild(child);\n });\n });\n },\n\n // 移除当前节点\n remove: function remove() {\n return this.forEach(function (elem) {\n if (elem.remove) {\n elem.remove();\n } else {\n var parent = elem.parentElement;\n parent && parent.removeChild(elem);\n }\n });\n },\n\n // 是否包含某个子节点\n isContain: function isContain($child) {\n var elem = this[0];\n var child = $child[0];\n return elem.contains(child);\n },\n\n // 尺寸数据\n getSizeData: function getSizeData() {\n var elem = this[0];\n return elem.getBoundingClientRect(); // 可得到 bottom height left right top width 的数据\n },\n\n // 封装 nodeName\n getNodeName: function getNodeName() {\n var elem = this[0];\n return elem.nodeName;\n },\n\n // 从当前元素查找\n find: function find(selector) {\n var elem = this[0];\n return $(elem.querySelectorAll(selector));\n },\n\n // 获取当前元素的 text\n text: function text(val) {\n if (!val) {\n // 获取 text\n var elem = this[0];\n return elem.innerHTML.replace(/<.*?>/g, function () {\n return '';\n });\n } else {\n // 设置 text\n return this.forEach(function (elem) {\n elem.innerHTML = val;\n });\n }\n },\n\n // 获取 html\n html: function html(value) {\n var elem = this[0];\n if (value == null) {\n return elem.innerHTML;\n } else {\n elem.innerHTML = value;\n return this;\n }\n },\n\n // 获取 value\n val: function val() {\n var elem = this[0];\n return elem.value.trim();\n },\n\n // focus\n focus: function focus() {\n return this.forEach(function (elem) {\n elem.focus();\n });\n },\n\n // parent\n parent: function parent() {\n var elem = this[0];\n return $(elem.parentElement);\n },\n\n // parentUntil 找到符合 selector 的父节点\n parentUntil: function parentUntil(selector, _currentElem) {\n var results = document.querySelectorAll(selector);\n var length = results.length;\n if (!length) {\n // 传入的 selector 无效\n return null;\n }\n\n var elem = _currentElem || this[0];\n if (elem.nodeName === 'BODY') {\n return null;\n }\n\n var parent = elem.parentElement;\n var i = void 0;\n for (i = 0; i < length; i++) {\n if (parent === results[i]) {\n // 找到,并返回\n return $(parent);\n }\n }\n\n // 继续查找\n return this.parentUntil(selector, parent);\n },\n\n // 判断两个 elem 是否相等\n equal: function equal($elem) {\n if ($elem.nodeType === 1) {\n return this[0] === $elem;\n } else {\n return this[0] === $elem[0];\n }\n },\n\n // 将该元素插入到某个元素前面\n insertBefore: function insertBefore(selector) {\n var $referenceNode = $(selector);\n var referenceNode = $referenceNode[0];\n if (!referenceNode) {\n return this;\n }\n return this.forEach(function (elem) {\n var parent = referenceNode.parentNode;\n parent.insertBefore(elem, referenceNode);\n });\n },\n\n // 将该元素插入到某个元素后面\n insertAfter: function insertAfter(selector) {\n var $referenceNode = $(selector);\n var referenceNode = $referenceNode[0];\n if (!referenceNode) {\n return this;\n }\n return this.forEach(function (elem) {\n var parent = referenceNode.parentNode;\n if (parent.lastChild === referenceNode) {\n // 最后一个元素\n parent.appendChild(elem);\n } else {\n // 不是最后一个元素\n parent.insertBefore(elem, referenceNode.nextSibling);\n }\n });\n }\n};\n\n// new 一个对象\nfunction $(selector) {\n return new DomElement(selector);\n}\n\n// 解绑所有事件,用于销毁编辑器\n$.offAll = function () {\n eventList.forEach(function (item) {\n var elem = item.elem;\n var type = item.type;\n var fn = item.fn;\n // 解绑\n elem.removeEventListener(type, fn);\n });\n};\n\n/*\n 配置信息\n*/\n\nvar config = {\n\n // 默认菜单配置\n menus: ['head', 'bold', 'fontSize', 'fontName', 'italic', 'underline', 'strikeThrough', 'foreColor', 'backColor', 'link', 'list', 'justify', 'quote', 'emoticon', 'image', 'table', 'video', 'code', 'undo', 'redo'],\n\n fontNames: ['宋体', '微软雅黑', 'Arial', 'Tahoma', 'Verdana'],\n\n colors: ['#000000', '#eeece0', '#1c487f', '#4d80bf', '#c24f4a', '#8baa4a', '#7b5ba1', '#46acc8', '#f9963b', '#ffffff'],\n\n // // 语言配置\n // lang: {\n // '设置标题': 'title',\n // '正文': 'p',\n // '链接文字': 'link text',\n // '链接': 'link',\n // '插入': 'insert',\n // '创建': 'init'\n // },\n\n // 表情\n emotions: [{\n // tab 的标题\n title: '默认',\n // type -> 'emoji' / 'image'\n type: 'image',\n // content -> 数组\n content: [{\n alt: '[坏笑]',\n src: 'http://img.t.sinajs.cn/t4/appstyle/expression/ext/normal/50/pcmoren_huaixiao_org.png'\n }, {\n alt: '[舔屏]',\n src: 'http://img.t.sinajs.cn/t4/appstyle/expression/ext/normal/40/pcmoren_tian_org.png'\n }, {\n alt: '[污]',\n src: 'http://img.t.sinajs.cn/t4/appstyle/expression/ext/normal/3c/pcmoren_wu_org.png'\n }]\n }, {\n // tab 的标题\n title: '新浪',\n // type -> 'emoji' / 'image'\n type: 'image',\n // content -> 数组\n content: [{\n src: 'http://img.t.sinajs.cn/t35/style/images/common/face/ext/normal/7a/shenshou_thumb.gif',\n alt: '[草泥马]'\n }, {\n src: 'http://img.t.sinajs.cn/t35/style/images/common/face/ext/normal/60/horse2_thumb.gif',\n alt: '[神马]'\n }, {\n src: 'http://img.t.sinajs.cn/t35/style/images/common/face/ext/normal/bc/fuyun_thumb.gif',\n alt: '[浮云]'\n }]\n }, {\n // tab 的标题\n title: 'emoji',\n // type -> 'emoji' / 'image'\n type: 'emoji',\n // content -> 数组\n content: '😀 😃 😄 😁 😆 😅 😂 😊 😇 🙂 🙃 😉 😓 😪 😴 🙄 🤔 😬 🤐'.split(/\\s/)\n }],\n\n // 编辑区域的 z-index\n zIndex: 10000,\n\n // 是否开启 debug 模式(debug 模式下错误会 throw error 形式抛出)\n debug: false,\n\n // 插入链接时候的格式校验\n linkCheck: function linkCheck(text, link) {\n // text 是插入的文字\n // link 是插入的链接\n return true; // 返回 true 即表示成功\n // return '校验失败' // 返回字符串即表示失败的提示信息\n },\n\n // 插入网络图片的校验\n linkImgCheck: function linkImgCheck(src) {\n // src 即图片的地址\n return true; // 返回 true 即表示成功\n // return '校验失败' // 返回字符串即表示失败的提示信息\n },\n\n // 粘贴过滤样式,默认开启\n pasteFilterStyle: true,\n\n // 粘贴内容时,忽略图片。默认关闭\n pasteIgnoreImg: false,\n\n // 对粘贴的文字进行自定义处理,返回处理后的结果。编辑器会将处理后的结果粘贴到编辑区域中。\n // IE 暂时不支持\n pasteTextHandle: function pasteTextHandle(content) {\n // content 即粘贴过来的内容(html 或 纯文本),可进行自定义处理然后返回\n return content;\n },\n\n // onchange 事件\n // onchange: function (html) {\n // // html 即变化之后的内容\n // console.log(html)\n // },\n\n // 是否显示添加网络图片的 tab\n showLinkImg: true,\n\n // 插入网络图片的回调\n linkImgCallback: function linkImgCallback(url) {\n // console.log(url) // url 即插入图片的地址\n },\n\n // 默认上传图片 max size: 5M\n uploadImgMaxSize: 5 * 1024 * 1024,\n\n // 配置一次最多上传几个图片\n // uploadImgMaxLength: 5,\n\n // 上传图片,是否显示 base64 格式\n uploadImgShowBase64: false,\n\n // 上传图片,server 地址(如果有值,则 base64 格式的配置则失效)\n // uploadImgServer: '/upload',\n\n // 自定义配置 filename\n uploadFileName: '',\n\n // 上传图片的自定义参数\n uploadImgParams: {\n // token: 'abcdef12345'\n },\n\n // 上传图片的自定义header\n uploadImgHeaders: {\n // 'Accept': 'text/x-json'\n },\n\n // 配置 XHR withCredentials\n withCredentials: false,\n\n // 自定义上传图片超时时间 ms\n uploadImgTimeout: 10000,\n\n // 上传图片 hook \n uploadImgHooks: {\n // customInsert: function (insertLinkImg, result, editor) {\n // console.log('customInsert')\n // // 图片上传并返回结果,自定义插入图片的事件,而不是编辑器自动插入图片\n // const data = result.data1 || []\n // data.forEach(link => {\n // insertLinkImg(link)\n // })\n // },\n before: function before(xhr, editor, files) {\n // 图片上传之前触发\n\n // 如果返回的结果是 {prevent: true, msg: 'xxxx'} 则表示用户放弃上传\n // return {\n // prevent: true,\n // msg: '放弃上传'\n // }\n },\n success: function success(xhr, editor, result) {\n // 图片上传并返回结果,图片插入成功之后触发\n },\n fail: function fail(xhr, editor, result) {\n // 图片上传并返回结果,但图片插入错误时触发\n },\n error: function error(xhr, editor) {\n // 图片上传出错时触发\n },\n timeout: function timeout(xhr, editor) {\n // 图片上传超时时触发\n }\n },\n\n // 是否上传七牛云,默认为 false\n qiniu: false\n\n};\n\n/*\n 工具\n*/\n\n// 和 UA 相关的属性\nvar UA = {\n _ua: navigator.userAgent,\n\n // 是否 webkit\n isWebkit: function isWebkit() {\n var reg = /webkit/i;\n return reg.test(this._ua);\n },\n\n // 是否 IE\n isIE: function isIE() {\n return 'ActiveXObject' in window;\n }\n};\n\n// 遍历对象\nfunction objForEach(obj, fn) {\n var key = void 0,\n result = void 0;\n for (key in obj) {\n if (obj.hasOwnProperty(key)) {\n result = fn.call(obj, key, obj[key]);\n if (result === false) {\n break;\n }\n }\n }\n}\n\n// 遍历类数组\nfunction arrForEach(fakeArr, fn) {\n var i = void 0,\n item = void 0,\n result = void 0;\n var length = fakeArr.length || 0;\n for (i = 0; i < length; i++) {\n item = fakeArr[i];\n result = fn.call(fakeArr, item, i);\n if (result === false) {\n break;\n }\n }\n}\n\n// 获取随机数\nfunction getRandom(prefix) {\n return prefix + Math.random().toString().slice(2);\n}\n\n// 替换 html 特殊字符\nfunction replaceHtmlSymbol(html) {\n if (html == null) {\n return '';\n }\n return html.replace(/</gm, '<').replace(/>/gm, '>').replace(/\"/gm, '"').replace(/(\\r\\n|\\r|\\n)/g, '<br/>');\n}\n\n// 返回百分比的格式\n\n\n// 判断是不是 function\nfunction isFunction(fn) {\n return typeof fn === 'function';\n}\n\n/*\n bold-menu\n*/\n// 构造函数\nfunction Bold(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-bold\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nBold.prototype = {\n constructor: Bold,\n\n // 点击事件\n onClick: function onClick(e) {\n // 点击菜单将触发这里\n\n var editor = this.editor;\n var isSeleEmpty = editor.selection.isSelectionEmpty();\n\n if (isSeleEmpty) {\n // 选区是空的,插入并选中一个“空白”\n editor.selection.createEmptyRange();\n }\n\n // 执行 bold 命令\n editor.cmd.do('bold');\n\n if (isSeleEmpty) {\n // 需要将选取折叠起来\n editor.selection.collapseRange();\n editor.selection.restoreSelection();\n }\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n if (editor.cmd.queryCommandState('bold')) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n 替换多语言\n */\n\nvar replaceLang = function (editor, str) {\n var langArgs = editor.config.langArgs || [];\n var result = str;\n\n langArgs.forEach(function (item) {\n var reg = item.reg;\n var val = item.val;\n\n if (reg.test(result)) {\n result = result.replace(reg, function () {\n return val;\n });\n }\n });\n\n return result;\n};\n\n/*\n droplist\n*/\nvar _emptyFn = function _emptyFn() {};\n\n// 构造函数\nfunction DropList(menu, opt) {\n var _this = this;\n\n // droplist 所依附的菜单\n var editor = menu.editor;\n this.menu = menu;\n this.opt = opt;\n // 容器\n var $container = $('<div class=\"w-e-droplist\"></div>');\n\n // 标题\n var $title = opt.$title;\n var titleHtml = void 0;\n if ($title) {\n // 替换多语言\n titleHtml = $title.html();\n titleHtml = replaceLang(editor, titleHtml);\n $title.html(titleHtml);\n\n $title.addClass('w-e-dp-title');\n $container.append($title);\n }\n\n var list = opt.list || [];\n var type = opt.type || 'list'; // 'list' 列表形式(如“标题”菜单) / 'inline-block' 块状形式(如“颜色”菜单)\n var onClick = opt.onClick || _emptyFn;\n\n // 加入 DOM 并绑定事件\n var $list = $('<ul class=\"' + (type === 'list' ? 'w-e-list' : 'w-e-block') + '\"></ul>');\n $container.append($list);\n list.forEach(function (item) {\n var $elem = item.$elem;\n\n // 替换多语言\n var elemHtml = $elem.html();\n elemHtml = replaceLang(editor, elemHtml);\n $elem.html(elemHtml);\n\n var value = item.value;\n var $li = $('<li class=\"w-e-item\"></li>');\n if ($elem) {\n $li.append($elem);\n $list.append($li);\n $li.on('click', function (e) {\n onClick(value);\n\n // 隐藏\n _this.hideTimeoutId = setTimeout(function () {\n _this.hide();\n }, 0);\n });\n }\n });\n\n // 绑定隐藏事件\n $container.on('mouseleave', function (e) {\n _this.hideTimeoutId = setTimeout(function () {\n _this.hide();\n }, 0);\n });\n\n // 记录属性\n this.$container = $container;\n\n // 基本属性\n this._rendered = false;\n this._show = false;\n}\n\n// 原型\nDropList.prototype = {\n constructor: DropList,\n\n // 显示(插入DOM)\n show: function show() {\n if (this.hideTimeoutId) {\n // 清除之前的定时隐藏\n clearTimeout(this.hideTimeoutId);\n }\n\n var menu = this.menu;\n var $menuELem = menu.$elem;\n var $container = this.$container;\n if (this._show) {\n return;\n }\n if (this._rendered) {\n // 显示\n $container.show();\n } else {\n // 加入 DOM 之前先定位位置\n var menuHeight = $menuELem.getSizeData().height || 0;\n var width = this.opt.width || 100; // 默认为 100\n $container.css('margin-top', menuHeight + 'px').css('width', width + 'px');\n\n // 加入到 DOM\n $menuELem.append($container);\n this._rendered = true;\n }\n\n // 修改属性\n this._show = true;\n },\n\n // 隐藏(移除DOM)\n hide: function hide() {\n if (this.showTimeoutId) {\n // 清除之前的定时显示\n clearTimeout(this.showTimeoutId);\n }\n\n var $container = this.$container;\n if (!this._show) {\n return;\n }\n // 隐藏并需改属性\n $container.hide();\n this._show = false;\n }\n};\n\n/*\n menu - header\n*/\n// 构造函数\nfunction Head(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-header\"></i></div>');\n this.type = 'droplist';\n\n // 当前是否 active 状态\n this._active = false;\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 100,\n $title: $('<p>设置标题</p>'),\n type: 'list', // droplist 以列表形式展示\n list: [{ $elem: $('<h1>H1</h1>'), value: '<h1>' }, { $elem: $('<h2>H2</h2>'), value: '<h2>' }, { $elem: $('<h3>H3</h3>'), value: '<h3>' }, { $elem: $('<h4>H4</h4>'), value: '<h4>' }, { $elem: $('<h5>H5</h5>'), value: '<h5>' }, { $elem: $('<p>正文</p>'), value: '<p>' }],\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 Head 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nHead.prototype = {\n constructor: Head,\n\n // 执行命令\n _command: function _command(value) {\n var editor = this.editor;\n\n var $selectionElem = editor.selection.getSelectionContainerElem();\n if (editor.$textElem.equal($selectionElem)) {\n // 不能选中多行来设置标题,否则会出现问题\n // 例如选中的是 <p>xxx</p><p>yyy</p> 来设置标题,设置之后会成为 <h1>xxx<br>yyy</h1> 不符合预期\n return;\n }\n\n editor.cmd.do('formatBlock', value);\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n var reg = /^h/i;\n var cmdValue = editor.cmd.queryCommandValue('formatBlock');\n if (reg.test(cmdValue)) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n menu - fontSize\n*/\n\n// 构造函数\nfunction FontSize(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-text-heigh\"></i></div>');\n this.type = 'droplist';\n\n // 当前是否 active 状态\n this._active = false;\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 160,\n $title: $('<p>字号</p>'),\n type: 'list', // droplist 以列表形式展示\n list: [{ $elem: $('<span style=\"font-size: x-small;\">x-small</span>'), value: '1' }, { $elem: $('<span style=\"font-size: small;\">small</span>'), value: '2' }, { $elem: $('<span>normal</span>'), value: '3' }, { $elem: $('<span style=\"font-size: large;\">large</span>'), value: '4' }, { $elem: $('<span style=\"font-size: x-large;\">x-large</span>'), value: '5' }, { $elem: $('<span style=\"font-size: xx-large;\">xx-large</span>'), value: '6' }],\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 FontSize 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nFontSize.prototype = {\n constructor: FontSize,\n\n // 执行命令\n _command: function _command(value) {\n var editor = this.editor;\n editor.cmd.do('fontSize', value);\n }\n};\n\n/*\n menu - fontName\n*/\n\n// 构造函数\nfunction FontName(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-font\"></i></div>');\n this.type = 'droplist';\n\n // 当前是否 active 状态\n this._active = false;\n\n // 获取配置的字体\n var config = editor.config;\n var fontNames = config.fontNames || [];\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 100,\n $title: $('<p>字体</p>'),\n type: 'list', // droplist 以列表形式展示\n list: fontNames.map(function (fontName) {\n return { $elem: $('<span style=\"font-family: ' + fontName + ';\">' + fontName + '</span>'), value: fontName };\n }),\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 FontName 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nFontName.prototype = {\n constructor: FontName,\n\n _command: function _command(value) {\n var editor = this.editor;\n editor.cmd.do('fontName', value);\n }\n};\n\n/*\n panel\n*/\n\nvar emptyFn = function emptyFn() {};\n\n// 记录已经显示 panel 的菜单\nvar _isCreatedPanelMenus = [];\n\n// 构造函数\nfunction Panel(menu, opt) {\n this.menu = menu;\n this.opt = opt;\n}\n\n// 原型\nPanel.prototype = {\n constructor: Panel,\n\n // 显示(插入DOM)\n show: function show() {\n var _this = this;\n\n var menu = this.menu;\n if (_isCreatedPanelMenus.indexOf(menu) >= 0) {\n // 该菜单已经创建了 panel 不能再创建\n return;\n }\n\n var editor = menu.editor;\n var $body = $('body');\n var $textContainerElem = editor.$textContainerElem;\n var opt = this.opt;\n\n // panel 的容器\n var $container = $('<div class=\"w-e-panel-container\"></div>');\n var width = opt.width || 300; // 默认 300px\n $container.css('width', width + 'px').css('margin-left', (0 - width) / 2 + 'px');\n\n // 添加关闭按钮\n var $closeBtn = $('<i class=\"w-e-icon-close w-e-panel-close\"></i>');\n $container.append($closeBtn);\n $closeBtn.on('click', function () {\n _this.hide();\n });\n\n // 准备 tabs 容器\n var $tabTitleContainer = $('<ul class=\"w-e-panel-tab-title\"></ul>');\n var $tabContentContainer = $('<div class=\"w-e-panel-tab-content\"></div>');\n $container.append($tabTitleContainer).append($tabContentContainer);\n\n // 设置高度\n var height = opt.height;\n if (height) {\n $tabContentContainer.css('height', height + 'px').css('overflow-y', 'auto');\n }\n\n // tabs\n var tabs = opt.tabs || [];\n var tabTitleArr = [];\n var tabContentArr = [];\n tabs.forEach(function (tab, tabIndex) {\n if (!tab) {\n return;\n }\n var title = tab.title || '';\n var tpl = tab.tpl || '';\n\n // 替换多语言\n title = replaceLang(editor, title);\n tpl = replaceLang(editor, tpl);\n\n // 添加到 DOM\n var $title = $('<li class=\"w-e-item\">' + title + '</li>');\n $tabTitleContainer.append($title);\n var $content = $(tpl);\n $tabContentContainer.append($content);\n\n // 记录到内存\n $title._index = tabIndex;\n tabTitleArr.push($title);\n tabContentArr.push($content);\n\n // 设置 active 项\n if (tabIndex === 0) {\n $title._active = true;\n $title.addClass('w-e-active');\n } else {\n $content.hide();\n }\n\n // 绑定 tab 的事件\n $title.on('click', function (e) {\n if ($title._active) {\n return;\n }\n // 隐藏所有的 tab\n tabTitleArr.forEach(function ($title) {\n $title._active = false;\n $title.removeClass('w-e-active');\n });\n tabContentArr.forEach(function ($content) {\n $content.hide();\n });\n\n // 显示当前的 tab\n $title._active = true;\n $title.addClass('w-e-active');\n $content.show();\n });\n });\n\n // 绑定关闭事件\n $container.on('click', function (e) {\n // 点击时阻止冒泡\n e.stopPropagation();\n });\n $body.on('click', function (e) {\n _this.hide();\n });\n\n // 添加到 DOM\n $textContainerElem.append($container);\n\n // 绑定 opt 的事件,只有添加到 DOM 之后才能绑定成功\n tabs.forEach(function (tab, index) {\n if (!tab) {\n return;\n }\n var events = tab.events || [];\n events.forEach(function (event) {\n var selector = event.selector;\n var type = event.type;\n var fn = event.fn || emptyFn;\n var $content = tabContentArr[index];\n $content.find(selector).on(type, function (e) {\n e.stopPropagation();\n var needToHide = fn(e);\n // 执行完事件之后,是否要关闭 panel\n if (needToHide) {\n _this.hide();\n }\n });\n });\n });\n\n // focus 第一个 elem\n var $inputs = $container.find('input[type=text],textarea');\n if ($inputs.length) {\n $inputs.get(0).focus();\n }\n\n // 添加到属性\n this.$container = $container;\n\n // 隐藏其他 panel\n this._hideOtherPanels();\n // 记录该 menu 已经创建了 panel\n _isCreatedPanelMenus.push(menu);\n },\n\n // 隐藏(移除DOM)\n hide: function hide() {\n var menu = this.menu;\n var $container = this.$container;\n if ($container) {\n $container.remove();\n }\n\n // 将该 menu 记录中移除\n _isCreatedPanelMenus = _isCreatedPanelMenus.filter(function (item) {\n if (item === menu) {\n return false;\n } else {\n return true;\n }\n });\n },\n\n // 一个 panel 展示时,隐藏其他 panel\n _hideOtherPanels: function _hideOtherPanels() {\n if (!_isCreatedPanelMenus.length) {\n return;\n }\n _isCreatedPanelMenus.forEach(function (menu) {\n var panel = menu.panel || {};\n if (panel.hide) {\n panel.hide();\n }\n });\n }\n};\n\n/*\n menu - link\n*/\n// 构造函数\nfunction Link(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-link\"></i></div>');\n this.type = 'panel';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nLink.prototype = {\n constructor: Link,\n\n // 点击事件\n onClick: function onClick(e) {\n var editor = this.editor;\n var $linkelem = void 0;\n\n if (this._active) {\n // 当前选区在链接里面\n $linkelem = editor.selection.getSelectionContainerElem();\n if (!$linkelem) {\n return;\n }\n // 将该元素都包含在选取之内,以便后面整体替换\n editor.selection.createRangeByElem($linkelem);\n editor.selection.restoreSelection();\n // 显示 panel\n this._createPanel($linkelem.text(), $linkelem.attr('href'));\n } else {\n // 当前选区不在链接里面\n if (editor.selection.isSelectionEmpty()) {\n // 选区是空的,未选中内容\n this._createPanel('', '');\n } else {\n // 选中内容了\n this._createPanel(editor.selection.getSelectionText(), '');\n }\n }\n },\n\n // 创建 panel\n _createPanel: function _createPanel(text, link) {\n var _this = this;\n\n // panel 中需要用到的id\n var inputLinkId = getRandom('input-link');\n var inputTextId = getRandom('input-text');\n var btnOkId = getRandom('btn-ok');\n var btnDelId = getRandom('btn-del');\n\n // 是否显示“删除链接”\n var delBtnDisplay = this._active ? 'inline-block' : 'none';\n\n // 初始化并显示 panel\n var panel = new Panel(this, {\n width: 300,\n // panel 中可包含多个 tab\n tabs: [{\n // tab 的标题\n title: '链接',\n // 模板\n tpl: '<div>\\n <input id=\"' + inputTextId + '\" type=\"text\" class=\"block\" value=\"' + text + '\" placeholder=\"\\u94FE\\u63A5\\u6587\\u5B57\"/></td>\\n <input id=\"' + inputLinkId + '\" type=\"text\" class=\"block\" value=\"' + link + '\" placeholder=\"http://...\"/></td>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + btnOkId + '\" class=\"right\">\\u63D2\\u5165</button>\\n <button id=\"' + btnDelId + '\" class=\"gray right\" style=\"display:' + delBtnDisplay + '\">\\u5220\\u9664\\u94FE\\u63A5</button>\\n </div>\\n </div>',\n // 事件绑定\n events: [\n // 插入链接\n {\n selector: '#' + btnOkId,\n type: 'click',\n fn: function fn() {\n // 执行插入链接\n var $link = $('#' + inputLinkId);\n var $text = $('#' + inputTextId);\n var link = $link.val();\n var text = $text.val();\n _this._insertLink(text, link);\n\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n },\n // 删除链接\n {\n selector: '#' + btnDelId,\n type: 'click',\n fn: function fn() {\n // 执行删除链接\n _this._delLink();\n\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n } // tab end\n ] // tabs end\n });\n\n // 显示 panel\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n // 删除当前链接\n _delLink: function _delLink() {\n if (!this._active) {\n return;\n }\n var editor = this.editor;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n var selectionText = editor.selection.getSelectionText();\n editor.cmd.do('insertHTML', '<span>' + selectionText + '</span>');\n },\n\n // 插入链接\n _insertLink: function _insertLink(text, link) {\n var editor = this.editor;\n var config = editor.config;\n var linkCheck = config.linkCheck;\n var checkResult = true; // 默认为 true\n if (linkCheck && typeof linkCheck === 'function') {\n checkResult = linkCheck(text, link);\n }\n if (checkResult === true) {\n editor.cmd.do('insertHTML', '<a href=\"' + link + '\" target=\"_blank\">' + text + '</a>');\n } else {\n alert(checkResult);\n }\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n if ($selectionELem.getNodeName() === 'A') {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n italic-menu\n*/\n// 构造函数\nfunction Italic(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-italic\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nItalic.prototype = {\n constructor: Italic,\n\n // 点击事件\n onClick: function onClick(e) {\n // 点击菜单将触发这里\n\n var editor = this.editor;\n var isSeleEmpty = editor.selection.isSelectionEmpty();\n\n if (isSeleEmpty) {\n // 选区是空的,插入并选中一个“空白”\n editor.selection.createEmptyRange();\n }\n\n // 执行 italic 命令\n editor.cmd.do('italic');\n\n if (isSeleEmpty) {\n // 需要将选取折叠起来\n editor.selection.collapseRange();\n editor.selection.restoreSelection();\n }\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n if (editor.cmd.queryCommandState('italic')) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n redo-menu\n*/\n// 构造函数\nfunction Redo(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-redo\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nRedo.prototype = {\n constructor: Redo,\n\n // 点击事件\n onClick: function onClick(e) {\n // 点击菜单将触发这里\n\n var editor = this.editor;\n\n // 执行 redo 命令\n editor.cmd.do('redo');\n }\n};\n\n/*\n strikeThrough-menu\n*/\n// 构造函数\nfunction StrikeThrough(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-strikethrough\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nStrikeThrough.prototype = {\n constructor: StrikeThrough,\n\n // 点击事件\n onClick: function onClick(e) {\n // 点击菜单将触发这里\n\n var editor = this.editor;\n var isSeleEmpty = editor.selection.isSelectionEmpty();\n\n if (isSeleEmpty) {\n // 选区是空的,插入并选中一个“空白”\n editor.selection.createEmptyRange();\n }\n\n // 执行 strikeThrough 命令\n editor.cmd.do('strikeThrough');\n\n if (isSeleEmpty) {\n // 需要将选取折叠起来\n editor.selection.collapseRange();\n editor.selection.restoreSelection();\n }\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n if (editor.cmd.queryCommandState('strikeThrough')) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n underline-menu\n*/\n// 构造函数\nfunction Underline(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-underline\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nUnderline.prototype = {\n constructor: Underline,\n\n // 点击事件\n onClick: function onClick(e) {\n // 点击菜单将触发这里\n\n var editor = this.editor;\n var isSeleEmpty = editor.selection.isSelectionEmpty();\n\n if (isSeleEmpty) {\n // 选区是空的,插入并选中一个“空白”\n editor.selection.createEmptyRange();\n }\n\n // 执行 underline 命令\n editor.cmd.do('underline');\n\n if (isSeleEmpty) {\n // 需要将选取折叠起来\n editor.selection.collapseRange();\n editor.selection.restoreSelection();\n }\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n if (editor.cmd.queryCommandState('underline')) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n undo-menu\n*/\n// 构造函数\nfunction Undo(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-undo\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nUndo.prototype = {\n constructor: Undo,\n\n // 点击事件\n onClick: function onClick(e) {\n // 点击菜单将触发这里\n\n var editor = this.editor;\n\n // 执行 undo 命令\n editor.cmd.do('undo');\n }\n};\n\n/*\n menu - list\n*/\n// 构造函数\nfunction List(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-list2\"></i></div>');\n this.type = 'droplist';\n\n // 当前是否 active 状态\n this._active = false;\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 120,\n $title: $('<p>设置列表</p>'),\n type: 'list', // droplist 以列表形式展示\n list: [{ $elem: $('<span><i class=\"w-e-icon-list-numbered\"></i> 有序列表</span>'), value: 'insertOrderedList' }, { $elem: $('<span><i class=\"w-e-icon-list2\"></i> 无序列表</span>'), value: 'insertUnorderedList' }],\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 List 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nList.prototype = {\n constructor: List,\n\n // 执行命令\n _command: function _command(value) {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n editor.selection.restoreSelection();\n if (editor.cmd.queryCommandState(value)) {\n return;\n }\n editor.cmd.do(value);\n\n // 验证列表是否被包裹在 <p> 之内\n var $selectionElem = editor.selection.getSelectionContainerElem();\n if ($selectionElem.getNodeName() === 'LI') {\n $selectionElem = $selectionElem.parent();\n }\n if (/^ol|ul$/i.test($selectionElem.getNodeName()) === false) {\n return;\n }\n if ($selectionElem.equal($textElem)) {\n // 证明是顶级标签,没有被 <p> 包裹\n return;\n }\n var $parent = $selectionElem.parent();\n if ($parent.equal($textElem)) {\n // $parent 是顶级标签,不能删除\n return;\n }\n\n $selectionElem.insertAfter($parent);\n $parent.remove();\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n if (editor.cmd.queryCommandState('insertUnOrderedList') || editor.cmd.queryCommandState('insertOrderedList')) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n menu - justify\n*/\n// 构造函数\nfunction Justify(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-paragraph-left\"></i></div>');\n this.type = 'droplist';\n\n // 当前是否 active 状态\n this._active = false;\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 100,\n $title: $('<p>对齐方式</p>'),\n type: 'list', // droplist 以列表形式展示\n list: [{ $elem: $('<span><i class=\"w-e-icon-paragraph-left\"></i> 靠左</span>'), value: 'justifyLeft' }, { $elem: $('<span><i class=\"w-e-icon-paragraph-center\"></i> 居中</span>'), value: 'justifyCenter' }, { $elem: $('<span><i class=\"w-e-icon-paragraph-right\"></i> 靠右</span>'), value: 'justifyRight' }],\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 List 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nJustify.prototype = {\n constructor: Justify,\n\n // 执行命令\n _command: function _command(value) {\n var editor = this.editor;\n editor.cmd.do(value);\n }\n};\n\n/*\n menu - Forecolor\n*/\n// 构造函数\nfunction ForeColor(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-pencil2\"></i></div>');\n this.type = 'droplist';\n\n // 获取配置的颜色\n var config = editor.config;\n var colors = config.colors || [];\n\n // 当前是否 active 状态\n this._active = false;\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 120,\n $title: $('<p>文字颜色</p>'),\n type: 'inline-block', // droplist 内容以 block 形式展示\n list: colors.map(function (color) {\n return { $elem: $('<i style=\"color:' + color + ';\" class=\"w-e-icon-pencil2\"></i>'), value: color };\n }),\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 ForeColor 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nForeColor.prototype = {\n constructor: ForeColor,\n\n // 执行命令\n _command: function _command(value) {\n var editor = this.editor;\n editor.cmd.do('foreColor', value);\n }\n};\n\n/*\n menu - BackColor\n*/\n// 构造函数\nfunction BackColor(editor) {\n var _this = this;\n\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-paint-brush\"></i></div>');\n this.type = 'droplist';\n\n // 获取配置的颜色\n var config = editor.config;\n var colors = config.colors || [];\n\n // 当前是否 active 状态\n this._active = false;\n\n // 初始化 droplist\n this.droplist = new DropList(this, {\n width: 120,\n $title: $('<p>背景色</p>'),\n type: 'inline-block', // droplist 内容以 block 形式展示\n list: colors.map(function (color) {\n return { $elem: $('<i style=\"color:' + color + ';\" class=\"w-e-icon-paint-brush\"></i>'), value: color };\n }),\n onClick: function onClick(value) {\n // 注意 this 是指向当前的 BackColor 对象\n _this._command(value);\n }\n });\n}\n\n// 原型\nBackColor.prototype = {\n constructor: BackColor,\n\n // 执行命令\n _command: function _command(value) {\n var editor = this.editor;\n editor.cmd.do('backColor', value);\n }\n};\n\n/*\n menu - quote\n*/\n// 构造函数\nfunction Quote(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-quotes-left\"></i>\\n </div>');\n this.type = 'click';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nQuote.prototype = {\n constructor: Quote,\n\n onClick: function onClick(e) {\n var editor = this.editor;\n var $selectionElem = editor.selection.getSelectionContainerElem();\n var nodeName = $selectionElem.getNodeName();\n\n if (!UA.isIE()) {\n if (nodeName === 'BLOCKQUOTE') {\n // 撤销 quote\n editor.cmd.do('formatBlock', '<P>');\n } else {\n // 转换为 quote\n editor.cmd.do('formatBlock', '<BLOCKQUOTE>');\n }\n return;\n }\n\n // IE 中不支持 formatBlock <BLOCKQUOTE> ,要用其他方式兼容\n var content = void 0,\n $targetELem = void 0;\n if (nodeName === 'P') {\n // 将 P 转换为 quote\n content = $selectionElem.text();\n $targetELem = $('<blockquote>' + content + '</blockquote>');\n $targetELem.insertAfter($selectionElem);\n $selectionElem.remove();\n return;\n }\n if (nodeName === 'BLOCKQUOTE') {\n // 撤销 quote\n content = $selectionElem.text();\n $targetELem = $('<p>' + content + '</p>');\n $targetELem.insertAfter($selectionElem);\n $selectionElem.remove();\n }\n },\n\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n var reg = /^BLOCKQUOTE$/i;\n var cmdValue = editor.cmd.queryCommandValue('formatBlock');\n if (reg.test(cmdValue)) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n menu - code\n*/\n// 构造函数\nfunction Code(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-terminal\"></i>\\n </div>');\n this.type = 'panel';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nCode.prototype = {\n constructor: Code,\n\n onClick: function onClick(e) {\n var editor = this.editor;\n var $startElem = editor.selection.getSelectionStartElem();\n var $endElem = editor.selection.getSelectionEndElem();\n var isSeleEmpty = editor.selection.isSelectionEmpty();\n var selectionText = editor.selection.getSelectionText();\n var $code = void 0;\n\n if (!$startElem.equal($endElem)) {\n // 跨元素选择,不做处理\n editor.selection.restoreSelection();\n return;\n }\n if (!isSeleEmpty) {\n // 选取不是空,用 <code> 包裹即可\n $code = $('<code>' + selectionText + '</code>');\n editor.cmd.do('insertElem', $code);\n editor.selection.createRangeByElem($code, false);\n editor.selection.restoreSelection();\n return;\n }\n\n // 选取是空,且没有夸元素选择,则插入 <pre><code></code></prev>\n if (this._active) {\n // 选中状态,将编辑内容\n this._createPanel($startElem.html());\n } else {\n // 未选中状态,将创建内容\n this._createPanel();\n }\n },\n\n _createPanel: function _createPanel(value) {\n var _this = this;\n\n // value - 要编辑的内容\n value = value || '';\n var type = !value ? 'new' : 'edit';\n var textId = getRandom('texxt');\n var btnId = getRandom('btn');\n\n var panel = new Panel(this, {\n width: 500,\n // 一个 Panel 包含多个 tab\n tabs: [{\n // 标题\n title: '插入代码',\n // 模板\n tpl: '<div>\\n <textarea id=\"' + textId + '\" style=\"height:145px;;\">' + value + '</textarea>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + btnId + '\" class=\"right\">\\u63D2\\u5165</button>\\n </div>\\n <div>',\n // 事件绑定\n events: [\n // 插入代码\n {\n selector: '#' + btnId,\n type: 'click',\n fn: function fn() {\n var $text = $('#' + textId);\n var text = $text.val() || $text.html();\n text = replaceHtmlSymbol(text);\n if (type === 'new') {\n // 新插入\n _this._insertCode(text);\n } else {\n // 编辑更新\n _this._updateCode(text);\n }\n\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n } // first tab end\n ] // tabs end\n }); // new Panel end\n\n // 显示 panel\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n // 插入代码\n _insertCode: function _insertCode(value) {\n var editor = this.editor;\n editor.cmd.do('insertHTML', '<pre><code>' + value + '</code></pre><p><br></p>');\n },\n\n // 更新代码\n _updateCode: function _updateCode(value) {\n var editor = this.editor;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n $selectionELem.html(value);\n editor.selection.restoreSelection();\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n var $parentElem = $selectionELem.parent();\n if ($selectionELem.getNodeName() === 'CODE' && $parentElem.getNodeName() === 'PRE') {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n menu - emoticon\n*/\n// 构造函数\nfunction Emoticon(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\">\\n <i class=\"w-e-icon-happy\"></i>\\n </div>');\n this.type = 'panel';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nEmoticon.prototype = {\n constructor: Emoticon,\n\n onClick: function onClick() {\n this._createPanel();\n },\n\n _createPanel: function _createPanel() {\n var _this = this;\n\n var editor = this.editor;\n var config = editor.config;\n // 获取表情配置\n var emotions = config.emotions || [];\n\n // 创建表情 dropPanel 的配置\n var tabConfig = [];\n emotions.forEach(function (emotData) {\n var emotType = emotData.type;\n var content = emotData.content || [];\n\n // 这一组表情最终拼接出来的 html\n var faceHtml = '';\n\n // emoji 表情\n if (emotType === 'emoji') {\n content.forEach(function (item) {\n if (item) {\n faceHtml += '<span class=\"w-e-item\">' + item + '</span>';\n }\n });\n }\n // 图片表情\n if (emotType === 'image') {\n content.forEach(function (item) {\n var src = item.src;\n var alt = item.alt;\n if (src) {\n // 加一个 data-w-e 属性,点击图片的时候不再提示编辑图片\n faceHtml += '<span class=\"w-e-item\"><img src=\"' + src + '\" alt=\"' + alt + '\" data-w-e=\"1\"/></span>';\n }\n });\n }\n\n tabConfig.push({\n title: emotData.title,\n tpl: '<div class=\"w-e-emoticon-container\">' + faceHtml + '</div>',\n events: [{\n selector: 'span.w-e-item',\n type: 'click',\n fn: function fn(e) {\n var target = e.target;\n var $target = $(target);\n var nodeName = $target.getNodeName();\n\n var insertHtml = void 0;\n if (nodeName === 'IMG') {\n // 插入图片\n insertHtml = $target.parent().html();\n } else {\n // 插入 emoji\n insertHtml = '<span>' + $target.html() + '</span>';\n }\n\n _this._insert(insertHtml);\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n });\n });\n\n var panel = new Panel(this, {\n width: 300,\n height: 200,\n // 一个 Panel 包含多个 tab\n tabs: tabConfig\n });\n\n // 显示 panel\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n // 插入表情\n _insert: function _insert(emotHtml) {\n var editor = this.editor;\n editor.cmd.do('insertHTML', emotHtml);\n }\n};\n\n/*\n menu - table\n*/\n// 构造函数\nfunction Table(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-table2\"></i></div>');\n this.type = 'panel';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nTable.prototype = {\n constructor: Table,\n\n onClick: function onClick() {\n if (this._active) {\n // 编辑现有表格\n this._createEditPanel();\n } else {\n // 插入新表格\n this._createInsertPanel();\n }\n },\n\n // 创建插入新表格的 panel\n _createInsertPanel: function _createInsertPanel() {\n var _this = this;\n\n // 用到的 id\n var btnInsertId = getRandom('btn');\n var textRowNum = getRandom('row');\n var textColNum = getRandom('col');\n\n var panel = new Panel(this, {\n width: 250,\n // panel 包含多个 tab\n tabs: [{\n // 标题\n title: '插入表格',\n // 模板\n tpl: '<div>\\n <p style=\"text-align:left; padding:5px 0;\">\\n \\u521B\\u5EFA\\n <input id=\"' + textRowNum + '\" type=\"text\" value=\"5\" style=\"width:40px;text-align:center;\"/>\\n \\u884C\\n <input id=\"' + textColNum + '\" type=\"text\" value=\"5\" style=\"width:40px;text-align:center;\"/>\\n \\u5217\\u7684\\u8868\\u683C\\n </p>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + btnInsertId + '\" class=\"right\">\\u63D2\\u5165</button>\\n </div>\\n </div>',\n // 事件绑定\n events: [{\n // 点击按钮,插入表格\n selector: '#' + btnInsertId,\n type: 'click',\n fn: function fn() {\n var rowNum = parseInt($('#' + textRowNum).val());\n var colNum = parseInt($('#' + textColNum).val());\n\n if (rowNum && colNum && rowNum > 0 && colNum > 0) {\n // form 数据有效\n _this._insert(rowNum, colNum);\n }\n\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n } // first tab end\n ] // tabs end\n }); // panel end\n\n // 展示 panel\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n // 插入表格\n _insert: function _insert(rowNum, colNum) {\n // 拼接 table 模板\n var r = void 0,\n c = void 0;\n var html = '<table border=\"0\" width=\"100%\" cellpadding=\"0\" cellspacing=\"0\">';\n for (r = 0; r < rowNum; r++) {\n html += '<tr>';\n if (r === 0) {\n for (c = 0; c < colNum; c++) {\n html += '<th> </th>';\n }\n } else {\n for (c = 0; c < colNum; c++) {\n html += '<td> </td>';\n }\n }\n html += '</tr>';\n }\n html += '</table><p><br></p>';\n\n // 执行命令\n var editor = this.editor;\n editor.cmd.do('insertHTML', html);\n\n // 防止 firefox 下出现 resize 的控制点\n editor.cmd.do('enableObjectResizing', false);\n editor.cmd.do('enableInlineTableEditing', false);\n },\n\n // 创建编辑表格的 panel\n _createEditPanel: function _createEditPanel() {\n var _this2 = this;\n\n // 可用的 id\n var addRowBtnId = getRandom('add-row');\n var addColBtnId = getRandom('add-col');\n var delRowBtnId = getRandom('del-row');\n var delColBtnId = getRandom('del-col');\n var delTableBtnId = getRandom('del-table');\n\n // 创建 panel 对象\n var panel = new Panel(this, {\n width: 320,\n // panel 包含多个 tab\n tabs: [{\n // 标题\n title: '编辑表格',\n // 模板\n tpl: '<div>\\n <div class=\"w-e-button-container\" style=\"border-bottom:1px solid #f1f1f1;padding-bottom:5px;margin-bottom:5px;\">\\n <button id=\"' + addRowBtnId + '\" class=\"left\">\\u589E\\u52A0\\u884C</button>\\n <button id=\"' + delRowBtnId + '\" class=\"red left\">\\u5220\\u9664\\u884C</button>\\n <button id=\"' + addColBtnId + '\" class=\"left\">\\u589E\\u52A0\\u5217</button>\\n <button id=\"' + delColBtnId + '\" class=\"red left\">\\u5220\\u9664\\u5217</button>\\n </div>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + delTableBtnId + '\" class=\"gray left\">\\u5220\\u9664\\u8868\\u683C</button>\\n </dv>\\n </div>',\n // 事件绑定\n events: [{\n // 增加行\n selector: '#' + addRowBtnId,\n type: 'click',\n fn: function fn() {\n _this2._addRow();\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n // 增加列\n selector: '#' + addColBtnId,\n type: 'click',\n fn: function fn() {\n _this2._addCol();\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n // 删除行\n selector: '#' + delRowBtnId,\n type: 'click',\n fn: function fn() {\n _this2._delRow();\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n // 删除列\n selector: '#' + delColBtnId,\n type: 'click',\n fn: function fn() {\n _this2._delCol();\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n // 删除表格\n selector: '#' + delTableBtnId,\n type: 'click',\n fn: function fn() {\n _this2._delTable();\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n }]\n });\n // 显示 panel\n panel.show();\n },\n\n // 获取选中的单元格的位置信息\n _getLocationData: function _getLocationData() {\n var result = {};\n var editor = this.editor;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n var nodeName = $selectionELem.getNodeName();\n if (nodeName !== 'TD' && nodeName !== 'TH') {\n return;\n }\n\n // 获取 td index\n var $tr = $selectionELem.parent();\n var $tds = $tr.children();\n var tdLength = $tds.length;\n $tds.forEach(function (td, index) {\n if (td === $selectionELem[0]) {\n // 记录并跳出循环\n result.td = {\n index: index,\n elem: td,\n length: tdLength\n };\n return false;\n }\n });\n\n // 获取 tr index\n var $tbody = $tr.parent();\n var $trs = $tbody.children();\n var trLength = $trs.length;\n $trs.forEach(function (tr, index) {\n if (tr === $tr[0]) {\n // 记录并跳出循环\n result.tr = {\n index: index,\n elem: tr,\n length: trLength\n };\n return false;\n }\n });\n\n // 返回结果\n return result;\n },\n\n // 增加行\n _addRow: function _addRow() {\n // 获取当前单元格的位置信息\n var locationData = this._getLocationData();\n if (!locationData) {\n return;\n }\n var trData = locationData.tr;\n var $currentTr = $(trData.elem);\n var tdData = locationData.td;\n var tdLength = tdData.length;\n\n // 拼接即将插入的字符串\n var newTr = document.createElement('tr');\n var tpl = '',\n i = void 0;\n for (i = 0; i < tdLength; i++) {\n tpl += '<td> </td>';\n }\n newTr.innerHTML = tpl;\n // 插入\n $(newTr).insertAfter($currentTr);\n },\n\n // 增加列\n _addCol: function _addCol() {\n // 获取当前单元格的位置信息\n var locationData = this._getLocationData();\n if (!locationData) {\n return;\n }\n var trData = locationData.tr;\n var tdData = locationData.td;\n var tdIndex = tdData.index;\n var $currentTr = $(trData.elem);\n var $trParent = $currentTr.parent();\n var $trs = $trParent.children();\n\n // 遍历所有行\n $trs.forEach(function (tr) {\n var $tr = $(tr);\n var $tds = $tr.children();\n var $currentTd = $tds.get(tdIndex);\n var name = $currentTd.getNodeName().toLowerCase();\n\n // new 一个 td,并插入\n var newTd = document.createElement(name);\n $(newTd).insertAfter($currentTd);\n });\n },\n\n // 删除行\n _delRow: function _delRow() {\n // 获取当前单元格的位置信息\n var locationData = this._getLocationData();\n if (!locationData) {\n return;\n }\n var trData = locationData.tr;\n var $currentTr = $(trData.elem);\n $currentTr.remove();\n },\n\n // 删除列\n _delCol: function _delCol() {\n // 获取当前单元格的位置信息\n var locationData = this._getLocationData();\n if (!locationData) {\n return;\n }\n var trData = locationData.tr;\n var tdData = locationData.td;\n var tdIndex = tdData.index;\n var $currentTr = $(trData.elem);\n var $trParent = $currentTr.parent();\n var $trs = $trParent.children();\n\n // 遍历所有行\n $trs.forEach(function (tr) {\n var $tr = $(tr);\n var $tds = $tr.children();\n var $currentTd = $tds.get(tdIndex);\n // 删除\n $currentTd.remove();\n });\n },\n\n // 删除表格\n _delTable: function _delTable() {\n var editor = this.editor;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n var $table = $selectionELem.parentUntil('table');\n if (!$table) {\n return;\n }\n $table.remove();\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n var $selectionELem = editor.selection.getSelectionContainerElem();\n if (!$selectionELem) {\n return;\n }\n var nodeName = $selectionELem.getNodeName();\n if (nodeName === 'TD' || nodeName === 'TH') {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n menu - video\n*/\n// 构造函数\nfunction Video(editor) {\n this.editor = editor;\n this.$elem = $('<div class=\"w-e-menu\"><i class=\"w-e-icon-play\"></i></div>');\n this.type = 'panel';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nVideo.prototype = {\n constructor: Video,\n\n onClick: function onClick() {\n this._createPanel();\n },\n\n _createPanel: function _createPanel() {\n var _this = this;\n\n // 创建 id\n var textValId = getRandom('text-val');\n var btnId = getRandom('btn');\n\n // 创建 panel\n var panel = new Panel(this, {\n width: 350,\n // 一个 panel 多个 tab\n tabs: [{\n // 标题\n title: '插入视频',\n // 模板\n tpl: '<div>\\n <input id=\"' + textValId + '\" type=\"text\" class=\"block\" placeholder=\"\\u683C\\u5F0F\\u5982\\uFF1A<iframe src=... ></iframe>\"/>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + btnId + '\" class=\"right\">\\u63D2\\u5165</button>\\n </div>\\n </div>',\n // 事件绑定\n events: [{\n selector: '#' + btnId,\n type: 'click',\n fn: function fn() {\n var $text = $('#' + textValId);\n var val = $text.val().trim();\n\n // 测试用视频地址\n // <iframe height=498 width=510 src='http://player.youku.com/embed/XMjcwMzc3MzM3Mg==' frameborder=0 'allowfullscreen'></iframe>\n\n if (val) {\n // 插入视频\n _this._insert(val);\n }\n\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n } // first tab end\n ] // tabs end\n }); // panel end\n\n // 显示 panel\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n // 插入视频\n _insert: function _insert(val) {\n var editor = this.editor;\n editor.cmd.do('insertHTML', val + '<p><br></p>');\n }\n};\n\n/*\n menu - img\n*/\n// 构造函数\nfunction Image(editor) {\n this.editor = editor;\n var imgMenuId = getRandom('w-e-img');\n this.$elem = $('<div class=\"w-e-menu\" id=\"' + imgMenuId + '\"><i class=\"w-e-icon-image\"></i></div>');\n editor.imgMenuId = imgMenuId;\n this.type = 'panel';\n\n // 当前是否 active 状态\n this._active = false;\n}\n\n// 原型\nImage.prototype = {\n constructor: Image,\n\n onClick: function onClick() {\n var editor = this.editor;\n var config = editor.config;\n if (config.qiniu) {\n return;\n }\n if (this._active) {\n this._createEditPanel();\n } else {\n this._createInsertPanel();\n }\n },\n\n _createEditPanel: function _createEditPanel() {\n var editor = this.editor;\n\n // id\n var width30 = getRandom('width-30');\n var width50 = getRandom('width-50');\n var width100 = getRandom('width-100');\n var delBtn = getRandom('del-btn');\n\n // tab 配置\n var tabsConfig = [{\n title: '编辑图片',\n tpl: '<div>\\n <div class=\"w-e-button-container\" style=\"border-bottom:1px solid #f1f1f1;padding-bottom:5px;margin-bottom:5px;\">\\n <span style=\"float:left;font-size:14px;margin:4px 5px 0 5px;color:#333;\">\\u6700\\u5927\\u5BBD\\u5EA6\\uFF1A</span>\\n <button id=\"' + width30 + '\" class=\"left\">30%</button>\\n <button id=\"' + width50 + '\" class=\"left\">50%</button>\\n <button id=\"' + width100 + '\" class=\"left\">100%</button>\\n </div>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + delBtn + '\" class=\"gray left\">\\u5220\\u9664\\u56FE\\u7247</button>\\n </dv>\\n </div>',\n events: [{\n selector: '#' + width30,\n type: 'click',\n fn: function fn() {\n var $img = editor._selectedImg;\n if ($img) {\n $img.css('max-width', '30%');\n }\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n selector: '#' + width50,\n type: 'click',\n fn: function fn() {\n var $img = editor._selectedImg;\n if ($img) {\n $img.css('max-width', '50%');\n }\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n selector: '#' + width100,\n type: 'click',\n fn: function fn() {\n var $img = editor._selectedImg;\n if ($img) {\n $img.css('max-width', '100%');\n }\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }, {\n selector: '#' + delBtn,\n type: 'click',\n fn: function fn() {\n var $img = editor._selectedImg;\n if ($img) {\n $img.remove();\n }\n // 返回 true,表示该事件执行完之后,panel 要关闭。否则 panel 不会关闭\n return true;\n }\n }]\n }];\n\n // 创建 panel 并显示\n var panel = new Panel(this, {\n width: 300,\n tabs: tabsConfig\n });\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n _createInsertPanel: function _createInsertPanel() {\n var editor = this.editor;\n var uploadImg = editor.uploadImg;\n var config = editor.config;\n\n // id\n var upTriggerId = getRandom('up-trigger');\n var upFileId = getRandom('up-file');\n var linkUrlId = getRandom('link-url');\n var linkBtnId = getRandom('link-btn');\n\n // tabs 的配置\n var tabsConfig = [{\n title: '上传图片',\n tpl: '<div class=\"w-e-up-img-container\">\\n <div id=\"' + upTriggerId + '\" class=\"w-e-up-btn\">\\n <i class=\"w-e-icon-upload2\"></i>\\n </div>\\n <div style=\"display:none;\">\\n <input id=\"' + upFileId + '\" type=\"file\" multiple=\"multiple\" accept=\"image/jpg,image/jpeg,image/png,image/gif,image/bmp\"/>\\n </div>\\n </div>',\n events: [{\n // 触发选择图片\n selector: '#' + upTriggerId,\n type: 'click',\n fn: function fn() {\n var $file = $('#' + upFileId);\n var fileElem = $file[0];\n if (fileElem) {\n fileElem.click();\n } else {\n // 返回 true 可关闭 panel\n return true;\n }\n }\n }, {\n // 选择图片完毕\n selector: '#' + upFileId,\n type: 'change',\n fn: function fn() {\n var $file = $('#' + upFileId);\n var fileElem = $file[0];\n if (!fileElem) {\n // 返回 true 可关闭 panel\n return true;\n }\n\n // 获取选中的 file 对象列表\n var fileList = fileElem.files;\n if (fileList.length) {\n uploadImg.uploadImg(fileList);\n }\n\n // 返回 true 可关闭 panel\n return true;\n }\n }]\n }, // first tab end\n {\n title: '网络图片',\n tpl: '<div>\\n <input id=\"' + linkUrlId + '\" type=\"text\" class=\"block\" placeholder=\"\\u56FE\\u7247\\u94FE\\u63A5\"/></td>\\n <div class=\"w-e-button-container\">\\n <button id=\"' + linkBtnId + '\" class=\"right\">\\u63D2\\u5165</button>\\n </div>\\n </div>',\n events: [{\n selector: '#' + linkBtnId,\n type: 'click',\n fn: function fn() {\n var $linkUrl = $('#' + linkUrlId);\n var url = $linkUrl.val().trim();\n\n if (url) {\n uploadImg.insertLinkImg(url);\n }\n\n // 返回 true 表示函数执行结束之后关闭 panel\n return true;\n }\n }]\n } // second tab end\n ]; // tabs end\n\n // 判断 tabs 的显示\n var tabsConfigResult = [];\n if ((config.uploadImgShowBase64 || config.uploadImgServer || config.customUploadImg) && window.FileReader) {\n // 显示“上传图片”\n tabsConfigResult.push(tabsConfig[0]);\n }\n if (config.showLinkImg) {\n // 显示“网络图片”\n tabsConfigResult.push(tabsConfig[1]);\n }\n\n // 创建 panel 并显示\n var panel = new Panel(this, {\n width: 300,\n tabs: tabsConfigResult\n });\n panel.show();\n\n // 记录属性\n this.panel = panel;\n },\n\n // 试图改变 active 状态\n tryChangeActive: function tryChangeActive(e) {\n var editor = this.editor;\n var $elem = this.$elem;\n if (editor._selectedImg) {\n this._active = true;\n $elem.addClass('w-e-active');\n } else {\n this._active = false;\n $elem.removeClass('w-e-active');\n }\n }\n};\n\n/*\n 所有菜单的汇总\n*/\n\n// 存储菜单的构造函数\nvar MenuConstructors = {};\n\nMenuConstructors.bold = Bold;\n\nMenuConstructors.head = Head;\n\nMenuConstructors.fontSize = FontSize;\n\nMenuConstructors.fontName = FontName;\n\nMenuConstructors.link = Link;\n\nMenuConstructors.italic = Italic;\n\nMenuConstructors.redo = Redo;\n\nMenuConstructors.strikeThrough = StrikeThrough;\n\nMenuConstructors.underline = Underline;\n\nMenuConstructors.undo = Undo;\n\nMenuConstructors.list = List;\n\nMenuConstructors.justify = Justify;\n\nMenuConstructors.foreColor = ForeColor;\n\nMenuConstructors.backColor = BackColor;\n\nMenuConstructors.quote = Quote;\n\nMenuConstructors.code = Code;\n\nMenuConstructors.emoticon = Emoticon;\n\nMenuConstructors.table = Table;\n\nMenuConstructors.video = Video;\n\nMenuConstructors.image = Image;\n\n/*\n 菜单集合\n*/\n// 构造函数\nfunction Menus(editor) {\n this.editor = editor;\n this.menus = {};\n}\n\n// 修改原型\nMenus.prototype = {\n constructor: Menus,\n\n // 初始化菜单\n init: function init() {\n var _this = this;\n\n var editor = this.editor;\n var config = editor.config || {};\n var configMenus = config.menus || []; // 获取配置中的菜单\n\n // 根据配置信息,创建菜单\n configMenus.forEach(function (menuKey) {\n var MenuConstructor = MenuConstructors[menuKey];\n if (MenuConstructor && typeof MenuConstructor === 'function') {\n // 创建单个菜单\n _this.menus[menuKey] = new MenuConstructor(editor);\n }\n });\n\n // 添加到菜单栏\n this._addToToolbar();\n\n // 绑定事件\n this._bindEvent();\n },\n\n // 添加到菜单栏\n _addToToolbar: function _addToToolbar() {\n var editor = this.editor;\n var $toolbarElem = editor.$toolbarElem;\n var menus = this.menus;\n var config = editor.config;\n // config.zIndex 是配置的编辑区域的 z-index,菜单的 z-index 得在其基础上 +1\n var zIndex = config.zIndex + 1;\n objForEach(menus, function (key, menu) {\n var $elem = menu.$elem;\n if ($elem) {\n // 设置 z-index\n $elem.css('z-index', zIndex);\n $toolbarElem.append($elem);\n }\n });\n },\n\n // 绑定菜单 click mouseenter 事件\n _bindEvent: function _bindEvent() {\n var menus = this.menus;\n var editor = this.editor;\n objForEach(menus, function (key, menu) {\n var type = menu.type;\n if (!type) {\n return;\n }\n var $elem = menu.$elem;\n var droplist = menu.droplist;\n var panel = menu.panel;\n\n // 点击类型,例如 bold\n if (type === 'click' && menu.onClick) {\n $elem.on('click', function (e) {\n if (editor.selection.getRange() == null) {\n return;\n }\n menu.onClick(e);\n });\n }\n\n // 下拉框,例如 head\n if (type === 'droplist' && droplist) {\n $elem.on('mouseenter', function (e) {\n if (editor.selection.getRange() == null) {\n return;\n }\n // 显示\n droplist.showTimeoutId = setTimeout(function () {\n droplist.show();\n }, 200);\n }).on('mouseleave', function (e) {\n // 隐藏\n droplist.hideTimeoutId = setTimeout(function () {\n droplist.hide();\n }, 0);\n });\n }\n\n // 弹框类型,例如 link\n if (type === 'panel' && menu.onClick) {\n $elem.on('click', function (e) {\n e.stopPropagation();\n if (editor.selection.getRange() == null) {\n return;\n }\n // 在自定义事件中显示 panel\n menu.onClick(e);\n });\n }\n });\n },\n\n // 尝试修改菜单状态\n changeActive: function changeActive() {\n var menus = this.menus;\n objForEach(menus, function (key, menu) {\n if (menu.tryChangeActive) {\n setTimeout(function () {\n menu.tryChangeActive();\n }, 100);\n }\n });\n }\n};\n\n/*\n 粘贴信息的处理\n*/\n\n// 获取粘贴的纯文本\nfunction getPasteText(e) {\n var clipboardData = e.clipboardData || e.originalEvent && e.originalEvent.clipboardData;\n var pasteText = void 0;\n if (clipboardData == null) {\n pasteText = window.clipboardData && window.clipboardData.getData('text');\n } else {\n pasteText = clipboardData.getData('text/plain');\n }\n\n return replaceHtmlSymbol(pasteText);\n}\n\n// 获取粘贴的html\nfunction getPasteHtml(e, filterStyle, ignoreImg) {\n var clipboardData = e.clipboardData || e.originalEvent && e.originalEvent.clipboardData;\n var pasteText = void 0,\n pasteHtml = void 0;\n if (clipboardData == null) {\n pasteText = window.clipboardData && window.clipboardData.getData('text');\n } else {\n pasteText = clipboardData.getData('text/plain');\n pasteHtml = clipboardData.getData('text/html');\n }\n if (!pasteHtml && pasteText) {\n pasteHtml = '<p>' + replaceHtmlSymbol(pasteText) + '</p>';\n }\n if (!pasteHtml) {\n return;\n }\n\n // 过滤word中状态过来的无用字符\n var docSplitHtml = pasteHtml.split('</html>');\n if (docSplitHtml.length === 2) {\n pasteHtml = docSplitHtml[0];\n }\n\n // 过滤无用标签\n pasteHtml = pasteHtml.replace(/<(meta|script|link).+?>/igm, '');\n // 去掉注释\n pasteHtml = pasteHtml.replace(/<!--.*?-->/mg, '');\n // 过滤 data-xxx 属性\n pasteHtml = pasteHtml.replace(/\\s?data-.+?=('|\").+?('|\")/igm, '');\n\n if (ignoreImg) {\n // 忽略图片\n pasteHtml = pasteHtml.replace(/<img.+?>/igm, '');\n }\n\n if (filterStyle) {\n // 过滤样式\n pasteHtml = pasteHtml.replace(/\\s?(class|style)=('|\").*?('|\")/igm, '');\n } else {\n // 保留样式\n pasteHtml = pasteHtml.replace(/\\s?class=('|\").*?('|\")/igm, '');\n }\n\n return pasteHtml;\n}\n\n// 获取粘贴的图片文件\nfunction getPasteImgs(e) {\n var result = [];\n var txt = getPasteText(e);\n if (txt) {\n // 有文字,就忽略图片\n return result;\n }\n\n var clipboardData = e.clipboardData || e.originalEvent && e.originalEvent.clipboardData || {};\n var items = clipboardData.items;\n if (!items) {\n return result;\n }\n\n objForEach(items, function (key, value) {\n var type = value.type;\n if (/image/i.test(type)) {\n result.push(value.getAsFile());\n }\n });\n\n return result;\n}\n\n/*\n 编辑区域\n*/\n\n// 获取一个 elem.childNodes 的 JSON 数据\nfunction getChildrenJSON($elem) {\n var result = [];\n var $children = $elem.childNodes() || []; // 注意 childNodes() 可以获取文本节点\n $children.forEach(function (curElem) {\n var elemResult = void 0;\n var nodeType = curElem.nodeType;\n\n // 文本节点\n if (nodeType === 3) {\n elemResult = curElem.textContent;\n elemResult = replaceHtmlSymbol(elemResult);\n }\n\n // 普通 DOM 节点\n if (nodeType === 1) {\n elemResult = {};\n\n // tag\n elemResult.tag = curElem.nodeName.toLowerCase();\n // attr\n var attrData = [];\n var attrList = curElem.attributes || {};\n var attrListLength = attrList.length || 0;\n for (var i = 0; i < attrListLength; i++) {\n var attr = attrList[i];\n attrData.push({\n name: attr.name,\n value: attr.value\n });\n }\n elemResult.attrs = attrData;\n // children(递归)\n elemResult.children = getChildrenJSON($(curElem));\n }\n\n result.push(elemResult);\n });\n return result;\n}\n\n// 构造函数\nfunction Text(editor) {\n this.editor = editor;\n}\n\n// 修改原型\nText.prototype = {\n constructor: Text,\n\n // 初始化\n init: function init() {\n // 绑定事件\n this._bindEvent();\n },\n\n // 清空内容\n clear: function clear() {\n this.html('<p><br></p>');\n },\n\n // 获取 设置 html\n html: function html(val) {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n var html = void 0;\n if (val == null) {\n html = $textElem.html();\n // 未选中任何内容的时候点击“加粗”或者“斜体”等按钮,就得需要一个空的占位符 ​ ,这里替换掉\n html = html.replace(/\\u200b/gm, '');\n return html;\n } else {\n $textElem.html(val);\n\n // 初始化选取,将光标定位到内容尾部\n editor.initSelection();\n }\n },\n\n // 获取 JSON\n getJSON: function getJSON() {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n return getChildrenJSON($textElem);\n },\n\n // 获取 设置 text\n text: function text(val) {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n var text = void 0;\n if (val == null) {\n text = $textElem.text();\n // 未选中任何内容的时候点击“加粗”或者“斜体”等按钮,就得需要一个空的占位符 ​ ,这里替换掉\n text = text.replace(/\\u200b/gm, '');\n return text;\n } else {\n $textElem.text('<p>' + val + '</p>');\n\n // 初始化选取,将光标定位到内容尾部\n editor.initSelection();\n }\n },\n\n // 追加内容\n append: function append(html) {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n $textElem.append($(html));\n\n // 初始化选取,将光标定位到内容尾部\n editor.initSelection();\n },\n\n // 绑定事件\n _bindEvent: function _bindEvent() {\n // 实时保存选取\n this._saveRangeRealTime();\n\n // 按回车建时的特殊处理\n this._enterKeyHandle();\n\n // 清空时保留 <p><br></p>\n this._clearHandle();\n\n // 粘贴事件(粘贴文字,粘贴图片)\n this._pasteHandle();\n\n // tab 特殊处理\n this._tabHandle();\n\n // img 点击\n this._imgHandle();\n\n // 拖拽事件\n this._dragHandle();\n },\n\n // 实时保存选取\n _saveRangeRealTime: function _saveRangeRealTime() {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n\n // 保存当前的选区\n function saveRange(e) {\n // 随时保存选区\n editor.selection.saveRange();\n // 更新按钮 ative 状态\n editor.menus.changeActive();\n }\n // 按键后保存\n $textElem.on('keyup', saveRange);\n $textElem.on('mousedown', function (e) {\n // mousedown 状态下,鼠标滑动到编辑区域外面,也需要保存选区\n $textElem.on('mouseleave', saveRange);\n });\n $textElem.on('mouseup', function (e) {\n saveRange();\n // 在编辑器区域之内完成点击,取消鼠标滑动到编辑区外面的事件\n $textElem.off('mouseleave', saveRange);\n });\n },\n\n // 按回车键时的特殊处理\n _enterKeyHandle: function _enterKeyHandle() {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n\n function insertEmptyP($selectionElem) {\n var $p = $('<p><br></p>');\n $p.insertBefore($selectionElem);\n editor.selection.createRangeByElem($p, true);\n editor.selection.restoreSelection();\n $selectionElem.remove();\n }\n\n // 将回车之后生成的非 <p> 的顶级标签,改为 <p>\n function pHandle(e) {\n var $selectionElem = editor.selection.getSelectionContainerElem();\n var $parentElem = $selectionElem.parent();\n\n if ($parentElem.html() === '<code><br></code>') {\n // 回车之前光标所在一个 <p><code>.....</code></p> ,忽然回车生成一个空的 <p><code><br></code></p>\n // 而且继续回车跳不出去,因此只能特殊处理\n insertEmptyP($selectionElem);\n return;\n }\n\n if (!$parentElem.equal($textElem)) {\n // 不是顶级标签\n return;\n }\n\n var nodeName = $selectionElem.getNodeName();\n if (nodeName === 'P') {\n // 当前的标签是 P ,不用做处理\n return;\n }\n\n if ($selectionElem.text()) {\n // 有内容,不做处理\n return;\n }\n\n // 插入 <p> ,并将选取定位到 <p>,删除当前标签\n insertEmptyP($selectionElem);\n }\n\n $textElem.on('keyup', function (e) {\n if (e.keyCode !== 13) {\n // 不是回车键\n return;\n }\n // 将回车之后生成的非 <p> 的顶级标签,改为 <p>\n pHandle(e);\n });\n\n // <pre><code></code></pre> 回车时 特殊处理\n function codeHandle(e) {\n var $selectionElem = editor.selection.getSelectionContainerElem();\n if (!$selectionElem) {\n return;\n }\n var $parentElem = $selectionElem.parent();\n var selectionNodeName = $selectionElem.getNodeName();\n var parentNodeName = $parentElem.getNodeName();\n\n if (selectionNodeName !== 'CODE' || parentNodeName !== 'PRE') {\n // 不符合要求 忽略\n return;\n }\n\n if (!editor.cmd.queryCommandSupported('insertHTML')) {\n // 必须原生支持 insertHTML 命令\n return;\n }\n\n // 处理:光标定位到代码末尾,联系点击两次回车,即跳出代码块\n if (editor._willBreakCode === true) {\n // 此时可以跳出代码块\n // 插入 <p> ,并将选取定位到 <p>\n var $p = $('<p><br></p>');\n $p.insertAfter($parentElem);\n editor.selection.createRangeByElem($p, true);\n editor.selection.restoreSelection();\n\n // 修改状态\n editor._willBreakCode = false;\n\n e.preventDefault();\n return;\n }\n\n var _startOffset = editor.selection.getRange().startOffset;\n\n // 处理:回车时,不能插入 <br> 而是插入 \\n ,因为是在 pre 标签里面\n editor.cmd.do('insertHTML', '\\n');\n editor.selection.saveRange();\n if (editor.selection.getRange().startOffset === _startOffset) {\n // 没起作用,再来一遍\n editor.cmd.do('insertHTML', '\\n');\n }\n\n var codeLength = $selectionElem.html().length;\n if (editor.selection.getRange().startOffset + 1 === codeLength) {\n // 说明光标在代码最后的位置,执行了回车操作\n // 记录下来,以便下次回车时候跳出 code\n editor._willBreakCode = true;\n }\n\n // 阻止默认行为\n e.preventDefault();\n }\n\n $textElem.on('keydown', function (e) {\n if (e.keyCode !== 13) {\n // 不是回车键\n // 取消即将跳转代码块的记录\n editor._willBreakCode = false;\n return;\n }\n // <pre><code></code></pre> 回车时 特殊处理\n codeHandle(e);\n });\n },\n\n // 清空时保留 <p><br></p>\n _clearHandle: function _clearHandle() {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n\n $textElem.on('keydown', function (e) {\n if (e.keyCode !== 8) {\n return;\n }\n var txtHtml = $textElem.html().toLowerCase().trim();\n if (txtHtml === '<p><br></p>') {\n // 最后剩下一个空行,就不再删除了\n e.preventDefault();\n return;\n }\n });\n\n $textElem.on('keyup', function (e) {\n if (e.keyCode !== 8) {\n return;\n }\n var $p = void 0;\n var txtHtml = $textElem.html().toLowerCase().trim();\n\n // firefox 时用 txtHtml === '<br>' 判断,其他用 !txtHtml 判断\n if (!txtHtml || txtHtml === '<br>') {\n // 内容空了\n $p = $('<p><br/></p>');\n $textElem.html(''); // 一定要先清空,否则在 firefox 下有问题\n $textElem.append($p);\n editor.selection.createRangeByElem($p, false, true);\n editor.selection.restoreSelection();\n }\n });\n },\n\n // 粘贴事件(粘贴文字 粘贴图片)\n _pasteHandle: function _pasteHandle() {\n var editor = this.editor;\n var config = editor.config;\n var pasteFilterStyle = config.pasteFilterStyle;\n var pasteTextHandle = config.pasteTextHandle;\n var ignoreImg = config.pasteIgnoreImg;\n var $textElem = editor.$textElem;\n\n // 粘贴图片、文本的事件,每次只能执行一个\n // 判断该次粘贴事件是否可以执行\n var pasteTime = 0;\n function canDo() {\n var now = Date.now();\n var flag = false;\n if (now - pasteTime >= 100) {\n // 间隔大于 100 ms ,可以执行\n flag = true;\n }\n pasteTime = now;\n return flag;\n }\n function resetTime() {\n pasteTime = 0;\n }\n\n // 粘贴文字\n $textElem.on('paste', function (e) {\n if (UA.isIE()) {\n return;\n } else {\n // 阻止默认行为,使用 execCommand 的粘贴命令\n e.preventDefault();\n }\n\n // 粘贴图片和文本,只能同时使用一个\n if (!canDo()) {\n return;\n }\n\n // 获取粘贴的文字\n var pasteHtml = getPasteHtml(e, pasteFilterStyle, ignoreImg);\n var pasteText = getPasteText(e);\n pasteText = pasteText.replace(/\\n/gm, '<br>');\n\n var $selectionElem = editor.selection.getSelectionContainerElem();\n if (!$selectionElem) {\n return;\n }\n var nodeName = $selectionElem.getNodeName();\n\n // code 中只能粘贴纯文本\n if (nodeName === 'CODE' || nodeName === 'PRE') {\n if (pasteTextHandle && isFunction(pasteTextHandle)) {\n // 用户自定义过滤处理粘贴内容\n pasteText = '' + (pasteTextHandle(pasteText) || '');\n }\n editor.cmd.do('insertHTML', '<p>' + pasteText + '</p>');\n return;\n }\n\n // 先放开注释,有问题再追查 ————\n // // 表格中忽略,可能会出现异常问题\n // if (nodeName === 'TD' || nodeName === 'TH') {\n // return\n // }\n\n if (!pasteHtml) {\n // 没有内容,可继续执行下面的图片粘贴\n resetTime();\n return;\n }\n try {\n // firefox 中,获取的 pasteHtml 可能是没有 <ul> 包裹的 <li>\n // 因此执行 insertHTML 会报错\n if (pasteTextHandle && isFunction(pasteTextHandle)) {\n // 用户自定义过滤处理粘贴内容\n pasteHtml = '' + (pasteTextHandle(pasteHtml) || '');\n }\n editor.cmd.do('insertHTML', pasteHtml);\n } catch (ex) {\n // 此时使用 pasteText 来兼容一下\n if (pasteTextHandle && isFunction(pasteTextHandle)) {\n // 用户自定义过滤处理粘贴内容\n pasteText = '' + (pasteTextHandle(pasteText) || '');\n }\n editor.cmd.do('insertHTML', '<p>' + pasteText + '</p>');\n }\n });\n\n // 粘贴图片\n $textElem.on('paste', function (e) {\n if (UA.isIE()) {\n return;\n } else {\n e.preventDefault();\n }\n\n // 粘贴图片和文本,只能同时使用一个\n if (!canDo()) {\n return;\n }\n\n // 获取粘贴的图片\n var pasteFiles = getPasteImgs(e);\n if (!pasteFiles || !pasteFiles.length) {\n return;\n }\n\n // 获取当前的元素\n var $selectionElem = editor.selection.getSelectionContainerElem();\n if (!$selectionElem) {\n return;\n }\n var nodeName = $selectionElem.getNodeName();\n\n // code 中粘贴忽略\n if (nodeName === 'CODE' || nodeName === 'PRE') {\n return;\n }\n\n // 上传图片\n var uploadImg = editor.uploadImg;\n uploadImg.uploadImg(pasteFiles);\n });\n },\n\n // tab 特殊处理\n _tabHandle: function _tabHandle() {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n\n $textElem.on('keydown', function (e) {\n if (e.keyCode !== 9) {\n return;\n }\n if (!editor.cmd.queryCommandSupported('insertHTML')) {\n // 必须原生支持 insertHTML 命令\n return;\n }\n var $selectionElem = editor.selection.getSelectionContainerElem();\n if (!$selectionElem) {\n return;\n }\n var $parentElem = $selectionElem.parent();\n var selectionNodeName = $selectionElem.getNodeName();\n var parentNodeName = $parentElem.getNodeName();\n\n if (selectionNodeName === 'CODE' && parentNodeName === 'PRE') {\n // <pre><code> 里面\n editor.cmd.do('insertHTML', ' ');\n } else {\n // 普通文字\n editor.cmd.do('insertHTML', ' ');\n }\n\n e.preventDefault();\n });\n },\n\n // img 点击\n _imgHandle: function _imgHandle() {\n var editor = this.editor;\n var $textElem = editor.$textElem;\n\n // 为图片增加 selected 样式\n $textElem.on('click', 'img', function (e) {\n var img = this;\n var $img = $(img);\n\n if ($img.attr('data-w-e') === '1') {\n // 是表情图片,忽略\n return;\n }\n\n // 记录当前点击过的图片\n editor._selectedImg = $img;\n\n // 修改选区并 restore ,防止用户此时点击退格键,会删除其他内容\n editor.selection.createRangeByElem($img);\n editor.selection.restoreSelection();\n });\n\n // 去掉图片的 selected 样式\n $textElem.on('click keyup', function (e) {\n if (e.target.matches('img')) {\n // 点击的是图片,忽略\n return;\n }\n // 删除记录\n editor._selectedImg = null;\n });\n },\n\n // 拖拽事件\n _dragHandle: function _dragHandle() {\n var editor = this.editor;\n\n // 禁用 document 拖拽事件\n var $document = $(document);\n $document.on('dragleave drop dragenter dragover', function (e) {\n e.preventDefault();\n });\n\n // 添加编辑区域拖拽事件\n var $textElem = editor.$textElem;\n $textElem.on('drop', function (e) {\n e.preventDefault();\n var files = e.dataTransfer && e.dataTransfer.files;\n if (!files || !files.length) {\n return;\n }\n\n // 上传图片\n var uploadImg = editor.uploadImg;\n uploadImg.uploadImg(files);\n });\n }\n};\n\n/*\n 命令,封装 document.execCommand\n*/\n\n// 构造函数\nfunction Command(editor) {\n this.editor = editor;\n}\n\n// 修改原型\nCommand.prototype = {\n constructor: Command,\n\n // 执行命令\n do: function _do(name, value) {\n var editor = this.editor;\n\n // 使用 styleWithCSS\n if (!editor._useStyleWithCSS) {\n document.execCommand('styleWithCSS', null, true);\n editor._useStyleWithCSS = true;\n }\n\n // 如果无选区,忽略\n if (!editor.selection.getRange()) {\n return;\n }\n\n // 恢复选取\n editor.selection.restoreSelection();\n\n // 执行\n var _name = '_' + name;\n if (this[_name]) {\n // 有自定义事件\n this[_name](value);\n } else {\n // 默认 command\n this._execCommand(name, value);\n }\n\n // 修改菜单状态\n editor.menus.changeActive();\n\n // 最后,恢复选取保证光标在原来的位置闪烁\n editor.selection.saveRange();\n editor.selection.restoreSelection();\n\n // 触发 onchange\n editor.change && editor.change();\n },\n\n // 自定义 insertHTML 事件\n _insertHTML: function _insertHTML(html) {\n var editor = this.editor;\n var range = editor.selection.getRange();\n\n if (this.queryCommandSupported('insertHTML')) {\n // W3C\n this._execCommand('insertHTML', html);\n } else if (range.insertNode) {\n // IE\n range.deleteContents();\n range.insertNode($(html)[0]);\n } else if (range.pasteHTML) {\n // IE <= 10\n range.pasteHTML(html);\n }\n },\n\n // 插入 elem\n _insertElem: function _insertElem($elem) {\n var editor = this.editor;\n var range = editor.selection.getRange();\n\n if (range.insertNode) {\n range.deleteContents();\n range.insertNode($elem[0]);\n }\n },\n\n // 封装 execCommand\n _execCommand: function _execCommand(name, value) {\n document.execCommand(name, false, value);\n },\n\n // 封装 document.queryCommandValue\n queryCommandValue: function queryCommandValue(name) {\n return document.queryCommandValue(name);\n },\n\n // 封装 document.queryCommandState\n queryCommandState: function queryCommandState(name) {\n return document.queryCommandState(name);\n },\n\n // 封装 document.queryCommandSupported\n queryCommandSupported: function queryCommandSupported(name) {\n return document.queryCommandSupported(name);\n }\n};\n\n/*\n selection range API\n*/\n\n// 构造函数\nfunction API(editor) {\n this.editor = editor;\n this._currentRange = null;\n}\n\n// 修改原型\nAPI.prototype = {\n constructor: API,\n\n // 获取 range 对象\n getRange: function getRange() {\n return this._currentRange;\n },\n\n // 保存选区\n saveRange: function saveRange(_range) {\n if (_range) {\n // 保存已有选区\n this._currentRange = _range;\n return;\n }\n\n // 获取当前的选区\n var selection = window.getSelection();\n if (selection.rangeCount === 0) {\n return;\n }\n var range = selection.getRangeAt(0);\n\n // 判断选区内容是否在编辑内容之内\n var $containerElem = this.getSelectionContainerElem(range);\n if (!$containerElem) {\n return;\n }\n\n // 判断选区内容是否在不可编辑区域之内\n if ($containerElem.attr('contenteditable') === 'false' || $containerElem.parentUntil('[contenteditable=false]')) {\n return;\n }\n\n var editor = this.editor;\n var $textElem = editor.$textElem;\n if ($textElem.isContain($containerElem)) {\n // 是编辑内容之内的\n this._currentRange = range;\n }\n },\n\n // 折叠选区\n collapseRange: function collapseRange(toStart) {\n if (toStart == null) {\n // 默认为 false\n toStart = false;\n }\n var range = this._currentRange;\n if (range) {\n range.collapse(toStart);\n }\n },\n\n // 选中区域的文字\n getSelectionText: function getSelectionText() {\n var range = this._currentRange;\n if (range) {\n return this._currentRange.toString();\n } else {\n return '';\n }\n },\n\n // 选区的 $Elem\n getSelectionContainerElem: function getSelectionContainerElem(range) {\n range = range || this._currentRange;\n var elem = void 0;\n if (range) {\n elem = range.commonAncestorContainer;\n return $(elem.nodeType === 1 ? elem : elem.parentNode);\n }\n },\n getSelectionStartElem: function getSelectionStartElem(range) {\n range = range || this._currentRange;\n var elem = void 0;\n if (range) {\n elem = range.startContainer;\n return $(elem.nodeType === 1 ? elem : elem.parentNode);\n }\n },\n getSelectionEndElem: function getSelectionEndElem(range) {\n range = range || this._currentRange;\n var elem = void 0;\n if (range) {\n elem = range.endContainer;\n return $(elem.nodeType === 1 ? elem : elem.parentNode);\n }\n },\n\n // 选区是否为空\n isSelectionEmpty: function isSelectionEmpty() {\n var range = this._currentRange;\n if (range && range.startContainer) {\n if (range.startContainer === range.endContainer) {\n if (range.startOffset === range.endOffset) {\n return true;\n }\n }\n }\n return false;\n },\n\n // 恢复选区\n restoreSelection: function restoreSelection() {\n var selection = window.getSelection();\n selection.removeAllRanges();\n selection.addRange(this._currentRange);\n },\n\n // 创建一个空白(即 ​ 字符)选区\n createEmptyRange: function createEmptyRange() {\n var editor = this.editor;\n var range = this.getRange();\n var $elem = void 0;\n\n if (!range) {\n // 当前无 range\n return;\n }\n if (!this.isSelectionEmpty()) {\n // 当前选区必须没有内容才可以\n return;\n }\n\n try {\n // 目前只支持 webkit 内核\n if (UA.isWebkit()) {\n // 插入 ​\n editor.cmd.do('insertHTML', '​');\n // 修改 offset 位置\n range.setEnd(range.endContainer, range.endOffset + 1);\n // 存储\n this.saveRange(range);\n } else {\n $elem = $('<strong>​</strong>');\n editor.cmd.do('insertElem', $elem);\n this.createRangeByElem($elem, true);\n }\n } catch (ex) {\n // 部分情况下会报错,兼容一下\n }\n },\n\n // 根据 $Elem 设置选区\n createRangeByElem: function createRangeByElem($elem, toStart, isContent) {\n // $elem - 经过封装的 elem\n // toStart - true 开始位置,false 结束位置\n // isContent - 是否选中Elem的内容\n if (!$elem.length) {\n return;\n }\n\n var elem = $elem[0];\n var range = document.createRange();\n\n if (isContent) {\n range.selectNodeContents(elem);\n } else {\n range.selectNode(elem);\n }\n\n if (typeof toStart === 'boolean') {\n range.collapse(toStart);\n }\n\n // 存储 range\n this.saveRange(range);\n }\n};\n\n/*\n 上传进度条\n*/\n\nfunction Progress(editor) {\n this.editor = editor;\n this._time = 0;\n this._isShow = false;\n this._isRender = false;\n this._timeoutId = 0;\n this.$textContainer = editor.$textContainerElem;\n this.$bar = $('<div class=\"w-e-progress\"></div>');\n}\n\nProgress.prototype = {\n constructor: Progress,\n\n show: function show(progress) {\n var _this = this;\n\n // 状态处理\n if (this._isShow) {\n return;\n }\n this._isShow = true;\n\n // 渲染\n var $bar = this.$bar;\n if (!this._isRender) {\n var $textContainer = this.$textContainer;\n $textContainer.append($bar);\n } else {\n this._isRender = true;\n }\n\n // 改变进度(节流,100ms 渲染一次)\n if (Date.now() - this._time > 100) {\n if (progress <= 1) {\n $bar.css('width', progress * 100 + '%');\n this._time = Date.now();\n }\n }\n\n // 隐藏\n var timeoutId = this._timeoutId;\n if (timeoutId) {\n clearTimeout(timeoutId);\n }\n timeoutId = setTimeout(function () {\n _this._hide();\n }, 500);\n },\n\n _hide: function _hide() {\n var $bar = this.$bar;\n $bar.remove();\n\n // 修改状态\n this._time = 0;\n this._isShow = false;\n this._isRender = false;\n }\n};\n\nvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) {\n return typeof obj;\n} : function (obj) {\n return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj;\n};\n\n/*\n 上传图片\n*/\n\n// 构造函数\nfunction UploadImg(editor) {\n this.editor = editor;\n}\n\n// 原型\nUploadImg.prototype = {\n constructor: UploadImg,\n\n // 根据 debug 弹出不同的信息\n _alert: function _alert(alertInfo, debugInfo) {\n var editor = this.editor;\n var debug = editor.config.debug;\n var customAlert = editor.config.customAlert;\n\n if (debug) {\n throw new Error('wangEditor: ' + (debugInfo || alertInfo));\n } else {\n if (customAlert && typeof customAlert === 'function') {\n customAlert(alertInfo);\n } else {\n alert(alertInfo);\n }\n }\n },\n\n // 根据链接插入图片\n insertLinkImg: function insertLinkImg(link) {\n var _this2 = this;\n\n if (!link) {\n return;\n }\n var editor = this.editor;\n var config = editor.config;\n\n // 校验格式\n var linkImgCheck = config.linkImgCheck;\n var checkResult = void 0;\n if (linkImgCheck && typeof linkImgCheck === 'function') {\n checkResult = linkImgCheck(link);\n if (typeof checkResult === 'string') {\n // 校验失败,提示信息\n alert(checkResult);\n return;\n }\n }\n\n editor.cmd.do('insertHTML', '<img src=\"' + link + '\" style=\"max-width:100%;\"/>');\n\n // 验证图片 url 是否有效,无效的话给出提示\n var img = document.createElement('img');\n img.onload = function () {\n var callback = config.linkImgCallback;\n if (callback && typeof callback === 'function') {\n callback(link);\n }\n\n img = null;\n };\n img.onerror = function () {\n img = null;\n // 无法成功下载图片\n _this2._alert('插入图片错误', 'wangEditor: \\u63D2\\u5165\\u56FE\\u7247\\u51FA\\u9519\\uFF0C\\u56FE\\u7247\\u94FE\\u63A5\\u662F \"' + link + '\"\\uFF0C\\u4E0B\\u8F7D\\u8BE5\\u94FE\\u63A5\\u5931\\u8D25');\n return;\n };\n img.onabort = function () {\n img = null;\n };\n img.src = link;\n },\n\n // 上传图片\n uploadImg: function uploadImg(files) {\n var _this3 = this;\n\n if (!files || !files.length) {\n return;\n }\n\n // ------------------------------ 获取配置信息 ------------------------------\n var editor = this.editor;\n var config = editor.config;\n var uploadImgServer = config.uploadImgServer;\n var uploadImgShowBase64 = config.uploadImgShowBase64;\n\n var maxSize = config.uploadImgMaxSize;\n var maxSizeM = maxSize / 1024 / 1024;\n var maxLength = config.uploadImgMaxLength || 10000;\n var uploadFileName = config.uploadFileName || '';\n var uploadImgParams = config.uploadImgParams || {};\n var uploadImgParamsWithUrl = config.uploadImgParamsWithUrl;\n var uploadImgHeaders = config.uploadImgHeaders || {};\n var hooks = config.uploadImgHooks || {};\n var timeout = config.uploadImgTimeout || 3000;\n var withCredentials = config.withCredentials;\n if (withCredentials == null) {\n withCredentials = false;\n }\n var customUploadImg = config.customUploadImg;\n\n if (!customUploadImg) {\n // 没有 customUploadImg 的情况下,需要如下两个配置才能继续进行图片上传\n if (!uploadImgServer && !uploadImgShowBase64) {\n return;\n }\n }\n\n // ------------------------------ 验证文件信息 ------------------------------\n var resultFiles = [];\n var errInfo = [];\n arrForEach(files, function (file) {\n var name = file.name;\n var size = file.size;\n\n // chrome 低版本 name === undefined\n if (!name || !size) {\n return;\n }\n\n if (/\\.(jpg|jpeg|png|bmp|gif|webp)$/i.test(name) === false) {\n // 后缀名不合法,不是图片\n errInfo.push('\\u3010' + name + '\\u3011\\u4E0D\\u662F\\u56FE\\u7247');\n return;\n }\n if (maxSize < size) {\n // 上传图片过大\n errInfo.push('\\u3010' + name + '\\u3011\\u5927\\u4E8E ' + maxSizeM + 'M');\n return;\n }\n\n // 验证通过的加入结果列表\n resultFiles.push(file);\n });\n // 抛出验证信息\n if (errInfo.length) {\n this._alert('图片验证未通过: \\n' + errInfo.join('\\n'));\n return;\n }\n if (resultFiles.length > maxLength) {\n this._alert('一次最多上传' + maxLength + '张图片');\n return;\n }\n\n // ------------------------------ 自定义上传 ------------------------------\n if (customUploadImg && typeof customUploadImg === 'function') {\n customUploadImg(resultFiles, this.insertLinkImg.bind(this));\n\n // 阻止以下代码执行\n return;\n }\n\n // 添加图片数据\n var formdata = new FormData();\n arrForEach(resultFiles, function (file) {\n var name = uploadFileName || file.name;\n formdata.append(name, file);\n });\n\n // ------------------------------ 上传图片 ------------------------------\n if (uploadImgServer && typeof uploadImgServer === 'string') {\n // 添加参数\n var uploadImgServerArr = uploadImgServer.split('#');\n uploadImgServer = uploadImgServerArr[0];\n var uploadImgServerHash = uploadImgServerArr[1] || '';\n objForEach(uploadImgParams, function (key, val) {\n // 因使用者反应,自定义参数不能默认 encode ,由 v3.1.1 版本开始注释掉\n // val = encodeURIComponent(val)\n\n // 第一,将参数拼接到 url 中\n if (uploadImgParamsWithUrl) {\n if (uploadImgServer.indexOf('?') > 0) {\n uploadImgServer += '&';\n } else {\n uploadImgServer += '?';\n }\n uploadImgServer = uploadImgServer + key + '=' + val;\n }\n\n // 第二,将参数添加到 formdata 中\n formdata.append(key, val);\n });\n if (uploadImgServerHash) {\n uploadImgServer += '#' + uploadImgServerHash;\n }\n\n // 定义 xhr\n var xhr = new XMLHttpRequest();\n xhr.open('POST', uploadImgServer);\n\n // 设置超时\n xhr.timeout = timeout;\n xhr.ontimeout = function () {\n // hook - timeout\n if (hooks.timeout && typeof hooks.timeout === 'function') {\n hooks.timeout(xhr, editor);\n }\n\n _this3._alert('上传图片超时');\n };\n\n // 监控 progress\n if (xhr.upload) {\n xhr.upload.onprogress = function (e) {\n var percent = void 0;\n // 进度条\n var progressBar = new Progress(editor);\n if (e.lengthComputable) {\n percent = e.loaded / e.total;\n progressBar.show(percent);\n }\n };\n }\n\n // 返回数据\n xhr.onreadystatechange = function () {\n var result = void 0;\n if (xhr.readyState === 4) {\n if (xhr.status < 200 || xhr.status >= 300) {\n // hook - error\n if (hooks.error && typeof hooks.error === 'function') {\n hooks.error(xhr, editor);\n }\n\n // xhr 返回状态错误\n _this3._alert('上传图片发生错误', '\\u4E0A\\u4F20\\u56FE\\u7247\\u53D1\\u751F\\u9519\\u8BEF\\uFF0C\\u670D\\u52A1\\u5668\\u8FD4\\u56DE\\u72B6\\u6001\\u662F ' + xhr.status);\n return;\n }\n\n result = xhr.responseText;\n if ((typeof result === 'undefined' ? 'undefined' : _typeof(result)) !== 'object') {\n try {\n result = JSON.parse(result);\n } catch (ex) {\n // hook - fail\n if (hooks.fail && typeof hooks.fail === 'function') {\n hooks.fail(xhr, editor, result);\n }\n\n _this3._alert('上传图片失败', '上传图片返回结果错误,返回结果是: ' + result);\n return;\n }\n }\n if (!hooks.customInsert && result.errno != '0') {\n // hook - fail\n if (hooks.fail && typeof hooks.fail === 'function') {\n hooks.fail(xhr, editor, result);\n }\n\n // 数据错误\n _this3._alert('上传图片失败', '上传图片返回结果错误,返回结果 errno=' + result.errno);\n } else {\n if (hooks.customInsert && typeof hooks.customInsert === 'function') {\n // 使用者自定义插入方法\n hooks.customInsert(_this3.insertLinkImg.bind(_this3), result, editor);\n } else {\n // 将图片插入编辑器\n var data = result.data || [];\n data.forEach(function (link) {\n _this3.insertLinkImg(link);\n });\n }\n\n // hook - success\n if (hooks.success && typeof hooks.success === 'function') {\n hooks.success(xhr, editor, result);\n }\n }\n }\n };\n\n // hook - before\n if (hooks.before && typeof hooks.before === 'function') {\n var beforeResult = hooks.before(xhr, editor, resultFiles);\n if (beforeResult && (typeof beforeResult === 'undefined' ? 'undefined' : _typeof(beforeResult)) === 'object') {\n if (beforeResult.prevent) {\n // 如果返回的结果是 {prevent: true, msg: 'xxxx'} 则表示用户放弃上传\n this._alert(beforeResult.msg);\n return;\n }\n }\n }\n\n // 自定义 headers\n objForEach(uploadImgHeaders, function (key, val) {\n xhr.setRequestHeader(key, val);\n });\n\n // 跨域传 cookie\n xhr.withCredentials = withCredentials;\n\n // 发送请求\n xhr.send(formdata);\n\n // 注意,要 return 。不去操作接下来的 base64 显示方式\n return;\n }\n\n // ------------------------------ 显示 base64 格式 ------------------------------\n if (uploadImgShowBase64) {\n arrForEach(files, function (file) {\n var _this = _this3;\n var reader = new FileReader();\n reader.readAsDataURL(file);\n reader.onload = function () {\n _this.insertLinkImg(this.result);\n };\n });\n }\n }\n};\n\n/*\n 编辑器构造函数\n*/\n\n// id,累加\nvar editorId = 1;\n\n// 构造函数\nfunction Editor(toolbarSelector, textSelector) {\n if (toolbarSelector == null) {\n // 没有传入任何参数,报错\n throw new Error('错误:初始化编辑器时候未传入任何参数,请查阅文档');\n }\n // id,用以区分单个页面不同的编辑器对象\n this.id = 'wangEditor-' + editorId++;\n\n this.toolbarSelector = toolbarSelector;\n this.textSelector = textSelector;\n\n // 自定义配置\n this.customConfig = {};\n}\n\n// 修改原型\nEditor.prototype = {\n constructor: Editor,\n\n // 初始化配置\n _initConfig: function _initConfig() {\n // _config 是默认配置,this.customConfig 是用户自定义配置,将它们 merge 之后再赋值\n var target = {};\n this.config = Object.assign(target, config, this.customConfig);\n\n // 将语言配置,生成正则表达式\n var langConfig = this.config.lang || {};\n var langArgs = [];\n objForEach(langConfig, function (key, val) {\n // key 即需要生成正则表达式的规则,如“插入链接”\n // val 即需要被替换成的语言,如“insert link”\n langArgs.push({\n reg: new RegExp(key, 'img'),\n val: val\n\n });\n });\n this.config.langArgs = langArgs;\n },\n\n // 初始化 DOM\n _initDom: function _initDom() {\n var _this = this;\n\n var toolbarSelector = this.toolbarSelector;\n var $toolbarSelector = $(toolbarSelector);\n var textSelector = this.textSelector;\n\n var config$$1 = this.config;\n var zIndex = config$$1.zIndex;\n\n // 定义变量\n var $toolbarElem = void 0,\n $textContainerElem = void 0,\n $textElem = void 0,\n $children = void 0;\n\n if (textSelector == null) {\n // 只传入一个参数,即是容器的选择器或元素,toolbar 和 text 的元素自行创建\n $toolbarElem = $('<div></div>');\n $textContainerElem = $('<div></div>');\n\n // 将编辑器区域原有的内容,暂存起来\n $children = $toolbarSelector.children();\n\n // 添加到 DOM 结构中\n $toolbarSelector.append($toolbarElem).append($textContainerElem);\n\n // 自行创建的,需要配置默认的样式\n $toolbarElem.css('background-color', '#f1f1f1').css('border', '1px solid #ccc');\n $textContainerElem.css('border', '1px solid #ccc').css('border-top', 'none').css('height', '300px');\n } else {\n // toolbar 和 text 的选择器都有值,记录属性\n $toolbarElem = $toolbarSelector;\n $textContainerElem = $(textSelector);\n // 将编辑器区域原有的内容,暂存起来\n $children = $textContainerElem.children();\n }\n\n // 编辑区域\n $textElem = $('<div></div>');\n $textElem.attr('contenteditable', 'true').css('width', '100%').css('height', '100%');\n\n // 初始化编辑区域内容\n if ($children && $children.length) {\n $textElem.append($children);\n } else {\n $textElem.append($('<p><br></p>'));\n }\n\n // 编辑区域加入DOM\n $textContainerElem.append($textElem);\n\n // 设置通用的 class\n $toolbarElem.addClass('w-e-toolbar');\n $textContainerElem.addClass('w-e-text-container');\n $textContainerElem.css('z-index', zIndex);\n $textElem.addClass('w-e-text');\n\n // 添加 ID\n var toolbarElemId = getRandom('toolbar-elem');\n $toolbarElem.attr('id', toolbarElemId);\n var textElemId = getRandom('text-elem');\n $textElem.attr('id', textElemId);\n\n // 记录属性\n this.$toolbarElem = $toolbarElem;\n this.$textContainerElem = $textContainerElem;\n this.$textElem = $textElem;\n this.toolbarElemId = toolbarElemId;\n this.textElemId = textElemId;\n\n // 记录输入法的开始和结束\n var compositionEnd = true;\n $textContainerElem.on('compositionstart', function () {\n // 输入法开始输入\n compositionEnd = false;\n });\n $textContainerElem.on('compositionend', function () {\n // 输入法结束输入\n compositionEnd = true;\n });\n\n // 绑定 onchange\n $textContainerElem.on('click keyup', function () {\n // 输入法结束才出发 onchange\n compositionEnd && _this.change && _this.change();\n });\n $toolbarElem.on('click', function () {\n this.change && this.change();\n });\n\n //绑定 onfocus 与 onblur 事件\n if (config$$1.onfocus || config$$1.onblur) {\n // 当前编辑器是否是焦点状态\n this.isFocus = false;\n\n $(document).on('click', function (e) {\n //判断当前点击元素是否在编辑器内\n var isChild = $textElem.isContain($(e.target));\n\n //判断当前点击元素是否为工具栏\n var isToolbar = $toolbarElem.isContain($(e.target));\n var isMenu = $toolbarElem[0] == e.target ? true : false;\n\n if (!isChild) {\n //若为选择工具栏中的功能,则不视为成blur操作\n if (isToolbar && !isMenu) {\n return;\n }\n\n if (_this.isFocus) {\n _this.onblur && _this.onblur();\n }\n _this.isFocus = false;\n } else {\n if (!_this.isFocus) {\n _this.onfocus && _this.onfocus();\n }\n _this.isFocus = true;\n }\n });\n }\n },\n\n // 封装 command\n _initCommand: function _initCommand() {\n this.cmd = new Command(this);\n },\n\n // 封装 selection range API\n _initSelectionAPI: function _initSelectionAPI() {\n this.selection = new API(this);\n },\n\n // 添加图片上传\n _initUploadImg: function _initUploadImg() {\n this.uploadImg = new UploadImg(this);\n },\n\n // 初始化菜单\n _initMenus: function _initMenus() {\n this.menus = new Menus(this);\n this.menus.init();\n },\n\n // 添加 text 区域\n _initText: function _initText() {\n this.txt = new Text(this);\n this.txt.init();\n },\n\n // 初始化选区,将光标定位到内容尾部\n initSelection: function initSelection(newLine) {\n var $textElem = this.$textElem;\n var $children = $textElem.children();\n if (!$children.length) {\n // 如果编辑器区域无内容,添加一个空行,重新设置选区\n $textElem.append($('<p><br></p>'));\n this.initSelection();\n return;\n }\n\n var $last = $children.last();\n\n if (newLine) {\n // 新增一个空行\n var html = $last.html().toLowerCase();\n var nodeName = $last.getNodeName();\n if (html !== '<br>' && html !== '<br\\/>' || nodeName !== 'P') {\n // 最后一个元素不是 <p><br></p>,添加一个空行,重新设置选区\n $textElem.append($('<p><br></p>'));\n this.initSelection();\n return;\n }\n }\n\n this.selection.createRangeByElem($last, false, true);\n this.selection.restoreSelection();\n },\n\n // 绑定事件\n _bindEvent: function _bindEvent() {\n // -------- 绑定 onchange 事件 --------\n var onChangeTimeoutId = 0;\n var beforeChangeHtml = this.txt.html();\n var config$$1 = this.config;\n\n // onchange 触发延迟时间\n var onchangeTimeout = config$$1.onchangeTimeout;\n onchangeTimeout = parseInt(onchangeTimeout, 10);\n if (!onchangeTimeout || onchangeTimeout <= 0) {\n onchangeTimeout = 200;\n }\n\n var onchange = config$$1.onchange;\n if (onchange && typeof onchange === 'function') {\n // 触发 change 的有三个场景:\n // 1. $textContainerElem.on('click keyup')\n // 2. $toolbarElem.on('click')\n // 3. editor.cmd.do()\n this.change = function () {\n // 判断是否有变化\n var currentHtml = this.txt.html();\n\n if (currentHtml.length === beforeChangeHtml.length) {\n // 需要比较每一个字符\n if (currentHtml === beforeChangeHtml) {\n return;\n }\n }\n\n // 执行,使用节流\n if (onChangeTimeoutId) {\n clearTimeout(onChangeTimeoutId);\n }\n onChangeTimeoutId = setTimeout(function () {\n // 触发配置的 onchange 函数\n onchange(currentHtml);\n beforeChangeHtml = currentHtml;\n }, onchangeTimeout);\n };\n }\n\n // -------- 绑定 onblur 事件 --------\n var onblur = config$$1.onblur;\n if (onblur && typeof onblur === 'function') {\n this.onblur = function () {\n var currentHtml = this.txt.html();\n onblur(currentHtml);\n };\n }\n\n // -------- 绑定 onfocus 事件 --------\n var onfocus = config$$1.onfocus;\n if (onfocus && typeof onfocus === 'function') {\n this.onfocus = function () {\n onfocus();\n };\n }\n },\n\n // 创建编辑器\n create: function create() {\n // 初始化配置信息\n this._initConfig();\n\n // 初始化 DOM\n this._initDom();\n\n // 封装 command API\n this._initCommand();\n\n // 封装 selection range API\n this._initSelectionAPI();\n\n // 添加 text\n this._initText();\n\n // 初始化菜单\n this._initMenus();\n\n // 添加 图片上传\n this._initUploadImg();\n\n // 初始化选区,将光标定位到内容尾部\n this.initSelection(true);\n\n // 绑定事件\n this._bindEvent();\n },\n\n // 解绑所有事件(暂时不对外开放)\n _offAllEvent: function _offAllEvent() {\n $.offAll();\n }\n};\n\n// 检验是否浏览器环境\ntry {\n document;\n} catch (ex) {\n throw new Error('请在浏览器环境下运行');\n}\n\n// polyfill\npolyfill();\n\n// 这里的 `inlinecss` 将被替换成 css 代码的内容,详情可去 ./gulpfile.js 中搜索 `inlinecss` 关键字\nvar inlinecss = '.w-e-toolbar,.w-e-text-container,.w-e-menu-panel { padding: 0; margin: 0; box-sizing: border-box;}.w-e-toolbar *,.w-e-text-container *,.w-e-menu-panel * { padding: 0; margin: 0; box-sizing: border-box;}.w-e-clear-fix:after { content: \"\"; display: table; clear: both;}.w-e-toolbar .w-e-droplist { position: absolute; left: 0; top: 0; background-color: #fff; border: 1px solid #f1f1f1; border-right-color: #ccc; border-bottom-color: #ccc;}.w-e-toolbar .w-e-droplist .w-e-dp-title { text-align: center; color: #999; line-height: 2; border-bottom: 1px solid #f1f1f1; font-size: 13px;}.w-e-toolbar .w-e-droplist ul.w-e-list { list-style: none; line-height: 1;}.w-e-toolbar .w-e-droplist ul.w-e-list li.w-e-item { color: #333; padding: 5px 0;}.w-e-toolbar .w-e-droplist ul.w-e-list li.w-e-item:hover { background-color: #f1f1f1;}.w-e-toolbar .w-e-droplist ul.w-e-block { list-style: none; text-align: left; padding: 5px;}.w-e-toolbar .w-e-droplist ul.w-e-block li.w-e-item { display: inline-block; *display: inline; *zoom: 1; padding: 3px 5px;}.w-e-toolbar .w-e-droplist ul.w-e-block li.w-e-item:hover { background-color: #f1f1f1;}@font-face { font-family: \\'w-e-icon\\'; src: url(data:application/x-font-woff;charset=utf-8;base64,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) format(\\'truetype\\'); font-weight: normal; font-style: normal;}[class^=\"w-e-icon-\"],[class*=\" w-e-icon-\"] { /* use !important to prevent issues with browser extensions that change fonts */ font-family: \\'w-e-icon\\' !important; speak: none; font-style: normal; font-weight: normal; font-variant: normal; text-transform: none; line-height: 1; /* Better Font Rendering =========== */ -webkit-font-smoothing: antialiased; -moz-osx-font-smoothing: grayscale;}.w-e-icon-close:before { content: \"\\\\f00d\";}.w-e-icon-upload2:before { content: \"\\\\e9c6\";}.w-e-icon-trash-o:before { content: \"\\\\f014\";}.w-e-icon-header:before { content: \"\\\\f1dc\";}.w-e-icon-pencil2:before { content: \"\\\\e906\";}.w-e-icon-paint-brush:before { content: \"\\\\f1fc\";}.w-e-icon-image:before { content: \"\\\\e90d\";}.w-e-icon-play:before { content: \"\\\\e912\";}.w-e-icon-location:before { content: \"\\\\e947\";}.w-e-icon-undo:before { content: \"\\\\e965\";}.w-e-icon-redo:before { content: \"\\\\e966\";}.w-e-icon-quotes-left:before { content: \"\\\\e977\";}.w-e-icon-list-numbered:before { content: \"\\\\e9b9\";}.w-e-icon-list2:before { content: \"\\\\e9bb\";}.w-e-icon-link:before { content: \"\\\\e9cb\";}.w-e-icon-happy:before { content: \"\\\\e9df\";}.w-e-icon-bold:before { content: \"\\\\ea62\";}.w-e-icon-underline:before { content: \"\\\\ea63\";}.w-e-icon-italic:before { content: \"\\\\ea64\";}.w-e-icon-strikethrough:before { content: \"\\\\ea65\";}.w-e-icon-table2:before { content: \"\\\\ea71\";}.w-e-icon-paragraph-left:before { content: \"\\\\ea77\";}.w-e-icon-paragraph-center:before { content: \"\\\\ea78\";}.w-e-icon-paragraph-right:before { content: \"\\\\ea79\";}.w-e-icon-terminal:before { content: \"\\\\f120\";}.w-e-icon-page-break:before { content: \"\\\\ea68\";}.w-e-icon-cancel-circle:before { content: \"\\\\ea0d\";}.w-e-icon-font:before { content: \"\\\\ea5c\";}.w-e-icon-text-heigh:before { content: \"\\\\ea5f\";}.w-e-toolbar { display: -webkit-box; display: -ms-flexbox; display: flex; padding: 0 5px; /* flex-wrap: wrap; */ /* 单个菜单 */}.w-e-toolbar .w-e-menu { position: relative; text-align: center; padding: 5px 10px; cursor: pointer;}.w-e-toolbar .w-e-menu i { color: #999;}.w-e-toolbar .w-e-menu:hover i { color: #333;}.w-e-toolbar .w-e-active i { color: #1e88e5;}.w-e-toolbar .w-e-active:hover i { color: #1e88e5;}.w-e-text-container .w-e-panel-container { position: absolute; top: 0; left: 50%; border: 1px solid #ccc; border-top: 0; box-shadow: 1px 1px 2px #ccc; color: #333; background-color: #fff; /* 为 emotion panel 定制的样式 */ /* 上传图片的 panel 定制样式 */}.w-e-text-container .w-e-panel-container .w-e-panel-close { position: absolute; right: 0; top: 0; padding: 5px; margin: 2px 5px 0 0; cursor: pointer; color: #999;}.w-e-text-container .w-e-panel-container .w-e-panel-close:hover { color: #333;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-title { list-style: none; display: -webkit-box; display: -ms-flexbox; display: flex; font-size: 14px; margin: 2px 10px 0 10px; border-bottom: 1px solid #f1f1f1;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-title .w-e-item { padding: 3px 5px; color: #999; cursor: pointer; margin: 0 3px; position: relative; top: 1px;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-title .w-e-active { color: #333; border-bottom: 1px solid #333; cursor: default; font-weight: 700;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content { padding: 10px 15px 10px 15px; font-size: 16px; /* 输入框的样式 */ /* 按钮的样式 */}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content input:focus,.w-e-text-container .w-e-panel-container .w-e-panel-tab-content textarea:focus,.w-e-text-container .w-e-panel-container .w-e-panel-tab-content button:focus { outline: none;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content textarea { width: 100%; border: 1px solid #ccc; padding: 5px;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content textarea:focus { border-color: #1e88e5;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content input[type=text] { border: none; border-bottom: 1px solid #ccc; font-size: 14px; height: 20px; color: #333; text-align: left;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content input[type=text].small { width: 30px; text-align: center;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content input[type=text].block { display: block; width: 100%; margin: 10px 0;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content input[type=text]:focus { border-bottom: 2px solid #1e88e5;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container button { font-size: 14px; color: #1e88e5; border: none; padding: 5px 10px; background-color: #fff; cursor: pointer; border-radius: 3px;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container button.left { float: left; margin-right: 10px;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container button.right { float: right; margin-left: 10px;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container button.gray { color: #999;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container button.red { color: #c24f4a;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container button:hover { background-color: #f1f1f1;}.w-e-text-container .w-e-panel-container .w-e-panel-tab-content .w-e-button-container:after { content: \"\"; display: table; clear: both;}.w-e-text-container .w-e-panel-container .w-e-emoticon-container .w-e-item { cursor: pointer; font-size: 18px; padding: 0 3px; display: inline-block; *display: inline; *zoom: 1;}.w-e-text-container .w-e-panel-container .w-e-up-img-container { text-align: center;}.w-e-text-container .w-e-panel-container .w-e-up-img-container .w-e-up-btn { display: inline-block; *display: inline; *zoom: 1; color: #999; cursor: pointer; font-size: 60px; line-height: 1;}.w-e-text-container .w-e-panel-container .w-e-up-img-container .w-e-up-btn:hover { color: #333;}.w-e-text-container { position: relative;}.w-e-text-container .w-e-progress { position: absolute; background-color: #1e88e5; bottom: 0; left: 0; height: 1px;}.w-e-text { padding: 0 10px; overflow-y: scroll;}.w-e-text p,.w-e-text h1,.w-e-text h2,.w-e-text h3,.w-e-text h4,.w-e-text h5,.w-e-text table,.w-e-text pre { margin: 10px 0; line-height: 1.5;}.w-e-text ul,.w-e-text ol { margin: 10px 0 10px 20px;}.w-e-text blockquote { display: block; border-left: 8px solid #d0e5f2; padding: 5px 10px; margin: 10px 0; line-height: 1.4; font-size: 100%; background-color: #f1f1f1;}.w-e-text code { display: inline-block; *display: inline; *zoom: 1; background-color: #f1f1f1; border-radius: 3px; padding: 3px 5px; margin: 0 3px;}.w-e-text pre code { display: block;}.w-e-text table { border-top: 1px solid #ccc; border-left: 1px solid #ccc;}.w-e-text table td,.w-e-text table th { border-bottom: 1px solid #ccc; border-right: 1px solid #ccc; padding: 3px 5px;}.w-e-text table th { border-bottom: 2px solid #ccc; text-align: center;}.w-e-text:focus { outline: none;}.w-e-text img { cursor: pointer;}.w-e-text img:hover { box-shadow: 0 0 5px #333;}';\n\n// 将 css 代码添加到 <style> 中\nvar style = document.createElement('style');\nstyle.type = 'text/css';\nstyle.innerHTML = inlinecss;\ndocument.getElementsByTagName('HEAD').item(0).appendChild(style);\n\n// 返回\nvar index = window.wangEditor || Editor;\n\nreturn index;\n\n})));\n","var render = function () {var _vm=this;var _h=_vm.$createElement;var _c=_vm._self._c||_h;return _c('div',{staticClass:\"editor\"},[_c('div',{ref:\"toolbar\",staticClass:\"toolbar\"}),_c('div',{ref:\"editor\",staticClass:\"text\"})])}\nvar staticRenderFns = []\n\nexport { render, staticRenderFns }","<template lang=\"html\">\r\n <div class=\"editor\">\r\n <div ref=\"toolbar\" class=\"toolbar\">\r\n </div>\r\n <div ref=\"editor\" class=\"text\">\r\n </div>\r\n </div>\r\n</template>\r\n\r\n<script>\r\nimport E from \"wangeditor\";\r\nexport default {\r\n name: \"editoritem\",\r\n data() {\r\n return {\r\n // uploadPath,\r\n editor: null,\r\n info_: null\r\n };\r\n },\r\n model: {\r\n prop: \"value\",\r\n event: \"change\"\r\n },\r\n props: {\r\n value: {\r\n type: String,\r\n default: \"\"\r\n },\r\n isClear: {\r\n type: Boolean,\r\n default: false\r\n }\r\n },\r\n watch: {\r\n isClear(val) {\r\n // 触发清除文本域内容\r\n if (val) {\r\n this.editor.txt.clear();\r\n this.info_ = null;\r\n }\r\n },\r\n value: function(value) {\r\n if (value !== this.editor.txt.html()) {\r\n this.editor.txt.html(this.value);\r\n }\r\n }\r\n //value为编辑框输入的内容,这里我监听了一下值,当父组件调用得时候,如果给value赋值了,子组件将会显示父组件赋给的值\r\n },\r\n mounted() {\r\n this.seteditor();\r\n this.editor.txt.html(this.value);\r\n },\r\n methods: {\r\n seteditor() {\r\n\r\n // http://192.168.2.125:8080/admin/storage/create\r\n this.editor = new E(this.$refs.toolbar, this.$refs.editor);\r\n this.editor.customConfig.uploadImgShowBase64 = false; // base 64 存储图片\r\n this.editor.customConfig.uploadImgServer =\r\n \"/api/admin/uploadfile\"; // 配置服务器端地址\r\n this.editor.customConfig.uploadImgHeaders = {}; // 自定义 header\r\n this.editor.customConfig.uploadFileName = \"file\"; // 后端接受上传文件的参数名\r\n this.editor.customConfig.uploadImgMaxSize = 2 * 1024 * 1024; // 将图片大小限制为 2M\r\n this.editor.customConfig.uploadImgMaxLength = 6; // 限制一次最多上传 3 张图片\r\n this.editor.customConfig.uploadImgTimeout = 3 * 60 * 1000; // 设置超时时间\r\n\r\n // 配置菜单\r\n this.editor.customConfig.menus = [\r\n \"head\", // 标题\r\n \"bold\", // 粗体\r\n \"fontSize\", // 字号\r\n \"fontName\", // 字体\r\n \"italic\", // 斜体\r\n \"underline\", // 下划线\r\n \"strikeThrough\", // 删除线\r\n \"foreColor\", // 文字颜色\r\n \"backColor\", // 背景颜色\r\n \"link\", // 插入链接\r\n \"list\", // 列表\r\n \"justify\", // 对齐方式\r\n \"quote\", // 引用\r\n \"emoticon\", // 表情\r\n \"image\", // 插入图片\r\n \"table\", // 表格\r\n \"video\", // 插入视频\r\n \"code\", // 插入代码\r\n \"undo\", // 撤销\r\n \"redo\", // 重复\r\n \"fullscreen\" // 全屏\r\n ];\r\n\r\n this.editor.customConfig.uploadImgHooks = {\r\n fail: (xhr, editor, result) => {\r\n // 插入图片失败回调\r\n },\r\n success: (xhr, editor, result) => {\r\n // 图片上传成功回调\r\n },\r\n timeout: (xhr, editor) => {\r\n // 网络超时的回调\r\n },\r\n error: (xhr, editor) => {\r\n // 图片上传错误的回调\r\n },\r\n customInsert: (insertImg, result, editor) => {\r\n // 图片上传成功,插入图片的回调\r\n //result为上传图片成功的时候返回的数据,这里我打印了一下发现后台返回的是data:[{url:\"路径的形式\"},...]\r\n // console.log(result.data[0].url)\r\n //insertImg()为插入图片的函数\r\n //循环插入图片\r\n // for (let i = 0; i < 1; i++) {\r\n // console.log(result)\r\n let url = result.data.url;\r\n console.log(result)\r\n insertImg(url);\r\n // }\r\n }\r\n };\r\n this.editor.customConfig.onchange = html => {\r\n this.info_ = html; // 绑定当前逐渐地值\r\n this.$emit(\"change\", this.info_); // 将内容同步到父组件中\r\n };\r\n // 创建富文本编辑器\r\n this.editor.create();\r\n }\r\n }\r\n};\r\n</script>\r\n\r\n<style lang=\"scss\" scoped>\r\n.editor {\r\n width: 100%;\r\n margin: 0 auto;\r\n position: relative;\r\n z-index: 0;\r\n /deep/ .toolbar {\r\n border: 1px solid #ccc;\r\n }\r\n /deep/ .text {\r\n border: 1px solid #ccc;\r\n min-height: 500px;\r\n }\r\n}\r\n</style>","import mod from \"-!../../../node_modules/_cache-loader@2.0.1@cache-loader/dist/cjs.js??ref--12-0!../../../node_modules/_thread-loader@2.1.3@thread-loader/dist/cjs.js!../../../node_modules/_babel-loader@8.2.2@babel-loader/lib/index.js!../../../node_modules/_cache-loader@2.0.1@cache-loader/dist/cjs.js??ref--0-0!../../../node_modules/_vue-loader@15.9.6@vue-loader/lib/index.js??vue-loader-options!./index.vue?vue&type=script&lang=js&\"; export default mod; export * from \"-!../../../node_modules/_cache-loader@2.0.1@cache-loader/dist/cjs.js??ref--12-0!../../../node_modules/_thread-loader@2.1.3@thread-loader/dist/cjs.js!../../../node_modules/_babel-loader@8.2.2@babel-loader/lib/index.js!../../../node_modules/_cache-loader@2.0.1@cache-loader/dist/cjs.js??ref--0-0!../../../node_modules/_vue-loader@15.9.6@vue-loader/lib/index.js??vue-loader-options!./index.vue?vue&type=script&lang=js&\"","import { render, staticRenderFns } from \"./index.vue?vue&type=template&id=55e58fac&scoped=true&lang=html&\"\nimport script from \"./index.vue?vue&type=script&lang=js&\"\nexport * from \"./index.vue?vue&type=script&lang=js&\"\nimport style0 from \"./index.vue?vue&type=style&index=0&id=55e58fac&lang=scss&scoped=true&\"\n\n\n/* normalize component */\nimport normalizer from \"!../../../node_modules/_vue-loader@15.9.6@vue-loader/lib/runtime/componentNormalizer.js\"\nvar component = normalizer(\n script,\n render,\n staticRenderFns,\n false,\n null,\n \"55e58fac\",\n null\n \n)\n\nexport default component.exports","export * from \"-!../../../node_modules/_mini-css-extract-plugin@0.8.2@mini-css-extract-plugin/dist/loader.js??ref--8-oneOf-1-0!../../../node_modules/_css-loader@1.0.1@css-loader/index.js??ref--8-oneOf-1-1!../../../node_modules/_vue-loader@15.9.6@vue-loader/lib/loaders/stylePostLoader.js!../../../node_modules/_postcss-loader@3.0.0@postcss-loader/src/index.js??ref--8-oneOf-1-2!../../../node_modules/_sass-loader@7.3.1@sass-loader/dist/cjs.js??ref--8-oneOf-1-3!../../../node_modules/_cache-loader@2.0.1@cache-loader/dist/cjs.js??ref--0-0!../../../node_modules/_vue-loader@15.9.6@vue-loader/lib/index.js??vue-loader-options!./index.vue?vue&type=style&index=0&id=55e58fac&lang=scss&scoped=true&\""],"sourceRoot":""}
|