print-template-wl.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. <template>
  2. <ele-modal
  3. title="入库单"
  4. :visible.sync="QRvisible"
  5. v-if="QRvisible"
  6. width="80%"
  7. >
  8. <div
  9. id="printSection"
  10. style="
  11. display: flex;
  12. align-items: center;
  13. justify-content: center;
  14. flex-direction: column;
  15. "
  16. >
  17. <div
  18. style="
  19. text-align: center;
  20. font-size: 20px;
  21. font-weight: 800;
  22. text-decoration: underline;
  23. margin-bottom: 10px;
  24. "
  25. >{{ groupName + '入库单' || '湖南万隆智能科技有限公司采购入库单' }}</div
  26. >
  27. <div
  28. style="
  29. display: flex;
  30. justify-content: space-between;
  31. width: 100%;
  32. min-width: 800px;
  33. margin-bottom: 10px;
  34. font-size: 13px;
  35. "
  36. >
  37. <span style="flex: 2; white-space: nowrap; padding-right: 10px"
  38. >供应商:{{ formData.supplierName }}</span
  39. >
  40. <span style="flex: 1; white-space: nowrap; padding: 0 10px"
  41. >日期:{{ formData.receiveDate }}</span
  42. >
  43. <span style="flex: 2; white-space: nowrap; padding-left: 10px"
  44. >编号:{{ formData.receiveNo }}</span
  45. >
  46. </div>
  47. <div
  48. style="
  49. display: flex;
  50. justify-content: space-between;
  51. width: 100%;
  52. margin-bottom: 10px;
  53. font-size: 13px;
  54. "
  55. >
  56. <span style="flex: 1">摘要:</span>
  57. </div>
  58. <table
  59. cellspacing="0"
  60. border
  61. style="
  62. width: 100%;
  63. table-layout: fixed;
  64. word-break: break-all;
  65. word-wrap: break-word;
  66. font-size: 12px;
  67. margin-bottom: 10px;
  68. "
  69. >
  70. <tbody>
  71. <tr align="center">
  72. <td style="padding: 5px" width="50px"> 编号 </td>
  73. <td style="padding: 5px"> 物料编码 </td>
  74. <td style="padding: 5px"> 物料名称 </td>
  75. <td style="padding: 5px">规格型号 </td>
  76. <td style="padding: 5px">单位 </td>
  77. <td style="padding: 5px"> 数量</td>
  78. <td style="padding: 5px"> 单价</td>
  79. <td style="padding: 5px"> 金额</td>
  80. <td style="padding: 5px"> 批次号</td>
  81. <td style="padding: 5px"> 备注</td>
  82. </tr>
  83. <tr align="center" v-for="(item, index) in formData?.productList">
  84. <td style="padding: 5px"> {{ index + 1 }} </td>
  85. <td style="padding: 5px"> {{ item.productCode }} </td>
  86. <td style="padding: 5px"> {{ item.productName }} </td>
  87. <td style="padding: 5px">
  88. {{ item.specification || ''
  89. }}{{ item.modelType ? '/' + item.modelType : '' }}
  90. </td>
  91. <td style="padding: 5px"> {{ item.measuringUnit }}</td>
  92. <td style="padding: 5px"> {{ item.totalCount }}</td>
  93. <td style="padding: 5px"> {{ item.singlePrice }}</td>
  94. <td style="padding: 5px"> {{ item.totalPrice }}</td>
  95. <td style="padding: 5px"> {{ item.batchNo }}</td>
  96. <td style="padding: 5px"> {{ item.remark }}</td>
  97. </tr>
  98. <tr align="center">
  99. <td style="padding: 5px" colspan="2">合计 </td>
  100. <td style="padding: 5px; text-align: left;" colspan="8">
  101. <div style="display: flex;">
  102. <span style="flex: 1"
  103. >本页数量小计:{{ this.getAll('totalCount') }}</span
  104. >
  105. <span style="flex: 1"
  106. >本页金额小计:{{ this.getAll('totalPrice') }}</span
  107. >
  108. <span style="flex: 1"
  109. >汇总数量:{{ this.getAll('totalCount') }}</span
  110. >
  111. <span style="flex: 1"
  112. >汇总金额:{{ this.getAll('totalPrice') }}</span
  113. >
  114. </div>
  115. </td>
  116. </tr>
  117. <tr align="center">
  118. <td style="padding: 5px" colspan="2">合计金额(大写金额) </td>
  119. <td style="padding: 5px; text-align: left;" colspan="8">
  120. <div style="display: flex; margin-top: 6px;">
  121. <span style="flex: 1"
  122. >{{ convertToChinese(getAll('totalPrice')) }}</span
  123. >
  124. </div>
  125. </td>
  126. </tr>
  127. </tbody>
  128. </table>
  129. <div style="display: flex; justify-content: space-between; width: 100%; font-size: 13px;">
  130. <span style="flex: 1">收货单审核人:{{formData.reviewerName}}</span>
  131. <span style="flex: 1">采购发起人:{{formData.purchaseInitiatorName}}</span>
  132. <span style="flex: 1">质检人:{{formData.qualityName}}</span>
  133. <span style="flex: 1">仓管员:{{formData.warehouseKeeperName}}</span>
  134. <!-- <span style="flex: 1">采购收货发起人:{{ formData.purchaseInitiatorName }}</span> -->
  135. </div>
  136. </div>
  137. <div slot="footer">
  138. <el-button @click="exportExcel">导出</el-button>
  139. <el-button @click="print">打印预览</el-button>
  140. <el-button @click="close">关闭</el-button>
  141. </div>
  142. </ele-modal>
  143. </template>
  144. <script>
  145. import { getReceiveSaleOrderrecordDetail } from '@/api/purchasingManage/purchaseorderreceive';
  146. import { formatPrice } from '@/BIZComponents/setProduct.js';
  147. import { mapGetters } from 'vuex';
  148. import { saveAs } from 'file-saver';
  149. import { convertToChinese } from '@/utils/util';
  150. export default {
  151. name: 'print',
  152. computed: {
  153. ...mapGetters(['user'])
  154. },
  155. props: {
  156. groupName: ''
  157. },
  158. data() {
  159. return {
  160. QRvisible: false,
  161. formData: {}
  162. };
  163. },
  164. methods: {
  165. convertToChinese,
  166. numToChinese(num) {
  167. console.log('num~~~', num);
  168. const digits = ['零', '壹', '贰', '叁', '肆', '伍', '陆', '柒', '捌', '玖'];
  169. const units = ['', '拾', '佰', '仟', '万', '拾', '佰', '仟', '亿'];
  170. const decimalUnits = ['角', '分'];
  171. num = Number(num).toFixed(2);
  172. const [integerStr, decimalStr] = num.split('.');
  173. let result = '';
  174. const integerLen = integerStr.length;
  175. if (integerLen === 0 || (integerLen === 1 && integerStr === '0')) {
  176. result += '零元';
  177. } else {
  178. for (let i = 0; i < integerLen; i++) {
  179. const digit = parseInt(integerStr[i]);
  180. const unitIndex = integerLen - 1 - i;
  181. if (digit === 0) {
  182. if (i < integerLen - 1 && parseInt(integerStr[i + 1]) !== 0) {
  183. result += '零';
  184. }
  185. } else {
  186. if (!(unitIndex === 1 && digit === 1 && i === 0)) {
  187. result += digits[digit];
  188. }
  189. result += units[unitIndex];
  190. }
  191. }
  192. result += '元';
  193. }
  194. const jiao = parseInt(decimalStr[0]);
  195. const fen = parseInt(decimalStr[1]);
  196. if (jiao === 0 && fen === 0) {
  197. result += '整';
  198. } else {
  199. if (jiao !== 0) result += digits[jiao] + decimalUnits[0];
  200. if (fen !== 0) result += digits[fen] + decimalUnits[1];
  201. }
  202. return result;
  203. },
  204. async open(id) {
  205. this.formData = await getReceiveSaleOrderrecordDetail(id);
  206. this.QRvisible = true;
  207. },
  208. close() {
  209. this.QRvisible = false;
  210. },
  211. getAll(key) {
  212. let num = 0;
  213. this.formData.productList.forEach((item) => {
  214. if (item[key]) {
  215. num += Number(item[key]);
  216. }
  217. });
  218. return num ? formatPrice(num) : '';
  219. },
  220. print() {
  221. const printSection = document.getElementById('printSection');
  222. // 创建打印任务
  223. const printWindow = window.open('', '_blank');
  224. printWindow.document.open();
  225. printWindow.document.write('<html><head><title>打印预览</title>');
  226. printWindow.document.write(
  227. '<link rel="stylesheet" href="your-stylesheet-url.css" type="text/css" />'
  228. );
  229. printWindow.document.write('</head><body>');
  230. printWindow.document.write(printSection.innerHTML);
  231. printWindow.document.write('</body></html>');
  232. printWindow.document.close();
  233. printWindow.onload = function () {
  234. printWindow.print();
  235. };
  236. },
  237. exportExcel() {
  238. const list = this.formData?.productList || [];
  239. const totalCount = this.getAll('totalCount');
  240. const totalPrice = this.getAll('totalPrice');
  241. const border = 'border-top:1px solid #000;border-right:1px solid #000;border-bottom:1px solid #000;border-left:1px solid #000;';
  242. const cellBase = `${border}padding:4px;mso-number-format:"\\@";vertical-align:middle;text-align:center;mso-horizontal-align:center;`;
  243. const th = `${cellBase}font-weight:bold;font-family:'宋体';font-size:12pt;`;
  244. const td = `${cellBase}font-family:'宋体';font-size:11pt;`;
  245. const labelLeft = `${border}padding:4px;mso-number-format:"\\@";vertical-align:middle;text-align:left;mso-horizontal-align:left;font-family:'宋体';font-size:11pt;`;
  246. const colWidths = [50, 100, 120, 100, 60, 80, 90, 100, 80, 120];
  247. const colGroup = colWidths.map((w) => `<col width="${w}" style="width:${w}pt;" />`).join('');
  248. const rows = list.map((item, index) => `
  249. <tr height="24" style="height:24pt;">
  250. <td align="center" valign="middle" style="${td}">${index + 1}</td>
  251. <td align="center" valign="middle" style="${td}">${item.productCode || ''}</td>
  252. <td align="center" valign="middle" style="${td}">${item.productName || ''}</td>
  253. <td align="center" valign="middle" style="${td}">${item.specification || ''}${item.modelType ? '/' + item.modelType : ''}</td>
  254. <td align="center" valign="middle" style="${td}">${item.measuringUnit || ''}</td>
  255. <td align="center" valign="middle" style="${td}">${item.totalCount ?? ''}</td>
  256. <td align="center" valign="middle" style="${td}">${item.singlePrice ?? ''}</td>
  257. <td align="center" valign="middle" style="${td}">${item.totalPrice ?? ''}</td>
  258. <td align="center" valign="middle" style="${td}">${item.batchNo || ''}</td>
  259. <td align="center" valign="middle" style="${td}">${item.remark || ''}</td>
  260. </tr>`).join('');
  261. const title = `${this.groupName || ''}入库单`;
  262. const table = `
  263. <table border="1" cellspacing="0" cellpadding="0" align="center"
  264. style="border-collapse:collapse;border:1px solid #000;mso-border-alt:solid #000 0.5pt;font-family:'宋体';font-size:11pt;text-align:center;">
  265. <colgroup>${colGroup}</colgroup>
  266. <tr height="40" style="height:40pt;">
  267. <td colspan="10" align="center" valign="middle" style="${border}padding:6px;text-align:center;font-family:'宋体';font-size:22pt;font-weight:bold;">${title}</td>
  268. </tr>
  269. <tr height="26" style="height:26pt;">
  270. <td colspan="4" align="left" valign="middle" style="${labelLeft}">供应商:${this.formData.supplierName || ''}</td>
  271. <td colspan="3" align="left" valign="middle" style="${labelLeft}">日期:${this.formData.receiveDate || ''}</td>
  272. <td colspan="3" align="left" valign="middle" style="${labelLeft}">编号:${this.formData.receiveNo || ''}</td>
  273. </tr>
  274. <tr height="26" style="height:26pt;">
  275. <td colspan="10" align="left" valign="middle" style="${labelLeft}">摘要:</td>
  276. </tr>
  277. <tr height="28" style="height:28pt;">
  278. <td align="center" valign="middle" style="${th}">编号</td>
  279. <td align="center" valign="middle" style="${th}">物料编码</td>
  280. <td align="center" valign="middle" style="${th}">物料名称</td>
  281. <td align="center" valign="middle" style="${th}">规格型号</td>
  282. <td align="center" valign="middle" style="${th}">单位</td>
  283. <td align="center" valign="middle" style="${th}">数量</td>
  284. <td align="center" valign="middle" style="${th}">单价</td>
  285. <td align="center" valign="middle" style="${th}">金额</td>
  286. <td align="center" valign="middle" style="${th}">批次号</td>
  287. <td align="center" valign="middle" style="${th}">备注</td>
  288. </tr>
  289. ${rows}
  290. <tr height="26" style="height:26pt;">
  291. <td align="center" valign="middle" style="${td};font-weight:bold;">合计</td>
  292. <td colspan="9" align="left" valign="middle" style="${td};font-weight:bold;">
  293. <span>本页数量小计:${totalCount}</span>
  294. <span style="margin-left:40px;">本页金额小计:${totalPrice}</span>
  295. <span style="margin-left:40px;">汇总数量:${totalCount}</span>
  296. <span style="margin-left:40px;">汇总金额:${totalPrice}</span>
  297. </td>
  298. </tr>
  299. <tr height="26" style="height:26pt;">
  300. <td align="center" valign="middle" style="${td};font-weight:bold;">合计金额(大写金额)</td>
  301. <td colspan="9" align="left" valign="middle" style="${td};font-weight:bold;">
  302. ${convertToChinese(totalPrice)}
  303. </td>
  304. </tr>
  305. <tr height="28" style="height:28pt;">
  306. <td colspan="2" align="left" valign="middle" style="${labelLeft}">收货单审核人:${this.formData.reviewerName || ''}</td>
  307. <td colspan="2" align="left" valign="middle" style="${labelLeft}">采购发起人:${this.formData.purchaseInitiatorName || ''}</td>
  308. <td colspan="2" align="left" valign="middle" style="${labelLeft}">质检人:${this.formData.qualityName || ''}</td>
  309. <td colspan="4" align="left" valign="middle" style="${labelLeft}">仓管员:${this.formData.warehouseKeeperName || ''}</td>
  310. </tr>
  311. </table>`;
  312. const html = `<html xmlns:o="urn:schemas-microsoft-com:office:office" xmlns:x="urn:schemas-microsoft-com:office:excel" xmlns="http://www.w3.org/TR/REC-html40">
  313. <head>
  314. <meta charset="UTF-8" />
  315. <style>
  316. br { mso-data-placement: same-cell; }
  317. table { mso-displayed-decimal-separator:"."; mso-displayed-thousand-separator:","; }
  318. td { mso-ignore:padding; }
  319. </style>
  320. <!--[if gte mso 9]><xml>
  321. <x:ExcelWorkbook><x:ExcelWorksheets><x:ExcelWorksheet>
  322. <x:Name>${title}</x:Name>
  323. <x:WorksheetOptions>
  324. <x:DefaultRowHeight>260</x:DefaultRowHeight>
  325. <x:Print><x:ValidPrinterInfo/><x:HorizontalResolution>600</x:HorizontalResolution><x:VerticalResolution>600</x:VerticalResolution></x:Print>
  326. <x:Selected/>
  327. <x:DoNotDisplayGridlines/>
  328. </x:WorksheetOptions>
  329. </x:ExcelWorksheet></x:ExcelWorksheets></x:ExcelWorkbook>
  330. </xml><![endif]-->
  331. </head>
  332. <body>${table}</body></html>`;
  333. const blob = new Blob(['' + html], {
  334. type: 'application/vnd.ms-excel;charset=utf-8'
  335. });
  336. saveAs(blob, `${title}_${this.formData.receiveNo || ''}.xls`);
  337. }
  338. }
  339. };
  340. </script>
  341. <style lang="scss"></style>