contractStatistics.vue 18 KB

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