index.vue 26 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012
  1. <template>
  2. <div class="page-payment-record">
  3. <div
  4. v-if="loading"
  5. v-loading="true"
  6. style="
  7. width: 100vw;
  8. height: 100vh;
  9. z-index: 999;
  10. position: fixed;
  11. top: 0;
  12. left: 0;
  13. "
  14. class=""
  15. element-loading-text="Loading..."
  16. element-loading-background="rgba(0, 0, 0, 0.3)"
  17. ></div>
  18. <div class="main-content">
  19. <navPaymentRecord></navPaymentRecord>
  20. <div class="flex justify-between">
  21. <div class="flex btn-wrap">
  22. <el-button
  23. :disabled="multipleSelection.length < 1"
  24. type="danger"
  25. @click="onDelete"
  26. >
  27. Delete
  28. </el-button>
  29. <el-button @click="downloadSample">Download XLSX sample</el-button>
  30. <el-button
  31. type="primary"
  32. @click="dialogVisible = true"
  33. >
  34. Upload Statement
  35. </el-button>
  36. <el-button
  37. :disabled="!tableData.length && !tableData.length"
  38. @click="addRow"
  39. >
  40. Add New Line
  41. </el-button>
  42. <el-button
  43. :disabled="tableData.length < 1 && tableData.length < 1"
  44. type="primary"
  45. @click="tryCreateStatement"
  46. >
  47. Save
  48. </el-button>
  49. <!-- <el-button @click="spliteDomesticTrackingAndPO">test</el-button> -->
  50. </div>
  51. <div class="logo-area">
  52. <img
  53. :src="getLogoPath()"
  54. alt=""
  55. />
  56. </div>
  57. </div>
  58. <div class="po-table">
  59. <el-table
  60. :data="computedTableData"
  61. @selection-change="handleSelectionChange"
  62. >
  63. <el-table-column
  64. fixed
  65. type="selection"
  66. width="55"
  67. />
  68. <el-table-column
  69. prop="payment_type"
  70. label="Payment Type"
  71. width="120"
  72. align="center"
  73. />
  74. <el-table-column
  75. prop="statement_name"
  76. label="Statement Name"
  77. min-width="150"
  78. align="center"
  79. />
  80. <el-table-column
  81. prop="po_number"
  82. label="PO Number"
  83. align="center"
  84. width="200"
  85. label-class-name="red-font"
  86. >
  87. <template #default="scope">
  88. {{
  89. [
  90. scope.row.PO_Number,
  91. scope.row.PO_Number2,
  92. scope.row.PO_Number3,
  93. ]
  94. .filter((i) => !!i)
  95. .join('; ')
  96. }}
  97. </template>
  98. </el-table-column>
  99. <el-table-column
  100. prop="Total"
  101. label="Total"
  102. align="center"
  103. width="110"
  104. label-class-name="red-font"
  105. />
  106. <el-table-column
  107. prop="Billable_Weight"
  108. label="Billable Weight"
  109. align="center"
  110. width="150"
  111. label-class-name="red-font"
  112. />
  113. <el-table-column
  114. prop="Tracking_Number"
  115. label="Tracking Number"
  116. align="center"
  117. width="150"
  118. label-class-name="red-font"
  119. />
  120. <el-table-column
  121. prop="Sender"
  122. label="Sender"
  123. align="center"
  124. width="110"
  125. />
  126. <el-table-column
  127. prop="ATTN"
  128. label="ATTN"
  129. align="center"
  130. width="110"
  131. />
  132. <el-table-column
  133. prop="From_Address"
  134. label="From Address"
  135. align="center"
  136. width="150"
  137. />
  138. <el-table-column
  139. prop="To_Address"
  140. label="To Address"
  141. align="center"
  142. width="150"
  143. />
  144. <el-table-column
  145. prop="Issue_Date"
  146. label="Issue Date"
  147. align="center"
  148. width="110"
  149. />
  150. <el-table-column
  151. prop="Weight"
  152. label="Weight"
  153. align="center"
  154. width="110"
  155. />
  156. <el-table-column
  157. prop="Volume"
  158. label="Volume"
  159. align="center"
  160. width="110"
  161. />
  162. <el-table-column
  163. prop="Fee_Type"
  164. label="Fee Type (运费/保费 etc...)"
  165. align="center"
  166. width="150"
  167. />
  168. <el-table-column
  169. prop="Package_Type"
  170. label="Package Type (次日达/顺丰特快 etc...)"
  171. align="center"
  172. width="180"
  173. />
  174. <el-table-column
  175. prop="Director"
  176. label="Director"
  177. align="center"
  178. width="110"
  179. />
  180. <el-table-column
  181. fixed="right"
  182. label="Action"
  183. width="80px"
  184. >
  185. <template #default="scope">
  186. <el-button
  187. size="small"
  188. type="warning"
  189. @click="editRow(scope.row, scope.$index)"
  190. >
  191. Edit
  192. </el-button>
  193. </template>
  194. </el-table-column>
  195. </el-table>
  196. <br />
  197. <div
  198. class="flex justify-between"
  199. style="align-items: flex-end"
  200. >
  201. <el-pagination
  202. v-model:currentPage="currentPage"
  203. v-model:pageSize="pageSize"
  204. layout="prev, pager, next, jumper, sizes"
  205. :page-sizes="[5, 10, 15, 20, 40, 100]"
  206. :total="tableData.length"
  207. @current-change="multipleSelection = []"
  208. @size-change="multipleSelection = []"
  209. />
  210. <div class="total-data">
  211. <div class="flex">
  212. <div>Total line:</div>
  213. <div>{{ tableData.length }}</div>
  214. </div>
  215. <div class="flex">
  216. <div class="">Sum Total:</div>
  217. <div class="">{{ computedSum }}</div>
  218. </div>
  219. <div class="flex">
  220. <div class="">Currency:</div>
  221. <div>
  222. {{ tableData[0]?.Currency || '0' }}
  223. </div>
  224. </div>
  225. </div>
  226. </div>
  227. </div>
  228. <div class="copyright">
  229. Copyright of Promocollection - Version 1.02 Released on 30/07/2024
  230. </div>
  231. </div>
  232. <dialog-upload
  233. v-model:visible="dialogVisible"
  234. v-model:currencyList="currencyList"
  235. @update-table-data="updateTableData"
  236. ></dialog-upload>
  237. <edit-item
  238. v-model:visible="dialogEditRowVisible"
  239. v-model:currencyList="currencyList"
  240. v-model:currentEditRow="computedCurrentEditRow"
  241. v-model:editMode="editMode"
  242. v-model:disableFlag="currentDisableFlag"
  243. :statement-list="computedStatementList"
  244. :locked-currency="tableData.length ? tableData[0].Currency : 'CNY'"
  245. @edit="onEditRow"
  246. @add="onAddRow"
  247. ></edit-item>
  248. </div>
  249. </template>
  250. <script lang="ts">
  251. export default defineComponent({
  252. name: 'PaymentRecord2',
  253. })
  254. </script>
  255. <script lang="ts" setup>
  256. import { defineComponent, ref, computed } from 'vue'
  257. import {
  258. ElButton,
  259. ElTable,
  260. ElTableColumn,
  261. ElPagination,
  262. ElMessage,
  263. ElNotification,
  264. } from 'element-plus'
  265. import { useRoute } from 'vue-router'
  266. import dialogUpload from './components/upload.vue'
  267. import editItem from './components/edit.vue'
  268. import { IUser } from '@/interface'
  269. import { IPoItem } from './inteface'
  270. import request from '@/utils/axios'
  271. import utils from '@/utils/index'
  272. import * as XLSX from 'xlsx'
  273. import navPaymentRecord from '@/pages/payment-record/components/nav.vue'
  274. const loading = ref(false)
  275. const getLogoPath = function () {
  276. return new URL('/assets/logo@2x.png', import.meta.url).href
  277. }
  278. const multipleSelection = ref<IPoItem[]>([])
  279. const handleSelectionChange = (val: IPoItem[]) => {
  280. multipleSelection.value = val
  281. }
  282. const onDelete = function () {
  283. const target = multipleSelection.value
  284. if (tableData.value.length) {
  285. tableData.value = tableData.value.filter((i) => {
  286. return !target.includes(i)
  287. })
  288. }
  289. }
  290. const sheetData = [
  291. {
  292. PO_Number: '',
  293. PO_Number2: '',
  294. PO_Number3: '',
  295. Total: '',
  296. Billable_Weight: '',
  297. Tracking_Number: '',
  298. Sender: '',
  299. ATTN: '',
  300. From_Address: '',
  301. To_Address: '',
  302. Issue_Date: '',
  303. Weight: '',
  304. Volume: '',
  305. Fee_Type: '',
  306. Package_Type: '',
  307. Director: '',
  308. },
  309. ]
  310. const downloadSample = function () {
  311. const sheet1 = XLSX.utils.json_to_sheet(sheetData)
  312. const wb = XLSX.utils.book_new()
  313. sheet1['!cols'] = [
  314. { wpx: 100 },
  315. { wpx: 100 },
  316. { wpx: 100 },
  317. { wpx: 100 },
  318. { wpx: 100 },
  319. { wpx: 100 },
  320. { wpx: 150 },
  321. { wpx: 150 },
  322. { wpx: 100 },
  323. { wpx: 100 },
  324. { wpx: 100 },
  325. { wpx: 100 },
  326. { wpx: 100 },
  327. { wpx: 100 },
  328. { wpx: 100 },
  329. { wpx: 100 },
  330. ]
  331. XLSX.utils.book_append_sheet(wb, sheet1, 'sheet1')
  332. XLSX.writeFile(wb, '国内运费模版.xlsx')
  333. }
  334. const computedSum = computed(() => {
  335. return utils.toFixed(
  336. tableData.value.reduce((total, current) => {
  337. total = total + Number(current.Total)
  338. return total
  339. }, 0),
  340. 1000,
  341. )
  342. })
  343. const tableData = ref([
  344. // Object.assign(
  345. // {},
  346. // {
  347. // PO_Number: 'PO13097',
  348. // PO_Number2: '',
  349. // PO_Number3: '',
  350. // Total: 12,
  351. // Billable_Weight: 2,
  352. // Tracking_Number: 'tracking number',
  353. // Sender: 'sender',
  354. // ATTN: 'attn',
  355. // From_Address: 'from addr',
  356. // To_Address: 'to addr',
  357. // Issue_Date: '2024-07-20',
  358. // Weight: 3,
  359. // Volume: 4,
  360. // Fee_Type: 'fee type',
  361. // Package_Type: 'package type',
  362. // Director: 'director',
  363. // },
  364. // {
  365. // Currency: 'CNY',
  366. // statement_name: '华信强0319测试上传月结单',
  367. // payment_type: '国内运费',
  368. // Weight_Unit: 'Kg',
  369. // },
  370. // ),
  371. ] as IPoItem[])
  372. const fileContainer = ref(null as any)
  373. const updateTableData = (p: { data: IPoItem[]; mode: string }, file: any) => {
  374. // console.log(p, 'p')
  375. tableData.value = p.data as IPoItem[]
  376. currentPage.value = 1
  377. fileContainer.value = file
  378. }
  379. const currentDisableFlag = computed(() => {
  380. return tableData.value.length > 0
  381. })
  382. const currentPage = ref(1)
  383. const pageSize = ref(15)
  384. // 手动分页. 可能会有很多条数据
  385. const computedTableData = computed(() => {
  386. return tableData.value.length < pageSize.value
  387. ? tableData.value
  388. : tableData.value.slice(
  389. (currentPage.value - 1) * pageSize.value,
  390. tableData.value.length > currentPage.value * pageSize.value
  391. ? currentPage.value * pageSize.value
  392. : tableData.value.length,
  393. )
  394. })
  395. const computedStatementList = computed(() => {
  396. const result: any[] = []
  397. if (tableData.value.length) {
  398. tableData.value.forEach((i) => {
  399. if (i.statement_name?.length && !result.includes(i.statement_name)) {
  400. result.push(i.statement_name)
  401. }
  402. })
  403. }
  404. return result.map((i, index) => {
  405. // statement调整成点击保存再创建了, 而不是之前的在上传对话框创建, 所以逻辑变动, 这里拿不到id, 只能拿name凑合.
  406. return {
  407. value: i,
  408. label: i,
  409. }
  410. })
  411. })
  412. const dialogEditRowVisible = ref(false)
  413. const currentEditIndex = ref(-1) // -1新增, 其余则为当前页的行号
  414. const editMode = ref(1) // 1新增 2编辑
  415. const computedCurrentEditRow = computed(() => {
  416. if (currentEditIndex.value > -1) {
  417. return tableData.value[
  418. (currentPage.value - 1) * pageSize.value + currentEditIndex.value
  419. ]
  420. }
  421. return {}
  422. })
  423. const addRow = function () {
  424. if (tableData.value.length) {
  425. // console.log('国内快递')
  426. editMode.value = 1
  427. currentEditIndex.value = -1
  428. dialogEditRowVisible.value = true
  429. }
  430. }
  431. const editRow = function (row: IPoItem, index: number) {
  432. editMode.value = 2
  433. currentEditIndex.value = index
  434. if (tableData.value.length) {
  435. dialogEditRowVisible.value = true
  436. }
  437. }
  438. const onAddRow = function (data: IPoItem) {
  439. if (tableData.value.length) {
  440. tableData.value.push(data as IPoItem)
  441. dialogEditRowVisible.value = false
  442. }
  443. }
  444. const onEditRow = function (data: IPoItem) {
  445. if (tableData.value.length) {
  446. dialogEditRowVisible.value = false
  447. tableData.value.splice(
  448. (currentPage.value - 1) * pageSize.value + currentEditIndex.value,
  449. 1,
  450. data as IPoItem,
  451. )
  452. }
  453. }
  454. const tryCreateStatement = function () {
  455. if (
  456. ![
  457. 'Finance Manager',
  458. 'CEO',
  459. 'Account Payable',
  460. 'Logistics Operator',
  461. 'Cashier',
  462. ].includes(userInfo.value.role.name)
  463. ) {
  464. ElMessage.error('当前用户没有处理的权限')
  465. return
  466. }
  467. if (tableData.value.length) {
  468. createStatement()
  469. }
  470. }
  471. const statementID = ref('')
  472. // const statementID = ref('4791186000143466085')
  473. // 上传原文件保存副本. 业务要求.
  474. const uploadStatementFile = function () {
  475. const fileForm = new FormData()
  476. fileForm.append('id', statementID.value)
  477. fileForm.append('file', fileContainer.value)
  478. request
  479. .post('/payment_request/uploadOriginalFile', fileForm, {
  480. headers: {
  481. 'Content-Type': 'multipart/form-data',
  482. },
  483. })
  484. .then((res: any) => {
  485. if (res.data.code === 1) {
  486. ElNotification({
  487. duration: 0,
  488. title: '上传原始表格成功',
  489. type: 'success',
  490. message: '上传原始表格成功',
  491. })
  492. }
  493. })
  494. }
  495. // 带后缀2的全部是国内运费的相关处理逻辑. 因为两个模式相同和不相同的逻辑基本参半, 干脆直接揉在一起写了
  496. const createStatement = function () {
  497. loading.value = true
  498. request
  499. .post('/payment_request/createStatementData', [
  500. {
  501. Total_Amount: computedSum.value,
  502. Payment_Type: tableData.value[0].payment_type,
  503. Currency: tableData.value[0].Currency,
  504. Name: tableData.value[0].statement_name,
  505. Owner: {
  506. name: userInfo.value.full_name,
  507. id: userInfo.value.id,
  508. },
  509. },
  510. ])
  511. .then((response) => {
  512. if (response.data.code !== 1) {
  513. ElMessage.error('创建statement出错')
  514. return
  515. }
  516. statementID.value = response.data.result.data[0].details.id
  517. splitPaymentRequestRecordForm() // 重写splitForm, 参数几乎完全不同
  518. uploadStatementFile() // 这步操作直接解开注释就行. 逻辑参数通用
  519. })
  520. .catch(() => {
  521. loading.value = false
  522. })
  523. }
  524. const splitPaymentRequestRecordForm = function () {
  525. const formData = tableData.value.reduce((t, i) => {
  526. const temp: any = {
  527. Tracking_Number: i.Tracking_Number,
  528. Requested_Module: 'Domestic_Package',
  529. Fee_type: i.Fee_Type,
  530. Unit_Price: 0,
  531. Quantity: '',
  532. Sample_Fee: 0,
  533. Setup_Service_Fee: 0,
  534. Total: i.Total || 0,
  535. Currency: i.Currency,
  536. Description: '',
  537. SKU: '',
  538. Unit_Price_Non_Currency: '',
  539. Payment_Type: i.payment_type,
  540. Statement: { name: i.statement_name, id: statementID.value },
  541. Request_Type: '月结申请',
  542. Name: '/',
  543. Owner: {
  544. name: userInfo.value.full_name,
  545. id: userInfo.value.id,
  546. },
  547. Payment_Status: 'Pending Verify',
  548. Batch_number: new Date().getTime().toString(),
  549. }
  550. if (i.PO_Number) {
  551. t.push(Object.assign({ PO_id: i.PO_Number }, temp))
  552. }
  553. if (i.PO_Number2) {
  554. t.push(Object.assign({ PO_id: i.PO_Number2 }, temp))
  555. }
  556. if (i.PO_Number3) {
  557. t.push(Object.assign({ PO_id: i.PO_Number3 }, temp))
  558. }
  559. return t
  560. }, [] as any[])
  561. // console.log(formData, 'form data 2')
  562. let size = 100
  563. const dataList = utils.splitArray(formData, size)
  564. const pool = []
  565. for (let i = 0; i < dataList.length; i++) {
  566. pool.push(
  567. createPaymentRequestRecord(
  568. dataList[i],
  569. i,
  570. size,
  571. i === dataList.length - 1 ? formData.length : 0,
  572. ),
  573. )
  574. }
  575. loading.value = true
  576. Promise.all(pool).finally(() => {
  577. splitDomesticTracking()
  578. })
  579. }
  580. const createPaymentRequestRecord = function (
  581. data: any[],
  582. currentPage = 0,
  583. pageSize = 1,
  584. finalValue: number,
  585. ) {
  586. return new Promise((resolve, reject) => {
  587. request
  588. .post('/payment_request/createPaymentRequestRecord', data)
  589. .then((response) => {
  590. if (response.data.code !== 1) {
  591. ElNotification({
  592. type: 'error',
  593. duration: 0,
  594. title: '创建异常',
  595. message: `第 ${currentPage * pageSize + 1} ~ ${
  596. finalValue || (currentPage + 1) * pageSize
  597. } 行数据创建异常`,
  598. })
  599. reject(0)
  600. return
  601. }
  602. const res = response.data.result
  603. if (Array.isArray(res.data)) {
  604. const temp = res.data.map((i: any, index: number) => {
  605. return {
  606. status: i.status,
  607. index,
  608. }
  609. })
  610. const temp2 = temp.filter((i: any) => i.status !== 'success')
  611. if (temp2.length) {
  612. ElNotification({
  613. type: 'error',
  614. duration: 0,
  615. title: '创建异常',
  616. message: `第 ${temp2
  617. .map((i: any) => i.index + 1 + currentPage * pageSize)
  618. .join(',')} 行数据创建异常`,
  619. })
  620. reject(0)
  621. } else {
  622. ElNotification({
  623. duration: 0,
  624. title: '创建成功',
  625. type: 'success',
  626. message: `第 ${currentPage * pageSize + 1} ~ ${
  627. finalValue || (currentPage + 1) * pageSize
  628. } 行数据创建成功`,
  629. })
  630. resolve(1)
  631. }
  632. }
  633. })
  634. })
  635. }
  636. const splitDomesticTracking = function () {
  637. const formData = tableData.value.reduce((t, i, index) => {
  638. const temp: any = {
  639. ...i,
  640. Billable_Weight: i.Billable_Weight || 0,
  641. Total: i.Total || 0,
  642. Weight: i.Weight || 0,
  643. Volume: i.Volume || 0,
  644. Statement: { name: i.statement_name, id: statementID.value },
  645. Name: `${i.Issue_Date} - ${i.Tracking_Number}`,
  646. From: i.From_Address,
  647. To: i.To_Address,
  648. Amount: i.Total,
  649. Weight_Unit: i.Weight_Unit,
  650. Owner: {
  651. name: userInfo.value.full_name,
  652. id: userInfo.value.id,
  653. },
  654. }
  655. // 删除不必要的参数字段. 以下几个数据有其他表示的名称
  656. delete temp.PO_Number
  657. delete temp.PO_Number2
  658. delete temp.PO_Number3
  659. delete temp.statement_name
  660. delete temp.payment_type
  661. delete temp.From_Address
  662. delete temp.To_Address
  663. delete temp.Total
  664. if (i.PO_Number) {
  665. t.push(Object.assign({ PO_id: i.PO_Number }, temp))
  666. }
  667. if (i.PO_Number2) {
  668. t.push(Object.assign({ PO_id: i.PO_Number2 }, temp))
  669. }
  670. if (i.PO_Number3) {
  671. t.push(Object.assign({ PO_id: i.PO_Number3 }, temp))
  672. }
  673. return t
  674. }, [] as any[])
  675. // console.log(formData, 'domestic tracking formdata')
  676. let size = 100
  677. const dataList = utils.splitArray(formData, size)
  678. const pool = []
  679. for (let i = 0; i < dataList.length; i++) {
  680. pool.push(
  681. createDomesticTrackingForm(
  682. dataList[i],
  683. i,
  684. size,
  685. i === dataList.length - 1 ? formData.length : 0,
  686. ),
  687. )
  688. }
  689. loading.value = true
  690. Promise.all(pool).finally(() => {
  691. spliteDomesticTrackingAndPO()
  692. })
  693. }
  694. const createDomesticTrackingForm = function (
  695. data: any[],
  696. currentPage = 0,
  697. pageSize = 1,
  698. finalValue: number,
  699. ) {
  700. return new Promise((resolve, reject) => {
  701. request
  702. .post('/payment_request/createDomesticTracking', data)
  703. .then((response) => {
  704. if (response.data.code !== 1) {
  705. ElNotification({
  706. type: 'error',
  707. duration: 0,
  708. title: 'DomesticTracking创建异常',
  709. message: `第 ${currentPage * pageSize + 1} ~ ${
  710. finalValue || (currentPage + 1) * pageSize
  711. } 行数据创建异常`,
  712. })
  713. reject(0)
  714. return
  715. }
  716. const res = response.data.result
  717. if (Array.isArray(res.data)) {
  718. const temp = res.data.map((i: any, index: number) => {
  719. return {
  720. status: i.status,
  721. index,
  722. }
  723. })
  724. const temp2 = temp.filter((i: any) => i.status !== 'success')
  725. if (temp2.length) {
  726. ElNotification({
  727. type: 'error',
  728. duration: 0,
  729. title: 'DomesticTracking创建异常',
  730. message: `第 ${temp2
  731. .map((i: any) => i.index + 1 + currentPage * pageSize)
  732. .join(',')} 行数据创建异常`,
  733. })
  734. reject(0)
  735. } else {
  736. ElNotification({
  737. duration: 0,
  738. title: 'DomesticTracking创建成功',
  739. type: 'success',
  740. message: `第 ${currentPage * pageSize + 1} ~ ${
  741. finalValue || (currentPage + 1) * pageSize
  742. } 行数据创建成功`,
  743. })
  744. resolve(1)
  745. }
  746. }
  747. })
  748. })
  749. }
  750. const spliteDomesticTrackingAndPO = function () {
  751. const formData: any[] = tableData.value.reduce((t, i, index) => {
  752. const temp: any = {
  753. Tracking_Number: i.Tracking_Number,
  754. // 这两个是固定的玩意, 暂定.
  755. Purchase_Order: {
  756. id: '4791186000155027611',
  757. name: 'PO11277 - 底层逻辑记录,不可删除 - 底层逻辑记录,不可删除',
  758. },
  759. Related_Domestic_Tracking: {
  760. id: '4791186000183903233',
  761. name: '底层逻辑记录,不可删除',
  762. },
  763. }
  764. if (i.PO_Number) {
  765. t.push(
  766. Object.assign(
  767. { PO_id: i.PO_Number, Name: `${i.Tracking_Number}-${i.PO_Number}` },
  768. temp,
  769. ),
  770. )
  771. }
  772. if (i.PO_Number2) {
  773. t.push(
  774. Object.assign(
  775. { PO_id: i.PO_Number2, Name: `${i.Tracking_Number}-${i.PO_Number2}` },
  776. temp,
  777. ),
  778. )
  779. }
  780. if (i.PO_Number3) {
  781. t.push(
  782. Object.assign(
  783. { PO_id: i.PO_Number3, Name: `${i.Tracking_Number}-${i.PO_Number3}` },
  784. temp,
  785. ),
  786. )
  787. }
  788. return t
  789. }, [] as any[])
  790. // console.log(formData, 'form DomesticTrackingAndPO 2')
  791. let size = 100
  792. const dataList = utils.splitArray(formData, size)
  793. const pool = []
  794. for (let i = 0; i < dataList.length; i++) {
  795. pool.push(
  796. createDomesticTrackingAndPOForm(
  797. dataList[i],
  798. i,
  799. size,
  800. i === dataList.length - 1 ? formData.length : 0,
  801. ),
  802. )
  803. }
  804. loading.value = true
  805. Promise.all(pool)
  806. .then(() => {
  807. tableData.value = []
  808. statementID.value = ''
  809. })
  810. .finally(() => {
  811. loading.value = false
  812. })
  813. }
  814. const createDomesticTrackingAndPOForm = function (
  815. data: any[],
  816. currentPage = 0,
  817. pageSize = 1,
  818. finalValue: number,
  819. ) {
  820. return new Promise((resolve, reject) => {
  821. request
  822. .post('/payment_request/createDomesticPkgandPO', data)
  823. .then((response) => {
  824. if (response.data.code !== 1) {
  825. ElNotification({
  826. type: 'error',
  827. duration: 0,
  828. title: 'DomesticPkg_and_PO创建异常',
  829. message: `第 ${currentPage * pageSize + 1} ~ ${
  830. finalValue || (currentPage + 1) * pageSize
  831. } 行数据创建异常`,
  832. })
  833. reject(0)
  834. return
  835. }
  836. const res = response.data.result
  837. if (Array.isArray(res.data)) {
  838. const temp = res.data.map((i: any, index: number) => {
  839. return {
  840. status: i.status,
  841. index,
  842. }
  843. })
  844. const temp2 = temp.filter((i: any) => i.status !== 'success')
  845. if (temp2.length) {
  846. ElNotification({
  847. type: 'error',
  848. duration: 0,
  849. title: 'DomesticPkg_and_PO创建异常',
  850. message: `第 ${temp2
  851. .map((i: any) => i.index + 1 + currentPage * pageSize)
  852. .join(',')} 行数据创建异常`,
  853. })
  854. reject(0)
  855. } else {
  856. ElNotification({
  857. duration: 0,
  858. title: 'DomesticPkg_and_PO创建成功',
  859. type: 'success',
  860. message: `第 ${currentPage * pageSize + 1} ~ ${
  861. finalValue || (currentPage + 1) * pageSize
  862. } 行数据创建成功`,
  863. })
  864. resolve(1)
  865. }
  866. }
  867. })
  868. })
  869. }
  870. const dialogVisible = ref(false)
  871. const currencyList = ref([
  872. {
  873. label: 'CNY',
  874. value: 'CNY',
  875. },
  876. {
  877. label: 'USD',
  878. value: 'USD',
  879. },
  880. {
  881. label: 'HKD',
  882. value: 'HKD',
  883. },
  884. {
  885. label: 'AUD',
  886. value: 'AUD',
  887. },
  888. {
  889. label: 'GBP',
  890. value: 'GBP',
  891. },
  892. {
  893. label: 'NZD',
  894. value: 'NZD',
  895. },
  896. {
  897. label: 'EUR',
  898. value: 'EUR',
  899. },
  900. ])
  901. const route = useRoute()
  902. const userInfo = ref({} as IUser)
  903. loading.value = true
  904. request
  905. .post('/common/getUsersData', { id: route.query.user })
  906. .then((response) => {
  907. const res = response.data
  908. if (res.code !== 1) return
  909. if (res.result.users && res.result.users.length) {
  910. userInfo.value = res.result.users[0]
  911. } else if (res.result.id) {
  912. userInfo.value = res.result || {}
  913. } else if (Array.isArray(res.result) && res.result.length) {
  914. userInfo.value = res.result[0] || {}
  915. } else {
  916. ElMessage.error('获取当前用户身份异常, 请联系管理员')
  917. }
  918. })
  919. .finally(() => {
  920. loading.value = false
  921. })
  922. </script>
  923. <style lang="scss">
  924. .page-payment-record {
  925. .red-font {
  926. color: red;
  927. }
  928. }
  929. </style>
  930. <style lang="scss" scoped>
  931. .main-content {
  932. background-color: #fff;
  933. padding: 12px 40px;
  934. width: 1900px;
  935. min-width: 1200px;
  936. min-height: 100vh;
  937. margin: 0 auto;
  938. box-shadow:
  939. 0 0 0 1px rgba(255, 255, 255, 0.4) inset,
  940. 0 0.5em 1em rgba(0, 0, 0, 0.6);
  941. }
  942. .btn-wrap {
  943. width: 1600px;
  944. min-width: 1200px;
  945. padding: 12px 0;
  946. margin: 0 auto;
  947. }
  948. .logo-area {
  949. img {
  950. height: 60px;
  951. }
  952. }
  953. .po-table {
  954. width: 100%;
  955. min-height: 80vh;
  956. margin: 0 auto;
  957. }
  958. .total-data {
  959. width: 150px;
  960. line-height: 22px;
  961. }
  962. .copyright {
  963. text-align: right;
  964. color: #ccc;
  965. font-family: Fun, sans-serif;
  966. font-size: 14px;
  967. line-height: 16px;
  968. font-style: italic;
  969. }
  970. </style>