contractStatistics.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  1. <template>
  2. <div id="customer-statistics-container" class="customer-statistics-container">
  3. <div class="search-zone">
  4. <div class="search-box">
  5. <el-form label-width="0" :inline="true" :model="formInline">
  6. <el-form-item label="">
  7. <el-input v-model="searchParams.Keyword" placeholder="请输入客户名称" clearable class="search-item"
  8. @input="searchList" prefix-icon="el-icon-search" style="width: 240px;" ></el-input>
  9. </el-form-item>
  10. <el-form-item label="" v-if="currentSdId==AdminId ||Role=='admin'||Role=='ficc_admin'||Role=='rai_admin'">
  11. <el-cascader :options="saleList" style="width: 240px;margin: 0 0 8px 0;" filterable v-model="checkedSale"
  12. @change="saleChange" placeholder="请选择销售" clearable collapse-tags :show-all-levels="false"
  13. :props="{multiple:true,label:'RealName',value:'AdminId',children:'ChildrenList',emitPath:false}" key="sale" >
  14. </el-cascader>
  15. </el-form-item>
  16. <el-form-item label="">
  17. <el-cascader :options="serviceList" style="width: 240px;margin: 0 0 8px 0;" filterable v-model="checkedService"
  18. @change="serviceChange" placeholder="请选择套餐信息" clearable collapse-tags :show-all-levels="false"
  19. :props="{multiple:true,label:'title',value:'service_template_id',children:'children',emitPath:false}" key="serivce" >
  20. </el-cascader>
  21. </el-form-item>
  22. <el-form-item label="">
  23. <div class="date-box">
  24. <el-date-picker v-model="searchDate" type="daterange" @change="currentDateTab=0" style="max-width: 240px;margin-right: 15px;"
  25. value-format="yyyy-MM-dd" start-placeholder="开始日期" end-placeholder="结束日期"></el-date-picker>
  26. <div class="composition-button-tabs">
  27. <el-button size="large" v-for="(item,index) in dateButtonData" :key="item.tabId"
  28. class="date-button"
  29. :class="[index==0?'first-button':index==(dateButtonData.length-1)?'last-button':'inner-button',currentDateTab==item.tabId?'selectTab':'']"
  30. @click="changeDateType(item)">{{ item.text }}</el-button>
  31. </div>
  32. </div>
  33. </el-form-item>
  34. <el-form-item label="">
  35. <el-select v-model="searchParams.TimeType" placeholder="请选择日期类型" @change="searchList" style="width: 240px;" clearable>
  36. <el-option :label="item.label" :value="item.value" v-for="item in timeTypeData" :key="item.value"></el-option>
  37. </el-select>
  38. </el-form-item>
  39. <el-form-item label="">
  40. <el-select v-model="searchParams.HasInvoice" placeholder="请选择开票状态" @change="searchList" style="width: 240px;" clearable>
  41. <el-option :label="item.label" :value="item.value" v-for="item in invoiceData" :key="item.value"></el-option>
  42. </el-select>
  43. </el-form-item>
  44. <el-form-item label="">
  45. <el-select v-model="searchParams.HasPayment" placeholder="请选择到款状态" @change="searchList" style="width: 240px;" clearable>
  46. <el-option :label="item.label" :value="item.value" v-for="item in paymentData" :key="item.value"></el-option>
  47. </el-select>
  48. </el-form-item>
  49. </el-form>
  50. </div>
  51. </div>
  52. <div class="amount-show-zone">
  53. <div class="amount-show-item" style="margin-right: 20px;">
  54. <div class="amount-item-head" @click="foldOrUnfold(0)"
  55. :style="{cursor:tableData.length>0?'pointer':'',padding:invoiceIsFold?'8px 20px 8px 20px':'20px'}">
  56. <div class="amount-item-head-title" >
  57. 已开票合计金额(换算后):{{ invoiceAmountTotal }}(CNY)
  58. </div>
  59. <div class="fold-expand-row" v-show="tableData.length>0">
  60. <span class="amount-item-head-icon">
  61. {{ invoiceIsFold?'展开':'收起' }}
  62. </span>
  63. <i class="el-icon-arrow-down" :style="!invoiceIsFold && 'transform: rotate(180deg)'" style="color: #409EFF;"></i>
  64. </div>
  65. </div>
  66. <div class="amount-item-body-package" :style="{height:invoiceIsFold?'0':'66px'}">
  67. <div class="amount-item-body">
  68. <div class="amount-item-body-box" v-for="item in invoiceAmountList" :key="item.code">
  69. <img :src="item.flag_img" style="height: 40px;width: 40px;margin-right: 20px;" />
  70. <div class="amount-item-body-info">
  71. <span>{{ item.name }}({{ item.code }})</span>
  72. <span>{{ item.amount }}</span>
  73. </div>
  74. </div>
  75. </div>
  76. </div>
  77. </div>
  78. <div class="amount-show-item">
  79. <div class="amount-item-head" @click="foldOrUnfold(1)"
  80. :style="{cursor:tableData.length>0?'pointer':'',padding:placementIsFold?'8px 20px 8px 20px':'20px'}">
  81. <div class="amount-item-head-title" >
  82. 已到款合计金额(换算后):{{ placementAmountTotal }}(CNY)
  83. </div>
  84. <div class="fold-expand-row" v-show="tableData.length>0">
  85. <span class="amount-item-head-icon">
  86. {{ placementIsFold?'展开':'收起' }}
  87. </span>
  88. <i class="el-icon-arrow-down" :style="!placementIsFold && 'transform: rotate(180deg)'" style="color: #409EFF;"></i>
  89. </div>
  90. </div>
  91. <div class="amount-item-body-package" :style="{height:placementIsFold?'0':'66px'}">
  92. <div class="amount-item-body">
  93. <div class="amount-item-body-box" v-for="item in placementAmountList" :key="item.code">
  94. <img :src="item.flag_img" style="height: 40px;width: 40px;margin-right: 20px;" />
  95. <div class="amount-item-body-info">
  96. <span>{{ item.name }}({{ item.code }})</span>
  97. <span>{{ item.amount }}</span>
  98. </div>
  99. </div>
  100. </div>
  101. </div>
  102. </div>
  103. </div>
  104. <div class="table-zone">
  105. <el-table :data="tableData" border ref="tableRef" max-height="580" @sort-change="sortChange" >
  106. <el-table-column label="序号" align="center" prop="serialNumber" width="60">
  107. <template slot-scope="{row}">
  108. {{ row.serialNumber }}
  109. </template>
  110. </el-table-column>
  111. <el-table-column label="客户名称" prop="company_name" align="center">
  112. <template slot-scope="{row,$index}">
  113. <el-tooltip :content="row.company_name" placement="top" :disabled="disabledCheck($index)">
  114. <div class="company-name-column"><span class="company-name-span">{{ row.company_name }}</span></div>
  115. </el-tooltip>
  116. </template>
  117. </el-table-column>
  118. <el-table-column label="是否新客户" prop="contract_type" align="center">
  119. <template slot-scope="{row}">
  120. {{ row.contract_type==1?'是':'否' }}
  121. </template>
  122. </el-table-column>
  123. <el-table-column label="合同有效期" prop="date" align="center" show-overflow-tooltip >
  124. <template slot-scope="{row}">
  125. {{ row.date }}
  126. </template>
  127. </el-table-column>
  128. <el-table-column label="开票日" show-overflow-tooltip sortable="custom" prop="invoice_time" align="center">
  129. <template slot-scope="{row}">
  130. {{ row.invoice_time }}
  131. </template>
  132. </el-table-column>
  133. <el-table-column label="开票金额" prop="invoice_amount" align="center">
  134. <template slot-scope="{row}">
  135. {{ row.invoice_amount }}
  136. </template>
  137. </el-table-column>
  138. <el-table-column label="到款日" show-overflow-tooltip sortable="custom" prop="payment_date" align="center">
  139. <template slot-scope="{row}">
  140. {{ row.payment_date }}
  141. </template>
  142. </el-table-column>
  143. <el-table-column label="到款金额" prop="payment_amount" align="center">
  144. <template slot-scope="{row}">
  145. {{ row.payment_amount }}
  146. </template>
  147. </el-table-column>
  148. <el-table-column label="付款方式" prop="pay_type" align="center">
  149. <template slot-scope="{row}">
  150. <span>{{ payTypeArray[row.pay_type]?payTypeArray[row.pay_type].type:'' }}</span>
  151. </template>
  152. </el-table-column>
  153. <template v-if="isAdmin">
  154. <el-table-column label="销售" prop="seller_name" align="center">
  155. <template slot-scope="{row}">
  156. {{ row.seller_name }}
  157. </template>
  158. </el-table-column>
  159. <el-table-column label="销售组别" prop="seller_group_name" align="center">
  160. <template slot-scope="{row}">
  161. {{ row.seller_group_name }}
  162. </template>
  163. </el-table-column>
  164. <el-table-column label="销售类型" prop="seller_type" align="center">
  165. <template slot-scope="{row}">
  166. {{ row.seller_type==1?'FICC销售':'权益销售' }}
  167. </template>
  168. </el-table-column>
  169. </template>
  170. <el-table-column label="套餐信息" prop="services_name" align="center">
  171. <template slot-scope="{row}">
  172. {{ row.services_name }}
  173. </template>
  174. </el-table-column>
  175. <template #empty>
  176. <div class="table-noData">
  177. <img src="~@/assets/img/cus_m/nodata.png">
  178. <span>暂无数据</span>
  179. </div>
  180. </template>
  181. </el-table>
  182. <!-- 分页 -->
  183. <m-page :pageSize="searchParams.PageSize" :page_no="searchParams.CurrentIndex"
  184. style="display: flex;justify-content: flex-end;margin-top: 20px;"
  185. :total="total" @handleCurrentChange="changePageNo" />
  186. </div>
  187. </div>
  188. </template>
  189. <script>
  190. import mPage from '@/components/mPage.vue';
  191. import {customInterence} from '@/api/api.js'
  192. export default {
  193. name:'contractStatistics',
  194. components:{mPage},
  195. data() {
  196. const startDate=this.$moment().startOf('year').format('YYYY-MM-DD')
  197. const endDate=this.$moment(new Date()).format('YYYY-MM-DD')
  198. this.payTypeArray=[{id:0,type:''},{id:1,type:'年付'},{id:2,type:'半年付'},{id:3,type:'季付'},{id:4,type:'次付'}]
  199. this.timeTypeData=[{label:'开票日期',value:1},{label:'到款日期',value:2},{label:'开票日期&到款日期',value:3}]
  200. this.invoiceData=[{label:'未开票',value:0 },{label:'已开票',value:1}]
  201. this.paymentData=[{label:'未到款',value:0 },{label:'到款',value:1}]
  202. this.dateButtonData=[{text:'近1周',tabId:1,type:'week',diff:1},{text:'近1月',tabId:2,type:'month',diff:1},
  203. {text:'近2月',tabId:3,type:'month',diff:2},{text:'近3月',tabId:4,type:'month',diff:3}]
  204. return {
  205. currentSdId:process.env.NODE_ENV=='production'?13:233,
  206. serviceList:[],
  207. saleList:[],
  208. checkedService:[],
  209. checkedSale:[],
  210. searchParams:{
  211. CurrentIndex:1,
  212. PageSize:10,
  213. Keyword:'',
  214. ServiceType:'',
  215. StartDate:startDate,
  216. EndDate:endDate,
  217. // 1-开票日期 2-到款日期 3-开票日期&到款日期
  218. TimeType:1,
  219. SortType:'',
  220. SortParam:'',
  221. HasInvoice:'',// 开票状态 0未开票,1已开票
  222. HasPayment:'',// 到款状态 0未到款,1到款
  223. SellerIds:''
  224. },
  225. currentDateTab:0,
  226. searchDate:[startDate,endDate],
  227. tableData:[],
  228. total:0,
  229. invoiceAmountTotal:0,
  230. invoiceAmountList:[],
  231. placementAmountTotal:0,
  232. placementAmountList:[],
  233. invoiceIsFold:true,//开票合计是否收起
  234. placementIsFold:true,//到款合计是否收起
  235. domList:[]
  236. }
  237. },
  238. created(){
  239. if(this.currentSdId==this.AdminId ||this.Role=='admin'||this.Role=='ficc_admin'||this.Role=='rai_admin'){
  240. this.saleSelect()
  241. }
  242. this.getSimpleServiceList()
  243. this.getList()
  244. },
  245. watch:{
  246. searchDate(newVal){
  247. if(newVal){
  248. this.searchParams.StartDate=newVal[0]
  249. this.searchParams.EndDate=newVal[1]
  250. }else{
  251. this.searchParams.StartDate=''
  252. this.searchParams.EndDate=''
  253. }
  254. this.searchList()
  255. }
  256. },
  257. computed:{
  258. isAdmin(){
  259. return localStorage.getItem('Role').indexOf('admin')!=-1 || localStorage.getItem('RoleIdentity').indexOf('ficc_group')!=-1
  260. },
  261. Role() {
  262. let role = localStorage.getItem('Role') || '';
  263. return role;
  264. },
  265. AdminId() {
  266. let role = localStorage.getItem('AdminId') || '';
  267. return role;
  268. },
  269. },
  270. methods: {
  271. getSimpleServiceList(){
  272. customInterence.getSimpleServiceList().then(res=>{
  273. if(res.Ret!=200) return
  274. this.serviceList = res.Data || []
  275. // 后端最外层的数据没有给 service_template_id 删除tag时会报错,手动加
  276. this.serviceList.map((item,index) =>{
  277. item.service_template_id = 500+index
  278. })
  279. })
  280. },
  281. getList(){
  282. // console.log(this.searchParams);
  283. customInterence.getCTContractStatistics(this.searchParams).then(res=>{
  284. if(res.Ret!=200) return
  285. this.tableData=[]
  286. let tempData = res.Data.data_list || []
  287. this.total = res.Data.Paging.Totals
  288. this.invoiceAmountTotal=res.Data.invoice_total
  289. this.invoiceAmountList = res.Data.invoice_currency_total || []
  290. this.placementAmountTotal=res.Data.payment_total
  291. this.placementAmountList = res.Data.payment_currency_total || []
  292. tempData.map((item,index) =>{
  293. item.invoice_payment_list.map((it,ind) =>{
  294. this.tableData.push({
  295. serialNumber:this.searchParams.PageSize*(this.searchParams.CurrentIndex-1)+index+1,
  296. ...item,date:item.start_date+'至'+item.end_date,...it
  297. })
  298. })
  299. })
  300. })
  301. },
  302. searchList(){
  303. this.searchParams.CurrentIndex=1
  304. this.getList()
  305. },
  306. changePageNo(pageNo){
  307. this.searchParams.CurrentIndex = pageNo
  308. this.getList()
  309. },
  310. saleChange(value){
  311. this.searchParams.SellerIds = value.join(',')
  312. this.searchList()
  313. },
  314. serviceChange(value){
  315. this.searchParams.ServiceType = value.join(',')
  316. this.searchList()
  317. },
  318. changeDateType({tabId,type,diff}){
  319. if(this.currentDateTab==tabId) return
  320. this.currentDateTab=tabId
  321. let startOfType='month'
  322. if(type=='week'){
  323. startOfType='isoWeek'
  324. }
  325. this.searchDate=[this.$moment().startOf(startOfType).subtract((diff-1), type+'s').format('YYYY-MM-DD'),
  326. this.$moment().format('YYYY-MM-DD')]
  327. },
  328. foldOrUnfold(type){
  329. if(this.tableData.length==0){
  330. return
  331. }
  332. // type: 0-开票 1-到款
  333. if(type){
  334. this.placementIsFold = !this.placementIsFold
  335. }else{
  336. this.invoiceIsFold = !this.invoiceIsFold
  337. }
  338. },
  339. sortChange({order,prop}){
  340. this.searchParams.SortType=order=='descending'?'desc':order=='ascending'?'asc':''
  341. this.searchParams.SortParam=order?prop:''
  342. this.searchList()
  343. },
  344. disabledCheck(e){
  345. if(!this.domList[e]) return true
  346. return this.domList[e].offsetWidth<=this.domList[e].parentNode.offsetWidth
  347. },
  348. // 获取销售下拉列表
  349. saleSelect(){
  350. customInterence.sellerSelectList({}).then(res=>{
  351. console.log(res)
  352. this.saleList = res.Data.List || []
  353. })
  354. }
  355. },
  356. mounted(){
  357. this.$nextTick(()=>{
  358. this.domList=$('.company-name-span')
  359. // console.log(this.domList);
  360. })
  361. },
  362. updated(){
  363. this.domList=$('.company-name-span')
  364. // console.log(this.domList);
  365. }
  366. }
  367. </script>
  368. <style lang="scss" scoped>
  369. .customer-statistics-container{
  370. min-height: calc(100vh - 110px);
  371. .search-zone{
  372. margin-bottom: 20px;
  373. padding: 20px 30px 12px;
  374. background-color: white;
  375. border-radius: 4px;
  376. .search-box{
  377. display: flex;
  378. align-items: center;
  379. flex-wrap: wrap;
  380. .search-item{
  381. width: 232px;
  382. margin: 0 0 8px 0;
  383. }
  384. .date-box{
  385. display: flex;
  386. align-items: center;
  387. margin: 0 0 8px 0;
  388. .composition-button-tabs{
  389. white-space: nowrap;
  390. .date-button{
  391. height: 40px;
  392. color: #999999;
  393. border: 1px solid #DCDFE6;
  394. margin: 0;
  395. width: 60px;
  396. padding: 12px 6px;
  397. }
  398. .first-button{
  399. border-color: #DCDFE6;
  400. border-style: solid;
  401. border-width: 1px 0 1px 1px;
  402. border-radius: 4px 0 0 4px;
  403. }
  404. .inner-button{
  405. border: 1px solid #DCDFE6;
  406. border-radius: 0;
  407. }
  408. .last-button{
  409. border-color: #DCDFE6;
  410. border-style: solid;
  411. border-width: 1px 1px 1px 0;
  412. border-radius:0 4px 4px 0 ;
  413. }
  414. .selectTab{
  415. color:white;
  416. background-color: #409EFF;
  417. }
  418. }
  419. }
  420. }
  421. }
  422. .amount-show-zone{
  423. display: flex;
  424. align-items: flex-start;
  425. flex-wrap: wrap;
  426. margin-bottom: 10px;
  427. .amount-show-item{
  428. margin-bottom: 10px;
  429. background: #FFFFFF;
  430. // border: 1px solid #DCDFE6;
  431. // box-shadow: 0px 4px 12px rgba(153, 153, 153, 0.08);
  432. border-radius: 8px;
  433. width: 725px;
  434. box-sizing: border-box;
  435. .amount-item-head{
  436. transition: all 0.1s ease;
  437. display: flex;
  438. align-items: center;
  439. justify-content: space-between;
  440. .amount-item-head-title{
  441. font-weight: 600;
  442. font-size: 18px;
  443. color: #000000;
  444. }
  445. .fold-expand-row{
  446. display: flex;
  447. align-items: center;
  448. .amount-item-head-icon{
  449. font-weight: 400;
  450. font-size: 14px;
  451. color: #409EFF;
  452. margin-right: 4px;
  453. }
  454. }
  455. }
  456. .amount-item-body-package{
  457. overflow: hidden;
  458. transition: height 0.1s ease;
  459. .amount-item-body{
  460. padding: 6px 20px 16px;
  461. box-sizing: border-box;
  462. display: flex;
  463. align-items: center;
  464. justify-content: space-between;
  465. .amount-item-body-box{
  466. display: flex;
  467. align-items: center;
  468. width: 220px;
  469. .amount-item-body-info{
  470. display: flex;
  471. flex-direction: column;
  472. justify-content: space-between;
  473. span{
  474. font-weight: 400;
  475. font-size: 14px;
  476. color: #666666;
  477. margin-bottom: 2px;
  478. &:last-child{
  479. font-weight: 700;
  480. font-size: 16px;
  481. color: #333333;
  482. }
  483. }
  484. }
  485. }
  486. }
  487. }
  488. }
  489. }
  490. .table-zone{
  491. padding: 20px 30px 65px;
  492. background-color: white;
  493. border-radius: 4px;
  494. .company-name-column{
  495. max-width: 100%;
  496. display: inline-block;
  497. white-space: nowrap;
  498. overflow: hidden;
  499. text-overflow: ellipsis;
  500. }
  501. .table-noData{
  502. display: flex;
  503. flex-direction: column;
  504. align-items: center;
  505. justify-content: center;
  506. margin: 18vh 0;
  507. img{
  508. height: 110px;
  509. width: 136px;
  510. }
  511. span{
  512. font-weight: 400;
  513. font-size: 16px;
  514. color: #999999;
  515. }
  516. }
  517. }
  518. }
  519. .el-form-item{
  520. margin-bottom: 0;
  521. }
  522. </style>