CurdWindowService.js 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675
  1. import SearchFieldService from './SearchFieldService.js';
  2. import CurdWindowResource from '../api/dic/CurdWindowResource.js';
  3. import ProcessReportResource from '../api/dic/ProcessReportResource.js';
  4. import FieldResource from '../api/dic/FieldResource.js';
  5. import SelectFieldService from './SelectFieldService.js';
  6. import EnumSelectFieldService from './EnumSelectFieldService.js';
  7. import Common from '../common/Common.js';
  8. import { Uuid } from 'pc-component-v3';
  9. import { Notify } from 'pc-component-v3';
  10. /**
  11. * CURD 窗口服务类
  12. */
  13. export default {
  14. /**
  15. * 初始化CURD窗口编号
  16. * @param {CURD窗口编号} windowNo
  17. */
  18. initCurdWindow: function (windowNo) {
  19. var _self = this;
  20. var tabIndex = 0;
  21. // 从后台加载CURD窗口的定义
  22. return new Promise((resolve, reject) => {
  23. CurdWindowResource.uniqueByNo(windowNo).then(curdWindowDto => {
  24. if (curdWindowDto != null && curdWindowDto.tabs != null) {
  25. _self.initGridColumns(curdWindowDto, tabIndex).then(successData => {
  26. successData.curdWindowDto = curdWindowDto;
  27. successData.tabDto = curdWindowDto.tabs[tabIndex];
  28. resolve(successData);
  29. }, errorData => {
  30. reject();
  31. });
  32. } else {
  33. reject();
  34. }
  35. }, errorData => {
  36. reject(errorData);
  37. });
  38. });
  39. },
  40. /**
  41. * 初始化表格列
  42. */
  43. initGridColumns: function (curdWindowDto, tabIndex) {
  44. var _self = this;
  45. var tab = curdWindowDto.tabs[tabIndex];
  46. var tabGridFields = tab.tabGridView.tabGridFields;
  47. var promises = [];
  48. // 表格列标题
  49. var colHeaders = [];
  50. // 表格列定义
  51. var columns = [];
  52. for (var columnIndex = 0; columnIndex < tabGridFields.length; columnIndex++) {
  53. var tabGridField = tabGridFields[columnIndex];
  54. // 冗余字段,用于afterChange
  55. tabGridField.windowNo = curdWindowDto.no;
  56. tabGridField.tabIndex = tabIndex;
  57. // 创建列名的集合
  58. var columName = tabGridField.displayName;
  59. colHeaders.push(columName);
  60. // 创建列定义
  61. var displayType = tabGridField.displayType;
  62. var fieldName = tabGridField.fieldName;
  63. var column = null;
  64. if (displayType == 'TextBoxEditor'
  65. || displayType == 'TextAreaEditor'
  66. || displayType == '') {
  67. column = {
  68. type: 'text',
  69. };
  70. } else if (displayType == 'DoubleBoxEditor'
  71. || displayType == 'IntegerBoxEditor'
  72. || displayType == 'LongBoxEditor'
  73. || displayType == 'FloatBoxEditor'
  74. || displayType == 'BigDecimal') {
  75. column = {
  76. type: 'numeric',
  77. };
  78. } else if (displayType == 'PasswordTextBoxEditor') {
  79. column = {
  80. type: 'password',
  81. };
  82. } else if (displayType == 'CheckBoxEditor') {
  83. column = {
  84. type: 'checkbox',
  85. };
  86. } else if (displayType == 'DateBoxEditor') {
  87. column = {
  88. type: 'date',
  89. dateFormat: 'YYYY-MM-DD',
  90. correctFormat: true,
  91. };
  92. } else if (displayType == 'TimeBoxEditor') {
  93. column = {
  94. type: 'time',
  95. timeFormat: 'hh:mm:ss',
  96. correctFormat: true,
  97. };
  98. } else if (displayType == 'DateTimeBoxEditor') {
  99. column = {
  100. type: 'date',
  101. dateFormat: 'YYYY-MM-DD',
  102. correctFormat: true,
  103. };
  104. } else if (displayType == 'RichTextAreaEditor') {
  105. column = {
  106. type: 'html',
  107. };
  108. } else if (displayType == 'ListBoxEditor') {
  109. column = {
  110. type: 'handsontable',
  111. autoColumnSize: true,
  112. observeChanges: true,
  113. };
  114. let promise = SelectFieldService.loadSelectData(column, infoWindowNo, tabGridField);
  115. promises.push(promise);
  116. } else if (displayType == 'SearchBoxEditor') {
  117. column = {
  118. type: 'handsontable',
  119. autoColumnSize: true,
  120. observeChanges: true,
  121. };
  122. var infoWindowNo = tabGridField.infoWindowNo;
  123. let promise = SearchFieldService.loadInfoWindowData(column, infoWindowNo, tabGridField);
  124. promises.push(promise);
  125. } else if (displayType == 'ButtonEditor') {
  126. column = {
  127. type: 'text',
  128. };
  129. } else if (displayType == 'ListBoxEnumEditor') {
  130. column = {
  131. type: 'handsontable',
  132. autoColumnSize: true,
  133. observeChanges: true,
  134. };
  135. let promise = EnumSelectFieldService.loadSelectData(column, infoWindowNo, tabGridField);
  136. promises.push(promise);
  137. } else if (displayType == 'ImageEditor') {
  138. column = {
  139. type: 'text',
  140. };
  141. } else if (displayType == 'ImageListEditor') {
  142. column = {
  143. type: 'text',
  144. };
  145. } else if (displayType == 'RadioButtonGroupEditor') {
  146. column = {
  147. type: 'select',
  148. };
  149. } else if (displayType == 'Label') {
  150. column = {
  151. type: 'text',
  152. };
  153. } else if (displayType == 'Video') {
  154. column = {
  155. type: 'text',
  156. };
  157. } else if (displayType == 'File') {
  158. column = {
  159. type: 'text',
  160. };
  161. } else if (displayType == 'EnumMultiSelect') {
  162. column = {
  163. type: 'text',
  164. };
  165. } else if (displayType == 'ManyToManySetBoxEditor') {
  166. column = {
  167. type: 'text',
  168. };
  169. } else if (displayType == 'RedGreenEditor') {
  170. column = {
  171. type: 'text',
  172. };
  173. } else if (displayType == 'YearEditor') {
  174. column = {
  175. type: 'text',
  176. };
  177. } else if (displayType == 'YearMonthEditor') {
  178. column = {
  179. type: 'text',
  180. };
  181. }
  182. column.data = _self.curdWindowFieldValue(fieldName, tabGridField);
  183. column.readOnly = tabGridField.readOnly;
  184. column.readOnlyCellClassName = 'handsontable-cell-readonly';
  185. if (tabGridField.mandatory) {
  186. column.validator = _self.mandatoryValidator;
  187. }
  188. // bug fixed by jack,允许删除
  189. // column.allowInvalid = false;
  190. column.allowInvalid = true;
  191. column.required = tabGridField.mandatory;
  192. columns.push(column);
  193. }
  194. let data = {
  195. colHeaders: colHeaders,
  196. columns: columns,
  197. };
  198. if (promises.length > 0) {
  199. return new Promise((resolve, reject) => {
  200. Promise.all(promises).then(successData => {
  201. resolve(data);
  202. }, errorData => {
  203. reject();
  204. });
  205. });
  206. } else {
  207. return new Promise((resolve, reject) => {
  208. resolve(data);
  209. });
  210. }
  211. },
  212. /**
  213. * 强制字段验证
  214. * @param {*} value
  215. * @param {*} callback
  216. */
  217. mandatoryValidator: function (value, callback) {
  218. if (value == null || value.length == 0) {
  219. callback(false);
  220. } else {
  221. callback(true);
  222. }
  223. },
  224. /**
  225. * CURD窗口 HandsonTable 获取列的显示内容
  226. */
  227. curdWindowFieldValue: function (attr, tabGridField) {
  228. return function (row, value) {
  229. if (row == '##getColumnDef##' && value == row) {
  230. return tabGridField;
  231. }
  232. if (attr == 'id') {
  233. return row.id;
  234. } else {
  235. if (row.data == null || row.data[attr] == null) {
  236. return null;
  237. } else {
  238. // TODO
  239. if (row.data[attr].displayValue.length > tabGridField.entityFieldIndex) {
  240. var key = row.data[attr].displayValue[tabGridField.entityFieldIndex];
  241. if (tabGridField.displayType == 'ListBoxEnumEditor') {
  242. if (tabGridField.keyValues == null || tabGridField.keyValues.length == 0) {
  243. return null;
  244. } else {
  245. let value1 = null;
  246. tabGridField.keyValues.forEach(keyValue => {
  247. if (keyValue.keyStr == key) {
  248. value1 = keyValue.value;
  249. }
  250. });
  251. return value1;
  252. }
  253. } else {
  254. return key;
  255. }
  256. } else {
  257. return '';
  258. }
  259. }
  260. }
  261. };
  262. },
  263. /**
  264. * 表格中单元格的数据发生改变
  265. * @param {*} changes 单元格和变化前后的数值
  266. * @param {*} source 数据变化的种类
  267. */
  268. afterChange: function (hot, changes, source) {
  269. if (source === 'loadData') {
  270. return; //don't save this change
  271. }
  272. if (changes == null || changes.length == 0) {
  273. return;
  274. }
  275. var _self = this;
  276. function toString(fieldValue) {
  277. if (fieldValue == null) {
  278. return null;
  279. } else {
  280. return fieldValue.id + ',' + fieldValue.displayName + ',' + fieldValue.fieldType;
  281. }
  282. }
  283. function isFieldValue(fieldValue) {
  284. if (fieldValue == null) {
  285. return false;
  286. } else {
  287. if (fieldValue.fieldType == 'String' || fieldValue.fieldType == 'Key' || fieldValue.fieldType == 'ManyToManyKey') {
  288. return true;
  289. } else {
  290. return false;
  291. }
  292. }
  293. }
  294. changes.forEach(([row, prop, oldValue, newValue]) => {
  295. if (oldValue == newValue) {
  296. return;
  297. }
  298. var tabGridField = prop('##getColumnDef##', '##getColumnDef##');
  299. if (tabGridField.entityFieldIndex >= 1) {
  300. return;
  301. }
  302. var newFieldValue = {
  303. id: '',
  304. displayValue: [],
  305. fieldType: 'Key',
  306. };
  307. var displayType = tabGridField.displayType;
  308. switch (displayType) {
  309. case 'ID':
  310. case 'TextBoxEditor':
  311. case 'DoubleBoxEditor':
  312. case 'IntegerBoxEditor':
  313. case 'LongBoxEditor':
  314. case 'FloatBoxEditor':
  315. case 'PasswordTextBoxEditor':
  316. case 'TextAreaEditor':
  317. case 'RichTextAreaEditor':
  318. case 'CheckBoxEditor':
  319. case 'TimeBoxEditor':
  320. case 'DateTimeBoxEditor':
  321. case 'ButtonEditor':
  322. case 'ImageEditor':
  323. case 'ImageListEditor':
  324. case 'Canvas':
  325. case 'RadioButtonGroupEditor':
  326. case 'Label':
  327. case 'Video':
  328. case 'File':
  329. case 'BigDecimalEditor':
  330. case 'EnumMultiSelect':
  331. case 'ManyToManySetBoxEditor':
  332. case 'RedGreenEditor':
  333. case 'YearEditor':
  334. case 'YearMonthEditor': {
  335. newFieldValue.fieldType = 'String';
  336. newFieldValue.displayValue[0] = newValue;
  337. break;
  338. }
  339. case 'DateBoxEditor': {
  340. newFieldValue.fieldType = 'String';
  341. var newValueFormat = moment(newValue).format('YYYY-MM-DD');
  342. if (newValueFormat == 'Invalid date') {
  343. Notify.error('错误', '当前时间格式不正确,请重新输入。', 3000);
  344. newFieldValue.displayValue[0] = newValue;
  345. } else {
  346. newFieldValue.displayValue[0] = newValueFormat;
  347. }
  348. break;
  349. }
  350. case 'ListBoxEnumEditor': {
  351. if (newValue == null || newValue == '') {
  352. newFieldValue.fieldType = 'String';
  353. newFieldValue.displayValue = [];
  354. } else {
  355. newFieldValue.fieldType = 'String';
  356. newFieldValue.displayValue = [newValue];
  357. }
  358. break;
  359. }
  360. case 'ListBoxEditor':
  361. if (newValue == null || newValue == '') {
  362. newFieldValue.fieldType = 'Key';
  363. newFieldValue.id = newValue;
  364. newFieldValue.displayValue = [];
  365. } else {
  366. newFieldValue.fieldType = 'Key';
  367. var newValueNumber = Number(newValue);
  368. newFieldValue.id = newValueNumber;
  369. if (tabGridField.keyValues == null || tabGridField.keyValues.length == 0) {
  370. newFieldValue.displayValue = [];
  371. } else {
  372. tabGridField.keyValues.forEach(keyValue => {
  373. if (keyValue.key == newValueNumber) {
  374. newFieldValue.displayValue = [keyValue.value];
  375. }
  376. });
  377. }
  378. }
  379. break;
  380. case 'SearchBoxEditor': {
  381. if (newValue == null || newValue == '') {
  382. newFieldValue.fieldType = 'Key';
  383. newFieldValue.id = newValue;
  384. } else {
  385. newFieldValue.fieldType = 'Key';
  386. newFieldValue.id = newValue;
  387. var requestUrl = 'FieldResource/getFieldValue1?';
  388. requestUrl += ('&windowNo=' + tabGridField.windowNo);
  389. requestUrl += ('&tabIndex=' + tabGridField.tabIndex);
  390. requestUrl += ('&fieldName=' + tabGridField.fieldName);
  391. requestUrl += ('&recordId=' + newValue);
  392. $.ajax({
  393. url: Common.getApiURL(requestUrl),
  394. type: 'get',
  395. dataType: 'json',
  396. async: false, // 设置成同步
  397. beforeSend: function (request) {
  398. Common.addTokenToRequest(request);
  399. },
  400. success: function (data) {
  401. newFieldValue.displayValue = data;
  402. },
  403. error: function (XMLHttpRequest, textStatus, errorThrown) {
  404. console.log(XMLHttpRequest);
  405. // Common.processException(XMLHttpRequest);
  406. },
  407. });
  408. }
  409. break;
  410. }
  411. }
  412. var rowData = hot.getSourceDataAtRow(row);
  413. if (rowData.data == null) {
  414. rowData.data = {};
  415. rowData.id = -1;
  416. }
  417. if (row.id == null) {
  418. rowData.id = -1;
  419. }
  420. rowData.data[tabGridField.fieldName] = newFieldValue;
  421. // 执行callout
  422. if (tabGridField.calloutProcessReportNo != undefined && tabGridField.calloutProcessReportNo != null) {
  423. _self.callout(hot, tabGridField.calloutProcessReportNo, rowData);
  424. }
  425. });
  426. },
  427. /**
  428. * 运行Callout
  429. * @param {流程报表编号} calloutProcessReportNo
  430. * @param {数据} modelData
  431. * @param {编号} index
  432. */
  433. callout: function (hot, calloutProcessReportNo, rowData) {
  434. var _self = this;
  435. // 查询流程和报表的定义
  436. ProcessReportResource.uniqueByNo(calloutProcessReportNo).then(successData => {
  437. if (successData == null) {
  438. return;
  439. }
  440. var modelData = {};
  441. modelData.id = rowData.id;
  442. modelData.data = rowData.data;
  443. modelData.changed = true;
  444. // 执行服务端的脚本
  445. ProcessReportResource.runCallout(calloutProcessReportNo, modelData).then(successData => {
  446. if (successData && successData.modelData) {
  447. var newModelData = successData.modelData;
  448. var newModelDataData = newModelData.data;
  449. for (let [key, value] of Object.entries(newModelDataData)) {
  450. rowData.data[key] = value;
  451. }
  452. hot.render();
  453. }
  454. }, errorData => {
  455. Common.processException(errorData);
  456. });
  457. }, errorData => {
  458. Common.processException(errorData);
  459. });
  460. },
  461. /**
  462. * 复制之前完成
  463. * @param {WEB表格} hot
  464. * @param {数据} data
  465. * @param {位置} coords
  466. */
  467. beforeCopy: function (hot, tabDto, data, coords) {
  468. // data -> [[1, 2, 3], [4, 5, 6]]
  469. // coords -> [{startRow: 0, startCol: 0, endRow: 1, endCol: 2}]
  470. var coord = coords[0];
  471. for (var row = coord.startRow; row <= coord.endRow; row++) {
  472. // 取出该行的数据
  473. var rowData = hot.getSourceDataAtRow(row);
  474. for (var col = coord.startCol; col <= coord.endCol; col++) {
  475. if (col < tabDto.tabGridView.tabGridFields.length) {
  476. var tabGridField = tabDto.tabGridView.tabGridFields[col];
  477. var displayType = tabGridField.displayType;
  478. // 对下拉框和搜索框类型的数据进行替换。
  479. switch (displayType) {
  480. //TODO
  481. case 'ListBoxEnumEditor': {
  482. let tempData = rowData.data[tabGridField.fieldName];
  483. if (tabGridField.keyValues != null
  484. && tempData != null
  485. && tempData.displayValue != null
  486. && tempData.displayValue.length > 0) {
  487. var value = tempData.displayValue[0];
  488. data[row - coord.startRow][col - coord.startCol] = value;
  489. } else {
  490. data[row - coord.startRow][col - coord.startCol] = null;
  491. }
  492. break;
  493. }
  494. case 'ListBoxEditor':
  495. case 'SearchBoxEditor': {
  496. let tempData = rowData.data[tabGridField.fieldName];
  497. if (tempData != null) {
  498. data[row - coord.startRow][col - coord.startCol] = tempData.id;
  499. } else {
  500. data[row - coord.startRow][col - coord.startCol] = null;
  501. }
  502. break;
  503. }
  504. }
  505. }
  506. }
  507. }
  508. },
  509. /**
  510. * 获取右击菜单
  511. */
  512. getContextMenu: function (hot, tabDto) {
  513. var contextMenu = {
  514. items: {
  515. 'removeRow': {
  516. name: '删除行',
  517. },
  518. 'openLinkWindow': {
  519. name: '新建数据',
  520. },
  521. 'viewInLinkWindow': {
  522. name: '查看数据',
  523. },
  524. 'editInLinkWindow': {
  525. name: '编辑数据',
  526. },
  527. 'refreshDropDown': {
  528. name: '刷新',
  529. },
  530. },
  531. callback: function (key, options) {
  532. var selected = hot.getSelected();
  533. function sortId(a, b) {
  534. return a.start.row - b.start.row;//由低到高
  535. }
  536. if (selected != null && selected.length > 0) {
  537. var row = selected[0][0];
  538. var col = selected[0][1];
  539. // 删除行
  540. if (key === 'removeRow') {
  541. let row1 = options;
  542. var num = 0;
  543. var aa = hot.getDataAtRow(0);
  544. if (row1.length > 0) {
  545. //将选中的值按照行号排序
  546. row1.sort(sortId);
  547. for (var i = 0; i < row1.length; i++) {
  548. var item = row1[i];
  549. var start = item.start.row - num;
  550. var end = item.end.row - num - start + 1;
  551. hot.alter('remove_row', start, end);
  552. num += end;
  553. }
  554. }
  555. } else if (key == 'openLinkWindow') {
  556. // 获取列的定义
  557. let tabGridField = tabDto.tabGridView.tabGridFields[col];
  558. // 进行界面的跳转
  559. let linkCurdWindowNo = tabGridField.linkCurdWindowNo;
  560. if (linkCurdWindowNo != null) {
  561. window.open(Common.getRedirectUrl('#/desktop/window/window-edit/create/'
  562. + linkCurdWindowNo
  563. + '/0?currPage=1&totalCount=4&saveClose=true&uuid='
  564. + Uuid.createUUID()));
  565. }
  566. } else if (key == 'viewInLinkWindow') {
  567. let data = hot.getSourceDataAtRow(row);
  568. if (data == null || data.id == null) {
  569. return;
  570. }
  571. // 编辑链接列
  572. // 获取列的定义
  573. let tabGridField = tabDto.tabGridView.tabGridFields[col];
  574. let fieldName = tabGridField.fieldName;
  575. if (data.data[fieldName] == null || data.data[fieldName].id == null) {
  576. return;
  577. }
  578. // 进行界面的跳转
  579. let linkCurdWindowNo = tabGridField.linkCurdWindowNo;
  580. if (linkCurdWindowNo != null) {
  581. window.open(Common.getRedirectUrl('#/desktop/window/window-read/view/'
  582. + linkCurdWindowNo
  583. + '/0/'
  584. + data.data[fieldName].id
  585. + '?currIndex=1&totalCount=1&saveClose=true&uuid=' + Uuid.createUUID()));
  586. }
  587. } else if (key == 'editInLinkWindow') {
  588. let data = hot.getSourceDataAtRow(row);
  589. if (data == null || data.id == null) {
  590. return;
  591. }
  592. // 编辑链接列
  593. // 获取列的定义
  594. let tabGridField = tabDto.tabGridView.tabGridFields[col];
  595. let fieldName = tabGridField.fieldName;
  596. if (data.data[fieldName] == null || data.data[fieldName].id == null) {
  597. return;
  598. }
  599. // 进行界面的跳转
  600. var linkCurdWindowNo = tabGridField.linkCurdWindowNo;
  601. if (linkCurdWindowNo != null) {
  602. window.open(Common.getRedirectUrl('#/desktop/window/window-edit/edit/'
  603. + linkCurdWindowNo
  604. + '/0/'
  605. + data.data[fieldName].id
  606. + '?currIndex=1&currIndex=1&totalCount=1&saveClose=true&uuid='
  607. + Uuid.createUUID()));
  608. }
  609. } else if (key == 'refreshDropDown') {
  610. let data = hot.getSourceDataAtRow(row);
  611. // 获取列的定义
  612. let tabGridField = tabDto.tabGridView.tabGridFields[col];
  613. if (tabGridField.displayType == 'ListBoxEditor') {
  614. var modelData = {};
  615. modelData.id = data.id;
  616. modelData.data = data.data;
  617. modelData.changed = true;
  618. modelData.windowNo = tabGridField.windowNo;
  619. modelData.tabIndex = 0;
  620. SelectFieldService.refreshField(tabGridField, tabGridField.windowNo, 0, modelData);
  621. }
  622. }
  623. }
  624. },
  625. };
  626. return contextMenu;
  627. },
  628. };