Htmleditor.js 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605
  1. MWF.xDesktop.requireApp("process.Xform", "$Module", null, false);
  2. /** @class Htmleditor HTML编辑器。
  3. * @example
  4. * //可以在脚本中获取该组件
  5. * //方法1:
  6. * var htmlEditor = this.form.get("name"); //获取组件
  7. * //方法2
  8. * var htmlEditor = this.target; //在组件事件脚本中获取
  9. * @extends MWF.xApplication.process.Xform.$Module
  10. * @o2category FormComponents
  11. * @o2range {Process|CMS}
  12. * @hideconstructor
  13. */
  14. MWF.xApplication.process.Xform.Htmleditor = MWF.APPHtmleditor = new Class(
  15. /** @lends MWF.xApplication.process.Xform.Htmleditor# */
  16. {
  17. Extends: MWF.APP$Module,
  18. options: {
  19. "moduleEvents": ["load", "postLoad", "afterLoad"]
  20. },
  21. initialize: function(node, json, form, options){
  22. this.node = $(node);
  23. this.node.store("module", this);
  24. this.json = json;
  25. this.form = form;
  26. this.field = true;
  27. },
  28. load: function(){
  29. if (this.fireEvent("queryLoad")){
  30. this._queryLoaded();
  31. this._loadUserInterface();
  32. this._loadStyles();
  33. //this._loadEvents();
  34. this._afterLoaded();
  35. this.fireEvent("postLoad");
  36. this.fireEvent("load");
  37. }
  38. },
  39. _loadUserInterface: function(){
  40. this.node.empty();
  41. if (this.readonly){
  42. this.node.set("html", this._getBusinessData());
  43. this.node.setStyles({
  44. "-webkit-user-select": "text",
  45. "-moz-user-select": "text"
  46. });
  47. }else{
  48. var config = Object.clone(this.json.editorProperties);
  49. if (this.json.config){
  50. if (this.json.config.code){
  51. var obj = this.form.Macro.exec(this.json.config.code, this);
  52. Object.each(obj, function(v, k){
  53. config[k] = v;
  54. });
  55. }
  56. }
  57. this.loadCkeditor(config);
  58. }
  59. // this._loadValue();
  60. },
  61. loadCkeditor: function(config){
  62. COMMON.AjaxModule.loadDom("ckeditor", function(){
  63. CKEDITOR.disableAutoInline = true;
  64. var editorDiv = new Element("div").inject(this.node);
  65. var htmlData = this._getBusinessData();
  66. if (htmlData){
  67. editorDiv.set("html", htmlData);
  68. }else if (this.json.templateCode){
  69. editorDiv.set("html", this.json.templateCode);
  70. }
  71. var height = this.node.getSize().y;
  72. var editorConfig = config || {};
  73. if (this.form.json.mode==="Mobile"){
  74. if (!editorConfig.toolbar && !editorConfig.toolbarGroups){
  75. editorConfig.toolbar = [
  76. { name: 'paragraph', items: [ 'Bold', 'Italic', "-" , 'TextColor', "BGColor", 'JustifyLeft', 'JustifyCenter', 'JustifyRight', 'JustifyBlock', "-", 'Undo', 'Redo' ] },
  77. { name: 'basicstyles', items: [ 'Styles', 'FontSize']},
  78. { name: 'insert', items : [ 'Image' ] }
  79. ];
  80. }
  81. }
  82. editorConfig.localImageMaxWidth = 800;
  83. editorConfig.reference = this.form.businessData.work.job;
  84. editorConfig.referenceType = "processPlatformJob";
  85. if( editorConfig && editorConfig.extraPlugins ){
  86. var extraPlugins = editorConfig.extraPlugins;
  87. extraPlugins = typeOf( extraPlugins ) === "array" ? extraPlugins : extraPlugins.split(",");
  88. extraPlugins.push( 'pagebreak' );
  89. editorConfig.extraPlugins = extraPlugins;
  90. }else{
  91. editorConfig.extraPlugins = ['pagebreak'];
  92. }
  93. // CKEDITOR.basePath = COMMON.contentPath+"/res/framework/htmleditor/ckeditor/";
  94. // CKEDITOR.plugins.basePath = COMMON.contentPath+"/res/framework/htmleditor/ckeditor/plugins/";
  95. this.editor = CKEDITOR.replace(editorDiv, editorConfig);
  96. this.editor.addCommand("ecnet", {
  97. exec:function(editor){
  98. this.ecnet();
  99. }.bind(this)
  100. });
  101. this.editor.ui.add('ecnet', CKEDITOR.UI_BUTTON, {
  102. label:'智能纠错',
  103. icon: '../x_component_process_Xform/$Form/default/icon/ecnet.png',
  104. command:"ecnet"
  105. });
  106. this._loadEvents();
  107. //this.editor.on("loaded", function(){
  108. // this._loadEvents();
  109. //}.bind(this));
  110. //this.setData(data)
  111. this.editor.on("change", function(){
  112. //this._setBusinessData(this.getData());
  113. }.bind(this));
  114. if (this.json.ecnet==="y"){
  115. // this.editor.on( "key", function( evt ) {
  116. // // var char = evt.data.domEvent.$.char;
  117. // // if ([".", ",", "?", ";", "'", " "].indexOf(char)!==-1){
  118. // // this.ecnet(evt.editor.getData());
  119. // // }
  120. // }.bind(this));
  121. // this.editor.on("blur", function(){
  122. // if (!this.notEcnetFlag) this.ecnet(this.getData());
  123. // }.bind(this));
  124. }
  125. // this._loadEvents();
  126. }.bind(this));
  127. },
  128. getEcnetString: function(node, nodes){
  129. for (var i=0; i<node.childNodes.length; i++){
  130. if (node.childNodes[i].nodeType===Node.TEXT_NODE){
  131. var s = this.ecnetString.length;
  132. this.ecnetString += node.childNodes[i].nodeValue;
  133. var e = this.ecnetString.length;
  134. nodes.push({
  135. "pnode": node,
  136. "node": node.childNodes[i],
  137. "start": s, "end": e
  138. });
  139. }else{
  140. this.getEcnetString(node.childNodes[i], nodes);
  141. }
  142. }
  143. },
  144. createEcnetNode: function(node){
  145. var newNode = node.node.ownerDocument.createElement("span");
  146. var increment = 0;
  147. var html = node.node.nodeValue;;
  148. node.ecnets.each(function(ecnet){
  149. var s = ecnet.begin+increment-node.start;
  150. var e = ecnet.end+increment-node.start;
  151. if (s<0) s=0;
  152. if (e>node.end+increment) e = node.end+increment;
  153. var length = html.length;
  154. var left = html.substring(0, s);
  155. var ecnetStr = html.substring(s, e);
  156. var right = html.substring(e, html.length);
  157. html = left+"<span class='o2_ecnet_item' style='color: red'><u>"+ecnetStr+"</u></span>"+right;
  158. increment += (html.length-length);
  159. }.bind(this));
  160. newNode.innerHTML = html;
  161. node.pnode.replaceChild(newNode, node.node);
  162. node.pnode.textNode = node.node;
  163. node.pnode.ecnetNode = newNode;
  164. var _self = this;
  165. var editorFrame = this.editor.document.$.defaultView.frameElement;
  166. var spans = newNode.getElementsByTagName("span");
  167. if (spans.length){
  168. for (var i = 0; i<spans.length; i++){
  169. var span = spans[i];
  170. if (span.className==="o2_ecnet_item"){
  171. var ecnetNode = new Element("div", {"styles": {
  172. "border": "1px solid #999999",
  173. "box-shadow": "0px 0px 5px #999999",
  174. "background-color": "#ffffff",
  175. "position": "fixed",
  176. "display": "none"
  177. }}).inject(editorFrame, "after");
  178. var correctNode = new Element("div", {
  179. "styles": {
  180. "padding": "3px 10px",
  181. "font-weight": "bold",
  182. "font-size": "12px",
  183. "cursor": "pointer"
  184. },
  185. "text": node.ecnets[i].origin+"->"+node.ecnets[i].correct,
  186. "events": {
  187. "mouseover": function(){this.setStyle("background-color", "#dddddd")},
  188. "mouseout": function(){this.setStyle("background-color", "#ffffff")},
  189. "mousedown": function(){
  190. var ecnetNode = this.getParent();
  191. var node = ecnetNode.node;
  192. var item = ecnetNode.node.ecnets[ecnetNode.idx];
  193. var textNode = node.node.ownerDocument.createTextNode(item.correct);
  194. ecnetNode.span.parentNode.replaceChild(textNode, ecnetNode.span);
  195. ecnetNode.destroy();
  196. node.node.nodeValue = node.pnode.ecnetNode.innerText;
  197. node.ecnets.erase(item);
  198. if (!node.ecnets.length){
  199. _self.ecnetNodes.erase(node);
  200. }
  201. }
  202. }
  203. }).inject(ecnetNode);
  204. var ignoreNode = new Element("div", {
  205. "styles": {
  206. "padding": "3px 10px",
  207. "font-size": "12px",
  208. "cursor": "pointer"
  209. },
  210. "text": MWF.xApplication.process.Xform.LP.ignore,
  211. "events": {
  212. "mouseover": function(){this.setStyle("background-color", "#dddddd")},
  213. "mouseout": function(){this.setStyle("background-color", "#ffffff")},
  214. "mousedown": function(){
  215. var ecnetNode = this.getParent();
  216. var node = ecnetNode.node;
  217. var item = ecnetNode.node.ecnets[ecnetNode.idx];
  218. var textNode = node.node.ownerDocument.createTextNode(ecnetNode.span.innerText);
  219. ecnetNode.span.parentNode.replaceChild(textNode, ecnetNode.span);
  220. ecnetNode.destroy();
  221. node.node.nodeValue = node.pnode.ecnetNode.innerText;
  222. node.ecnets.erase(item);
  223. if (!node.ecnets.length){
  224. _self.ecnetNodes.erase(node);
  225. }
  226. }
  227. }
  228. }).inject(ecnetNode);
  229. ecnetNode.node = node;
  230. ecnetNode.idx = i;
  231. span.ecnetNode = ecnetNode;
  232. ecnetNode.span = span;
  233. span.addEventListener("click", function(){
  234. var ecnetNode = this.ecnetNode;
  235. ecnetNode.show();
  236. var y = this.offsetTop;
  237. var x = this.offsetLeft;
  238. var w = this.offsetWidth;
  239. var h = this.offsetHeight;
  240. var p = editorFrame.getPosition();
  241. var s = ecnetNode.getSize();
  242. var top = y+p.y+h+5;
  243. var left = x+p.x-((s.x-w)/2);
  244. ecnetNode.style.left = ""+left+"px";
  245. ecnetNode.style.top = ""+top+"px";
  246. var _span = this;
  247. var hideEcnetNode = function(){
  248. ecnetNode.hide();
  249. _span.ownerDocument.removeEventListener("mousedown", hideEcnetNode);
  250. };
  251. this.ownerDocument.addEventListener("mousedown", hideEcnetNode);
  252. });
  253. }
  254. }
  255. }
  256. //node.pnode.ecnetInforNode = ecnetNode;
  257. // var spans = newNode.getElementsByTagName("span");
  258. // if (spans.length){
  259. // var span = spans[0];
  260. // span.addEventListener("click", function(){
  261. // ecnetNode.style.display = "block";
  262. // var y = span.offsetTop;
  263. // var x = span.offsetLeft;
  264. // var w = span.offsetWidth;
  265. // var h = span.offsetHeight;
  266. // var p = editorFrame.getPosition();
  267. // var s = ecnetNode.getSize();
  268. // var top = y+p.y+h+5;
  269. // var left = x+p.x-((s.x-w)/2);
  270. //
  271. // ecnetNode.style.left = ""+left+"px";
  272. // ecnetNode.style.top = ""+top+"px";
  273. // });
  274. // span.addEventListener("mouseout", function(){});
  275. // }
  276. },
  277. clearEcnetNodes: function(){
  278. if (this.ecnetNodes && this.ecnetNodes.length){
  279. this.ecnetNodes.each(function(node){
  280. if (node.pnode.ecnetNode){
  281. if (node.pnode.ecnetInforNode) node.pnode.ecnetInforNode.destroy();
  282. node.pnode.ecnetInforNode = null;
  283. node.pnode.replaceChild(node.pnode.textNode, node.pnode.ecnetNode);
  284. }
  285. }.bind(this));
  286. this.ecnetNodes = [];
  287. }
  288. },
  289. ecnet: function(data){
  290. //this.editor.document.$.body.innerText
  291. var editorFrame = this.editor.document.$.defaultView.frameElement;
  292. //var data = this.editor.getData();
  293. var body = this.editor.document.$.body;
  294. if (!this.ecnetNodes) this.ecnetNodes = [];
  295. if (this.ecnetNodes.length) this.clearEcnetNodes();
  296. var nodes = [];
  297. this.ecnetString = "";
  298. this.getEcnetString(body, nodes);
  299. MWF.Actions.get("x_general_assemble_control").ecnetCheck({"value": this.ecnetString}, function(json){
  300. if (json.data.itemList && json.data.itemList.length){
  301. nodes.each(function(node){
  302. var items = [];
  303. json.data.itemList.each(function(item){
  304. if ((node.end<=item.end && node.end>item.begin) || (node.start>=item.begin && node.start<item.end) || (node.start<=item.begin && node.end>item.end)){
  305. items.push(item);
  306. }
  307. }.bind(this));
  308. if (items.length){
  309. node.ecnets = items;
  310. this.ecnetNodes.push(node);
  311. }
  312. }.bind(this));
  313. this.ecnetNodes.each(function(node){
  314. this.createEcnetNode(node);
  315. }.bind(this));
  316. // var item = json.data.itemList[0];
  317. // var left = data.substring(0, item.begin);
  318. // var ecnetStr = data.substring(item.begin, item.end);
  319. // var right = data.substring(item.end, data.length);
  320. //
  321. // var newData = left+"<span class='o2_ecnet_item' style='color:red' title='"+item.origin+"->"+item.correc+"'><u>"+ecnetStr+"</u></span>"+right;
  322. //this.editor.document.$.body.setSelectionRange(item.begin, item.end);
  323. //this.editor.setData(newData);
  324. // var iframe = editorFrame.clone();
  325. // iframe.inject(this.node);
  326. // iframe.position({
  327. // "relativeTo": editorFrame,
  328. // "position": 'upperLeft',
  329. // "edge": 'upperLeft'
  330. // });
  331. // iframe.contentWindow.document.body.set("html", newData);
  332. }else{
  333. body = null;
  334. nodes = null;
  335. }
  336. }.bind(this));
  337. },
  338. _loadEvents: function(editorConfig){
  339. Object.each(this.json.events, function(e, key){
  340. if (e.code){
  341. this.editor.on(key, function(event){
  342. return this.form.Macro.fire(e.code, this, event);
  343. }.bind(this), this);
  344. }
  345. }.bind(this));
  346. },
  347. addModuleEvent: function(key, fun){
  348. this.editor.on(key, function(event){
  349. return (fun) ? fun(this, event) : null;
  350. }.bind(this), this);
  351. },
  352. _loadValue: function(){
  353. var data = this._getBusinessData();
  354. },
  355. /**
  356. * @summary 重置组件的值为默认值或置空。
  357. * @example
  358. * this.form.get('fieldId').resetData();
  359. */
  360. resetData: function(){
  361. this.setData(this._getBusinessData());
  362. },
  363. /**
  364. * @summary 判断组件值是否为空.
  365. * @example
  366. * if( this.form.get('fieldId').isEmpty() ){
  367. * this.form.notice('HTML编辑器不能为空', 'warn');
  368. * }
  369. * @return {Boolean} 值是否为空.
  370. */
  371. isEmpty : function(){
  372. return !this.getData().trim();
  373. },
  374. /**
  375. * 当表单上没有对应组件的时候,可以使用this.data[fieldId]获取值,但是this.form.get('fieldId')无法获取到组件。
  376. * @summary 获取组件值。
  377. * @example
  378. * var data = this.form.get('fieldId').getData();
  379. * @example
  380. * //如果无法确定表单上是否有组件,需要判断
  381. * var data;
  382. * if( this.form.get('fieldId') ){ //判断表单是否有无对应组件
  383. * data = this.form.get('fieldId').getData();
  384. * }else{
  385. * data = this.data['fieldId']; //直接从数据中获取字段值
  386. * }
  387. * @return 组件的数据.
  388. */
  389. getData: function(){
  390. this.clearEcnetNodes();
  391. return this.editor ? this.editor.getData() : "";
  392. },
  393. /**
  394. * 当表单上没有对应组件的时候,可以使用this.data[fieldId] = data赋值。
  395. * @summary 为组件赋值。
  396. * @param data{String} .
  397. * @example
  398. * this.form.get("fieldId").setData("test"); //赋文本值
  399. * @example
  400. * //如果无法确定表单上是否有组件,需要判断
  401. * if( this.form.get('fieldId') ){ //判断表单是否有无对应组件
  402. * this.form.get('fieldId').setData( data );
  403. * }else{
  404. * this.data['fieldId'] = data;
  405. * }
  406. */
  407. setData: function(data){
  408. this._setBusinessData(data);
  409. if (this.editor) this.editor.setData(data);
  410. },
  411. createErrorNode: function(text){
  412. var node = new Element("div");
  413. var iconNode = new Element("div", {
  414. "styles": {
  415. "width": "20px",
  416. "height": "20px",
  417. "float": "left",
  418. "background": "url("+"../x_component_process_Xform/$Form/default/icon/error.png) center center no-repeat"
  419. }
  420. }).inject(node);
  421. var textNode = new Element("div", {
  422. "styles": {
  423. "line-height": "20px",
  424. "margin-left": "20px",
  425. "color": "red",
  426. "word-break": "keep-all"
  427. },
  428. "text": text
  429. }).inject(node);
  430. return node;
  431. },
  432. notValidationMode: function(text){
  433. if (!this.isNotValidationMode){
  434. this.isNotValidationMode = true;
  435. this.node.store("borderStyle", this.node.getStyles("border-left", "border-right", "border-top", "border-bottom"));
  436. this.node.setStyle("border", "1px solid red");
  437. this.errNode = this.createErrorNode(text).inject(this.node, "after");
  438. this.showNotValidationMode(this.node);
  439. if (!this.node.isIntoView()) this.node.scrollIntoView();
  440. }
  441. },
  442. showNotValidationMode: function(node){
  443. var p = node.getParent("div");
  444. if (p){
  445. if (p.get("MWFtype") == "tab$Content"){
  446. if (p.getParent("div").getStyle("display")=="none"){
  447. var contentAreaNode = p.getParent("div").getParent("div");
  448. var tabAreaNode = contentAreaNode.getPrevious("div");
  449. var idx = contentAreaNode.getChildren().indexOf(p.getParent("div"));
  450. var tabNode = tabAreaNode.getLast().getFirst().getChildren()[idx];
  451. tabNode.click();
  452. p = tabAreaNode.getParent("div");
  453. }
  454. }
  455. this.showNotValidationMode(p);
  456. }
  457. },
  458. validationMode: function(){
  459. if (this.isNotValidationMode){
  460. this.isNotValidationMode = false;
  461. this.node.setStyles(this.node.retrieve("borderStyle"));
  462. if (this.errNode){
  463. this.errNode.destroy();
  464. this.errNode = null;
  465. }
  466. }
  467. },
  468. validationConfigItem: function(routeName, data){
  469. var flag = (data.status=="all") ? true: (routeName == data.decision);
  470. if (flag){
  471. var n = this.getData();
  472. var v = (data.valueType=="value") ? n : n.length;
  473. switch (data.operateor){
  474. case "isnull":
  475. if (!v){
  476. this.notValidationMode(data.prompt);
  477. return false;
  478. }
  479. break;
  480. case "notnull":
  481. if (v){
  482. this.notValidationMode(data.prompt);
  483. return false;
  484. }
  485. break;
  486. case "gt":
  487. if (v>data.value){
  488. this.notValidationMode(data.prompt);
  489. return false;
  490. }
  491. break;
  492. case "lt":
  493. if (v<data.value){
  494. this.notValidationMode(data.prompt);
  495. return false;
  496. }
  497. break;
  498. case "equal":
  499. if (v==data.value){
  500. this.notValidationMode(data.prompt);
  501. return false;
  502. }
  503. break;
  504. case "neq":
  505. if (v!=data.value){
  506. this.notValidationMode(data.prompt);
  507. return false;
  508. }
  509. break;
  510. case "contain":
  511. if (v.indexOf(data.value)!=-1){
  512. this.notValidationMode(data.prompt);
  513. return false;
  514. }
  515. break;
  516. case "notcontain":
  517. if (v.indexOf(data.value)==-1){
  518. this.notValidationMode(data.prompt);
  519. return false;
  520. }
  521. break;
  522. }
  523. }
  524. return true;
  525. },
  526. validationConfig: function(routeName, opinion){
  527. if (this.json.validationConfig){
  528. if (this.json.validationConfig.length){
  529. for (var i=0; i<this.json.validationConfig.length; i++) {
  530. var data = this.json.validationConfig[i];
  531. if (!this.validationConfigItem(routeName, data)) return false;
  532. }
  533. }
  534. return true;
  535. }
  536. return true;
  537. },
  538. /**
  539. * @summary 根据组件的校验设置进行校验。
  540. * @param {String} [routeName] - 可选,路由名称.
  541. * @example
  542. * if( !this.form.get('fieldId').validation() ){
  543. * return false;
  544. * }
  545. * @return {Boolean} 是否通过校验
  546. */
  547. validation: function(routeName, opinion){
  548. if (!this.validationConfig(routeName, opinion)) return false;
  549. if (!this.json.validation) return true;
  550. if (!this.json.validation.code) return true;
  551. this.currentRouteName = routeName;
  552. var flag = this.form.Macro.exec(this.json.validation.code, this);
  553. this.currentRouteName = "";
  554. if (!flag) flag = MWF.xApplication.process.Xform.LP.notValidation;
  555. if (flag.toString()!="true"){
  556. this.notValidationMode(flag);
  557. return false;
  558. }
  559. return true;
  560. }
  561. });