ledger_check_modal.ejs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. <!--数据检查-->
  2. <div class="modal fade" id="ledger-check-modal" data-backdrop="static">
  3. <div class="modal-dialog" role="document">
  4. <div class="modal-content">
  5. <div class="modal-header">
  6. <h5 class="modal-title">数据检查</h5>
  7. </div>
  8. <div class="modal-body">
  9. <p>数据检查,将检查罗列台帐中以下内容:</p>
  10. <div class="card mb-2 p-2 border-success" id="check-sibling">
  11. <div class="d-flex justify-content-between">
  12. 项目节、清单同层
  13. <span class="text-muted" name="check-status">待检查</span>
  14. </div>
  15. </div>
  16. <div class="card mb-2 p-2 border-success" id="check-empty-code">
  17. <div class="d-flex justify-content-between">
  18. 项目节、清单编号同时为空
  19. <span class="text-success" title="完成" name="check-status"><i class="fa fa-check"></i></span>
  20. </div>
  21. </div>
  22. <div class="card mb-2 p-2 border-success" id="check-calc">
  23. <div class="d-flex justify-content-between">
  24. 清单数量不等于计量单元之和
  25. <span class="text-success" title="完成" name="check-status"><i class="fa fa-check"></i></span>
  26. </div>
  27. </div>
  28. <div class="card mb-2 p-2 border-success" id="check-zero">
  29. <div class="d-flex justify-content-between">
  30. 清单数量或单价为0
  31. <span class="text-success" title="完成" name="check-status"><i class="fa fa-check"></i></span>
  32. </div>
  33. </div>
  34. <div class="card mb-2 p-2 border-success" id="check-tp">
  35. <div class="d-flex justify-content-between">
  36. 清单金额≠数量×单价
  37. <span class="text-success" title="完成" name="check-status"><i class="fa fa-check"></i></span>
  38. </div>
  39. </div>
  40. <a href="javascript: void(0);" class="btn btn-sm btn-block btn-primary" id="ledger-check-begin">开始检查</a>
  41. <a href="#" class="btn btn-sm btn-block btn-primary disabled" id="ledger-check-waiting">检查中,请等待...</a>
  42. <p class="text-center text-success" id="ledger-check-hint">检查完成,现在您可以查看结果。</p>
  43. </div>
  44. <div class="modal-footer">
  45. <button type="button" class="btn btn-sm btn-secondary" data-dismiss="modal">关闭</button>
  46. <!--检查完有结果显示按钮-->
  47. <button type="button" class="btn btn-sm btn-primary" id="ledger-check-show">查看结果</button>
  48. </div>
  49. </div>
  50. </div>
  51. </div>
  52. <script>
  53. const ledgerCheckType = {
  54. sibling: {value: 1, text: '项目节、清单同层'},
  55. empty_code: {value: 2, text: '项目节、清单编号同时为空'},
  56. calc: {value: 3, text: '清单数量不等于计量单元之和'},
  57. zero: {value: 4, text: '清单数量或单价为0'},
  58. tp: {value: 5, text: '清单金额≠数量×单价'},
  59. };
  60. const LedgerChecker = function (setting) {
  61. const ledger = setting.ledgerTree, ledgerPos = setting.ledgerPos, decimal = setting.decimal;
  62. const initWaitingModal = function () {
  63. $('.card', '#ledger-check-modal').removeClass('border-success');
  64. $('[name=check-status]', '#ledger-check-modal').removeClass('text-success').addClass('text-muted').html('待检查');
  65. $('#ledger-check-begin').show();
  66. $('#ledger-check-waiting').hide();
  67. $('#ledger-check-hint').removeClass('text-warning').addClass('text-success').html('检查完成,现在您可以查看结果。').hide();
  68. $('#ledger-check-show').hide();
  69. }
  70. const doSomeCheck = function (selector, checkFun) {
  71. const checkStatus = $('[name=check-status]', selector);
  72. checkStatus.html('<i class="fa fa-spinner fa-spin"></i>');
  73. const result = checkFun(ledger);
  74. checkStatus.removeClass('text-muted').addClass('text-success').html('<i class="fa fa-check"></i>');
  75. $(selector).addClass('border-success');
  76. return result;
  77. }
  78. const ledgerCheckUtil = {
  79. checkSibling: function (ledgerTree) {
  80. const error = [];
  81. for (const node of ledgerTree.nodes) {
  82. if (!node.children || node.children.length === 0) continue;
  83. let hasXmj, hasGcl;
  84. for (const child of node.children) {
  85. if (child.b_code) hasXmj = true;
  86. if (!child.b_code) hasGcl = true;
  87. }
  88. if (hasXmj && hasGcl) error.push(node);
  89. }
  90. return error;
  91. },
  92. checkCodeEmpty: function (ledgerTree) {
  93. const error = [];
  94. const checkNodeCode = function (node) {
  95. if ((!node.code || node.code === '') && (!node.b_code || node.b_code === '')) error.push(node);
  96. if (node.children && node.children.length > 0) {
  97. for (const child of node.children) {
  98. checkNodeCode(child);
  99. }
  100. }
  101. }
  102. for (const topLevel of ledgerTree.children) {
  103. if (topLevel.node_type !== 1) continue;
  104. checkNodeCode(topLevel);
  105. }
  106. return error;
  107. },
  108. checkCalc: function (ledgerTree) {
  109. const error = [];
  110. for (const node of ledgerTree.nodes) {
  111. if (node.children && node.children.length > 0) continue;
  112. const nodePos = ledgerPos.getLedgerPos(node.id);
  113. if (!nodePos || nodePos.length === 0) continue;
  114. const checkData = {
  115. sgfh_qty: node.sgfh_qty || 0, qtcl_qty: node.qtcl_qty || 0, sjcl_qty: node.sjcl_qty || 0,
  116. quantity: node.quantity || 0
  117. };
  118. const calcData = {
  119. sgfh_qty: 0, qtcl_qty: 0, sjcl_qty: 0,
  120. quantity: 0
  121. };
  122. for (const np of nodePos) {
  123. calcData.sgfh_qty = ZhCalc.add(calcData.sgfh_qty, np.sgfh_qty) || 0;
  124. calcData.qtcl_qty = ZhCalc.add(calcData.qtcl_qty, np.qtcl_qty) || 0;
  125. calcData.sjcl_qty = ZhCalc.add(calcData.sjcl_qty, np.sjcl_qty) || 0;
  126. calcData.quantity = ZhCalc.add(calcData.quantity, np.quantity) || 0;
  127. }
  128. if (!_.isMatch(checkData, calcData)) error.push(node);
  129. }
  130. return error;
  131. },
  132. checkZero: function (ledgerTree) {
  133. const error = [];
  134. for (const node of ledgerTree.nodes) {
  135. if ((!node.b_code || node.b_code === '')) continue;
  136. if (node.children && node.children.length > 0) continue;
  137. if ((checkZero(node.sgfh_qty) && checkZero(node.qtcl_qty) && checkZero(node.sjcl_qty)
  138. && checkZero(node.deal_qty) && checkZero(node.quantity))
  139. || checkZero(node.unit_price)) error.push(node);
  140. }
  141. return error;
  142. },
  143. checkTp: function (ledgerTree) {
  144. const error = [];
  145. for (const node of ledgerTree.nodes) {
  146. if (node.children && node.children.length > 0) continue;
  147. const checkData = {
  148. sgfh_tp: node.sgfh_tp || 0, qtcl_tp: node.qtcl_tp || 0, sjcl_tp: node.sjcl_tp || 0,
  149. total_price: node.total_price || 0, deal_tp: node.deal_tp || 0,
  150. };
  151. const calcData = {
  152. sgfh_tp: ZhCalc.mul(node.unit_price, node.sgfh_qty, decimal.tp) || 0,
  153. qtcl_tp: ZhCalc.mul(node.unit_price, node.qtcl_qty, decimal.tp) || 0,
  154. sjcl_tp: ZhCalc.mul(node.unit_price, node.sjcl_qty, decimal.tp) || 0,
  155. total_price: ZhCalc.mul(node.unit_price, node.quantity, decimal.tp) || 0,
  156. deal_tp: ZhCalc.mul(node.unit_price, node.deal_qty, decimal.tp) || 0,
  157. };
  158. if (!_.isMatch(checkData, calcData)) error.push(node);
  159. }
  160. return error;
  161. }
  162. };
  163. const assignWarningData = function (nodes, checkType, warningData) {
  164. for (const node of nodes) {
  165. warningData.push({
  166. type: checkType,
  167. ledger_id: node.ledger_id,
  168. code: node.code,
  169. b_code: node.b_code,
  170. name: node.name,
  171. })
  172. }
  173. }
  174. const checkData = function () {
  175. $('#ledger-check-begin').hide();
  176. $('#ledger-check-waiting').show();
  177. const checkData = {
  178. check_time: new Date(),
  179. warning_data: [],
  180. }
  181. const sibling = doSomeCheck('#check-sibling', ledgerCheckUtil.checkSibling) || [];
  182. assignWarningData(sibling, ledgerCheckType.sibling.value, checkData.warning_data);
  183. const empty_code = doSomeCheck('#check-empty-code', ledgerCheckUtil.checkCodeEmpty) || [];
  184. assignWarningData(empty_code, ledgerCheckType.empty_code.value, checkData.warning_data);
  185. const calc = doSomeCheck('#check-calc', ledgerCheckUtil.checkCalc) || [];
  186. assignWarningData(calc, ledgerCheckType.calc.value, checkData.warning_data);
  187. const zero = doSomeCheck('#check-zero', ledgerCheckUtil.checkZero) || [];
  188. assignWarningData(zero, ledgerCheckType.zero.value, checkData.warning_data);
  189. const tp = doSomeCheck('#check-tp', ledgerCheckUtil.checkTp) || [];
  190. assignWarningData(tp, ledgerCheckType.tp.value, checkData.warning_data);
  191. $('#ledger-check-waiting').hide();
  192. setting.checkList.clearCheckData();
  193. if (checkData.warning_data.length > 0) {
  194. $('#ledger-check-hint').removeClass('text-success').addClass('text-warning').html('检查完成,发现问题,请查阅检查结果。').show();
  195. $('#ledger-check-show').show();
  196. setting.checkList.loadCheckData(checkData);
  197. } else {
  198. setting.checkList.hide();
  199. $('#ledger-check-show').hide();
  200. $('#ledger-check-hint').html('检查完成,没有任何问题。').show();
  201. }
  202. }
  203. $('#ledger-check-begin').bind('click', checkData);
  204. $('#ledger-check-modal').bind('show.bs.modal', initWaitingModal);
  205. $('#ledger-check-show').bind('click', function () {
  206. $('#ledger-check-modal').modal('hide');
  207. setting.checkList.show();
  208. });
  209. }
  210. </script>