finstlbillsitems.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476
  1. <template>
  2. <div>
  3. <!--:row-style="{height:'20px',padding:'0px',fontSize:'12px'}"-->
  4. <el-table
  5. ref="tableRef"
  6. :cell-style="cellStyle"
  7. :header-cell-style="tableHeaderCellStyle"
  8. :data="tableData"
  9. border
  10. style="width: 100%"
  11. @selection-change="handleSelectionChange"
  12. @row-click="rowClick"
  13. :row-style="rowStyle"
  14. :row-class-name="rowClassName"
  15. @expand-change="expandChange" >
  16. <el-table-column
  17. v-if="brfalse"
  18. type="selection"
  19. width="55">
  20. </el-table-column>
  21. <el-table-column type="expand" v-if="brfalse">
  22. <template slot-scope="{row}">
  23. <costDetails :costDate="row.costDate || []"></costDetails>
  24. </template>
  25. </el-table-column>
  26. <el-table-column
  27. prop="lineNo"
  28. label="行号" width="50px">
  29. <template slot-scope="scope" >{{Number(scope.$index) + 1}}</template>
  30. </el-table-column>
  31. <el-table-column
  32. prop="isCleared"
  33. label="核销" width="50px">
  34. <template slot-scope="{ row }">
  35. <span v-if="row.isCleared == item.dictKey" v-for="item in ifInvoiceData" :key="item.dictKey">
  36. {{item.dictValue}}
  37. </span>
  38. </template>
  39. </el-table-column>
  40. <el-table-column
  41. prop="dc"
  42. label="收/付" width="50px">
  43. <template slot-scope="{ row }">
  44. <span>{{row.dc}}</span>
  45. </template>
  46. </el-table-column>
  47. <el-table-column
  48. prop="billNo"
  49. label="原业务编号" width="120px">
  50. <template slot-scope="{ row }">
  51. <span class="textHide" >{{row.billNo}}</span>
  52. </template>
  53. </el-table-column>
  54. <el-table-column
  55. prop="accBillNo"
  56. label="账单编号" width="140px">
  57. </el-table-column>
  58. <el-table-column
  59. prop="corpCnName"
  60. label="客户名称" width="130px">
  61. <template slot-scope="{ row }">
  62. <span class="textHide">{{row.corpCnName}}</span>
  63. </template>
  64. </el-table-column>
  65. <el-table-column
  66. prop="mblno"
  67. label="MB/L NO" width="120px">
  68. <template slot-scope="{ row }">
  69. <span class="textHide">{{row.mblno}}</span>
  70. </template>
  71. </el-table-column>
  72. <el-table-column
  73. prop="currentStlExrate"
  74. label="汇率">
  75. </el-table-column>
  76. <el-table-column
  77. prop="currentStlAmountRMB"
  78. label="本次CNY" width="100px">
  79. <template slot-scope="scope">
  80. <el-input style="width: 100%;" v-model="scope.row.currentStlAmountRMB"
  81. :disabled="scope.row.curCode != 'CNY' || settlementdistar"
  82. v-if="(brfalse && !editSave)"
  83. size="mini" autocomplete="off"
  84. clearable placeholder="请输入USD" >
  85. </el-input>
  86. <span v-else>{{scope.row.currentStlAmountRMB}}</span>
  87. </template>
  88. </el-table-column>
  89. <el-table-column
  90. prop="currentStlAmountUSD"
  91. label="本次USD" width="100px">
  92. <template slot-scope="scope">
  93. <el-input style="width: 100%;" v-model="scope.row.currentStlAmountUSD"
  94. :disabled="scope.row.curCode != 'USD' || settlementdistar"
  95. v-if="(brfalse && !editSave)"
  96. size="mini" autocomplete="off"
  97. clearable placeholder="请输入USD" >
  98. </el-input>
  99. <span v-else>{{scope.row.currentStlAmountUSD}}</span>
  100. </template>
  101. </el-table-column>
  102. <el-table-column
  103. prop="isSignfor"
  104. label="签收">
  105. <template slot-scope="{ row }">
  106. <span v-if="row.isSignfor == item.dictKey" v-for="item in ifInvoiceData" :key="item.dictKey">
  107. {{item.dictValue}}
  108. </span>
  109. </template>
  110. </el-table-column>
  111. <el-table-column
  112. prop="signforDate"
  113. label="签收日期">
  114. </el-table-column>
  115. <el-table-column
  116. prop="currentInvoiceAmountRMB"
  117. label="发票CNY">
  118. </el-table-column>
  119. <el-table-column
  120. prop="currentInvoiceAmountUSD"
  121. label="发票USD">
  122. </el-table-column>
  123. <el-table-column
  124. prop="etd"
  125. label="ETD" width="100px">
  126. </el-table-column>
  127. <el-table-column
  128. prop="vesselCnName"
  129. label="船名" width="140px">
  130. <template slot-scope="{ row }" >
  131. <span class="textHide" >{{row.vesselCnName}}</span>
  132. </template>
  133. </el-table-column>
  134. <el-table-column
  135. prop="voyageNo"
  136. label="航次" width="100px">
  137. <template slot-scope="{ row }">
  138. <span class="textHide" >{{row.voyageNo}}</span>
  139. </template>
  140. </el-table-column>
  141. <el-table-column
  142. prop="podCnName"
  143. label="目的港" width="100px">
  144. </el-table-column>
  145. <el-table-column
  146. prop="quantityCntrTypesDescr"
  147. label="箱量">
  148. </el-table-column>
  149. <!--amount 应结算金额 有税-->
  150. <!--amountNet 应结算金额 没有税-->
  151. <el-table-column
  152. prop="amountRMB"
  153. label="应结算CNY" width="90px">
  154. </el-table-column>
  155. <el-table-column
  156. prop="amountUSD"
  157. label="应结算USD" width="90px">
  158. </el-table-column>
  159. <el-table-column
  160. prop="stlTtlAmountRMB"
  161. label="已结算CNY" width="100px">
  162. </el-table-column>
  163. <el-table-column
  164. prop="stlTtlAmountUSD"
  165. label="已结算USD" width="100px">
  166. </el-table-column>
  167. <!-- stlTtlAmount 乘 taxRate + surchargeRate-->
  168. <el-table-column
  169. prop="eta"
  170. label="ETA">
  171. <template slot-scope="{ row }">
  172. <span class="textHide" >{{row.eta}}</span>
  173. </template>
  174. </el-table-column>
  175. <el-table-column
  176. prop="remarkss"
  177. label="备注" width="150px">
  178. <template slot-scope="scope">
  179. <el-input style="width: 100%;" v-model="scope.row.remarkss" v-if="brfalse"
  180. size="mini" autocomplete="off"
  181. clearable placeholder="请输入备注" >
  182. </el-input>
  183. <span v-else>{{scope.row.remarkss}}</span>
  184. </template>
  185. </el-table-column>
  186. <el-table-column
  187. fixed="right"
  188. label="操作"
  189. width="100" v-if="brfalse">
  190. <template slot-scope="scope">
  191. <el-button type="text" size="small" :disabled="editSave" @click="deletefun(scope.row,scope.$index)">删除</el-button>
  192. </template>
  193. </el-table-column>
  194. </el-table>
  195. </div>
  196. </template>
  197. <script>
  198. import {getWorkDicts} from "@/api/system/dictbiz";
  199. import SearchQuery from "@/components/iosbasic-data/searchquery.vue";
  200. import {getRateList} from "@/api/iosBasicData/rateManagement";
  201. import costDetails from "@/views/iosBasicData/finstlbills/assembly/costDetails.vue";
  202. import {feecenterSelectByAccNoList} from "@/api/iosBasicData/finstlbills";
  203. export default {
  204. components: {costDetails, SearchQuery},
  205. props:{
  206. tableData:{
  207. type:Array,
  208. default:[],
  209. },
  210. brfalse:{
  211. type:Boolean,
  212. default:true,
  213. },
  214. handleSelectionData:{
  215. type:Array,
  216. default:[]
  217. },
  218. // 是否禁用
  219. editSave:{
  220. type:Boolean,
  221. default:false
  222. },
  223. settlementdistar:{
  224. type:Boolean,
  225. default:false
  226. },
  227. form:{
  228. type:Object,
  229. default:{},
  230. },
  231. },
  232. data(){
  233. return {
  234. ifInvoiceData:[], // 是否数据
  235. invoicelosDara:[],// 发票
  236. curCodeData:[], // 币种
  237. }
  238. },
  239. created() {
  240. this.isSignforWorkDicts()
  241. this.invoicelosWorkDictsfun()
  242. },
  243. methods:{
  244. // 展开行或者关闭
  245. expandChange(row){
  246. let accBillId = ''
  247. if (this.form.id) {
  248. accBillId = row.accBillId
  249. }else {
  250. accBillId = row.id
  251. }
  252. feecenterSelectByAccNoList({accBillId}).then(res=>{
  253. row.costDate = res.data.data.map(item=>{
  254. if (item.curCode == 'CNY') {
  255. this.$set(item,'rmbAmount',item.amount)
  256. this.$set(item,'usdAmount','')
  257. this.$set(item,'rmbAmountNet',item.amountNet)
  258. this.$set(item,'usdAmountNet','')
  259. }else {
  260. this.$set(item,'usdAmount',item.amount)
  261. this.$set(item,'rmbAmount','')
  262. this.$set(item,'usdAmountNet',item.amountNet)
  263. this.$set(item,'rmbAmountNet','')
  264. }
  265. return item
  266. })
  267. })
  268. },
  269. // 明细删除
  270. deletefun(row,index){
  271. this.$emit('deletefun',row.id,index)
  272. },
  273. // 币别切换
  274. corpChange(value,row){
  275. this.$set(row,'currentStlCurCode',value)
  276. if (value == 'CNY') {
  277. this.$set(row,'currentStlAmountRMB',Number(row.amount) - Number(row.stlTtlAmount))
  278. }else {
  279. this.$set(row,'currentStlAmountUSD',Number(row.amount) - Number(row.stlTtlAmount))
  280. }
  281. },
  282. // 接口请求
  283. // 是否接口
  284. isSignforWorkDicts(){
  285. getWorkDicts('ifInvoice').then(res=>{
  286. this.ifInvoiceData = res.data.data
  287. })
  288. },
  289. // 发票
  290. invoicelosWorkDictsfun(){
  291. getWorkDicts('invoice_los').then(res=>{
  292. this.invoicelosDara = res.data.data
  293. })
  294. },
  295. // 获取币别数据
  296. getRateListfun(cnName){
  297. getRateList({current:1,size:10,cnName}).then(res=>{
  298. this.curCodeData = res.data.data.records
  299. })
  300. },
  301. // 表头样式
  302. tableHeaderCellStyle({row,column,rowIndex, columnIndex}){
  303. return "padding:4px 0px;fontSize:12px;color:#000;background:#ecf5ff"
  304. },
  305. // 更改表格颜色
  306. headerClassName(tab) {
  307. //颜色间隔
  308. let back = ""
  309. if (tab.columnIndex >= 0 && tab.column.level === 1) {
  310. if (tab.columnIndex % 2 === 0) {
  311. back = "back-one"
  312. } else if (tab.columnIndex % 2 === 1) {
  313. back = "back-two"
  314. }
  315. }
  316. return back;
  317. },
  318. // Element UI 表格点击选中行/取消选中 快捷多选 以及快捷连续多选,高亮选中行 ——-------------------------------------——
  319. // 多选选择的数据
  320. handleSelectionChange(arr){
  321. // // 全选
  322. // if (arr.length == this.tableData.length) {
  323. // for (let item of arr) {
  324. // this.$set(item,'tableSelect',1)
  325. // }
  326. // }
  327. // // 清除全选
  328. // if (arr.length == 0) {
  329. // for (let item of this.tableData) {
  330. // this.$set(item,'tableSelect',0)
  331. // }
  332. // }
  333. this.$emit('handleSelectionChange',arr)
  334. },
  335. // // 多选
  336. // toggleSelection(rows){
  337. // if (rows) {
  338. // rows.forEach(row => {
  339. // this.$refs.tableRef.toggleRowSelection(row);
  340. // });
  341. // } else {
  342. // this.$refs.tableRef.clearSelection();
  343. // }
  344. // },
  345. // 监听点击表格事件
  346. rowClick(row, column, event) {
  347. let refsElTable = this.$refs.tableRef; // 获取表格对象
  348. if (this.CtrlDown) {
  349. refsElTable.toggleRowSelection(row); // ctrl多选 如果点击两次同样会取消选中
  350. return;
  351. }
  352. if ( this.shiftOrAltDown && this.handleSelectionData.length > 0) {
  353. // 通过rowIndex判断已选择的行中最上面和最下面的是哪行,再对比按住shift/alt点击的当前行得到新的最上面和最下面的行,把这两行中间的行进行循环选中。
  354. let topAndBottom = this.getTopAndBottom( row, this.bottomSelectionRow, this.topSelectionRow );
  355. refsElTable.clearSelection(); //先清空 不然会导致在这两行中间之外的行状态不变
  356. for (let index = topAndBottom.top; index <= topAndBottom.bottom; index++) { //选中两行之间的所有行
  357. refsElTable.toggleRowSelection(this.tableData[index], true);
  358. }
  359. } else {
  360. let findRow = this.handleSelectionData.find(c => c.rowIndex == row.rowIndex); //找出当前选中行
  361. //如果只有一行且点击的也是这一行则取消选择 否则清空再选中当前点击行
  362. if (findRow&& this.handleSelectionData.length === 1 ) {
  363. refsElTable.toggleRowSelection(row, false);
  364. return;
  365. }
  366. // refsElTable.clearSelection(); // 清空之前选择的数据(如果放开,选择之前会变成单选)
  367. refsElTable.toggleRowSelection(row); // 调用选中行方法
  368. }
  369. },
  370. // 行的 style 的回调方法
  371. rowStyle({row,rowIndex}) {
  372. // 直接在一个对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象
  373. // object: 要添加或者修改属性的目标对象;prop: 要定义或修改属性的名称;descript: 是一个对象,里面是我们上述的对象属性的特性;
  374. Object.defineProperty(row, 'rowIndex', { //给每一行添加不可枚举属性rowIndex来标识当前行
  375. value: rowIndex, // 设置age的值,不设置的话默认为undefined
  376. writable: true, // 表示属性的值true可以修改,false不可以被修改
  377. enumerable: false, // 设置为false表示不能通过 for-in 循环返回
  378. // configurable: false, // configurable 设置为 false,意味着这个属性不能从对象上删除
  379. })
  380. },
  381. keyDown(event) {
  382. let key = event.keyCode;
  383. if (key == 17) this.CtrlDown = true;
  384. if (key == 16 || key == 18) this.shiftOrAltDown = true;
  385. },
  386. keyUp(event) {
  387. let key = event.keyCode;
  388. if (key == 17) this.CtrlDown = false;
  389. if (key == 16 || key == 18) this.shiftOrAltDown = false;
  390. },
  391. // 文章说明 https://www.jianshu.com/p/48f2c522d2a2
  392. getTopAndBottom(row, bottom, top){
  393. let n = row.rowIndex,
  394. mx = bottom.rowIndex,
  395. mi = top.rowIndex;
  396. if (n > mx) {
  397. return {
  398. top: mi,
  399. bottom: n
  400. };
  401. } else if (n < mx && n > mi) {
  402. return {
  403. top: mi,
  404. bottom: n
  405. };
  406. } else if (n < mi) {
  407. return {
  408. top: n,
  409. bottom: mx
  410. };
  411. } else if (n == mi || n == mx) {
  412. return {
  413. top: mi,
  414. bottom: mx
  415. };
  416. }
  417. },
  418. // 给选中行加上current-row这个class类,所以要使用row-class-name这个属性(其实给每一行添加rowIndex也可以用这个属性),
  419. // 判断方式也是通过判断rowIndex对比
  420. rowClassName({ row, rowIndex }) {
  421. let rowName = "",
  422. findRow = this.handleSelectionData.find(c => c.rowIndex === row.rowIndex);
  423. if (findRow) {
  424. rowName = "current-row "; // elementUI 默认高亮行的class类 不用再样式了^-^,也可通过css覆盖改变背景颜色
  425. }
  426. return rowName; //也可以再加上其他类名 如果有需求的话
  427. },
  428. // 收付展示不一样的颜色
  429. cellStyle({row,rowIndex,columnIndex}){
  430. let rowStyle = ''
  431. if (row.dc == 'D') {
  432. rowStyle = 'color:#8cb24b;'
  433. }else if (row.dc == 'C') {
  434. rowStyle = 'color:#7ebace;'
  435. }else {}
  436. return rowStyle + 'padding:0px;fontSize:12px'
  437. },
  438. },
  439. mounted() {
  440. // 按住ctrl实现多选 设置监听keydown事件,以及keyup事件,
  441. addEventListener("keydown", this.keyDown, false);
  442. addEventListener("keyup", this.keyUp, false);
  443. },
  444. beforeDestroy() { //解绑
  445. removeEventListener("keydown", this.keyDown);
  446. removeEventListener("keyup", this.keyUp);
  447. },
  448. computed: { //实时得到最上行和最下行
  449. bottomSelectionRow() {
  450. if (this.handleSelectionData.length == 0) return null;
  451. return this.handleSelectionData.reduce((start, end) => {
  452. return start.rowIndex > end.rowIndex ? start : end;
  453. });
  454. },
  455. topSelectionRow() {
  456. if (this.handleSelectionData.length == 0) return null;
  457. return this.handleSelectionData.reduce((start, end) => {
  458. return start.rowIndex < end.rowIndex ? start : end;
  459. });
  460. }
  461. },
  462. }
  463. </script>
  464. <style scoped>
  465. .textHide {
  466. width: 100%;
  467. overflow: hidden;
  468. white-space: nowrap;
  469. text-overflow: ellipsis;
  470. }
  471. </style>