You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

638 lines
20 KiB

2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
3 weeks ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
3 weeks ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
2 months ago
3 weeks ago
2 months ago
3 weeks ago
2 months ago
3 weeks ago
2 months ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
2 months ago
  1. <script setup>
  2. import { ref, onMounted, computed, nextTick } from 'vue'
  3. import { ElMessage } from 'element-plus'
  4. import axios from 'axios'
  5. import moment from 'moment'
  6. import API from '@/util/http'
  7. import { writeFile, utils } from 'xlsx'
  8. import request from "@/util/request.js";
  9. // 定义ref 变量来存储合计数据
  10. const totalPermanentGold = ref(0)
  11. const totalFreeGold = ref(0)
  12. const totalTaskGold = ref(0)
  13. const totalGoldTotal = ref(0)
  14. // 变量
  15. const adminData = ref({})
  16. const getAdminData = async function () {
  17. try {
  18. const result = await API({
  19. url: '/admin/userinfo',
  20. method: 'post',
  21. data: {}
  22. })
  23. adminData.value = result
  24. // console.log('请求成功', result)
  25. console.log('管理员用户信息', adminData.value)
  26. } catch (error) {
  27. console.log('管理员用户信息请求失败', error)
  28. }
  29. }
  30. // 精网号去空格,处理 goldDetail 中的 jwcode
  31. const trimJwCode = () => {
  32. if (goldDetail.value.jwcode) {
  33. goldDetail.value.jwcode = goldDetail.value.jwcode.replace(/\s/g, '');
  34. }
  35. }
  36. //定义平台信息的加载状态
  37. const isLoadingPlatform = ref(false)
  38. // 平台信息
  39. const platform = ref([])
  40. //获取平台信息的函数
  41. const getPlatform = async () => {
  42. isLoadingPlatform.value = true;
  43. try {
  44. const result = await API({
  45. url: '/general/platform',
  46. method: 'post',
  47. data: {}// 这里添加参数
  48. })
  49. // 假设后端返回的是字符串数组,转换为 { value, label } 格式
  50. if (Array.isArray(result.data)) {
  51. platform.value = result.data.map(item => ({ value: item, label: item }));
  52. } else {
  53. console.error('平台信息格式错误', result)
  54. ElMessage.error('平台信息格式错误,请联系管理员')
  55. }
  56. } catch (error) {
  57. console.error('获取平台信息失败:', error);
  58. ElMessage.error('获取平台信息失败,请稍后重试');
  59. } finally {
  60. isLoadingPlatform.value = false
  61. }
  62. }
  63. // 数量更新类型选项
  64. const type = [
  65. {
  66. value: '0',
  67. label: '充值'
  68. },
  69. {
  70. value: '1',
  71. label: '消耗'
  72. },
  73. {
  74. value: '2',
  75. label: '退款'
  76. }
  77. ]
  78. const market = ref([])
  79. /*
  80. // 定义加载状态,获取地区数据
  81. const isLoadingArea = ref(false);
  82. const market = ref([])
  83. const getArea = async () => {
  84. isLoadingArea.value = true;
  85. try {
  86. const result = await API({
  87. url: '/general/market'
  88. });
  89. // 假设后端返回的是字符串数组,转换为 { value, label } 格式
  90. if (Array.isArray(result.data) && typeof result.data[0] === 'string') {
  91. market.value = result.data.map(item => ({ value: item, label: item }));
  92. } else {
  93. market.value = result.data;
  94. }
  95. } catch (error) {
  96. console.error('获取地区数据失败:', error);
  97. ElMessage.error('获取地区数据失败,请稍后重试');
  98. // 可以提供默认数据
  99. market.value = [];
  100. } finally {
  101. isLoadingArea.value = false;
  102. }
  103. };*/
  104. // 地区下拉框
  105. const getMarket = async function () {
  106. try {
  107. const result = await API({
  108. url: '/general/market',
  109. data: {}
  110. })
  111. market.value = result.data
  112. console.log('地区', market.value)
  113. } catch (error) {
  114. console.log('请求失败', error)
  115. }
  116. }
  117. // 充值明细表格
  118. const tableData = ref([])
  119. // 各金币字段变量
  120. // const sumGoldTotal = ref(0)
  121. const permanentGold = ref(0)
  122. //const freeGold = ref(0)
  123. const taskGold = ref(0)
  124. // 搜索===========================================
  125. //分页总条目
  126. const total = ref(100)
  127. // 搜索对象时间
  128. const getTime = ref([])
  129. // 搜索goldDetail
  130. const goldDetail = ref({})
  131. // 搜索对象
  132. const getObj = ref({
  133. pageNum: 1,
  134. pageSize: 50
  135. })
  136. // 开启条件筛选导出excel
  137. // const getPutEX = ref(false)
  138. // 方法
  139. // 搜索===========================================================================
  140. // 搜索方法
  141. const get = async function (val) {
  142. try {
  143. // 搜索参数页码赋值
  144. if (typeof val === 'number') {
  145. getObj.value.pageNum = val
  146. }
  147. if (getTime.value.length === 2) {//检查是否同时选择了开始时间和结束时间,如果不是则置空
  148. goldDetail.value.startTime = moment(getTime.value[0]).format('YYYY-MM-DD HH:mm:ss');
  149. goldDetail.value.endTime = moment(getTime.value[1]).format('YYYY-MM-DD HH:mm:ss');
  150. } else {
  151. goldDetail.value.startTime = ''
  152. goldDetail.value.endTime = ''
  153. }
  154. // 添加排序字段和排序方式到请求参数
  155. goldDetail.value.sortField = sortField.value
  156. goldDetail.value.sortOrder = sortOrder.value
  157. console.log('搜索参数', getObj.value)
  158. console.log('jwcode 类型:', typeof goldDetail.value.jwcode);
  159. console.log('jwcode 值:', goldDetail.value.jwcode);
  160. const requestData = { ...getObj.value, goldDetail: { ...goldDetail.value } };
  161. console.log('最终请求参数', JSON.stringify(requestData, null, 2)); // 打印格式化后的请求参数
  162. const result = await API({
  163. url: '/goldDetail/getGoldDetail',
  164. method: 'post',
  165. data: { ...getObj.value, goldDetail: { ...goldDetail.value } }
  166. })
  167. console.log('响应数据', result)
  168. tableData.value = result.data.list
  169. total.value = result.data.total
  170. // 更新永久金币、任务金币
  171. // permanentGold.value = tableData.value.reduce((total, row) => {
  172. // return total + (Number(row.permanentGold) || 0);
  173. // }, 0);
  174. // taskGold.value = tableData.value.reduce((total, row) => {
  175. // return total + (Number(row.taskGold) || 0);
  176. // }, 0);
  177. //由于免费金币的计算方式是6月免费+12月免费,所以需要单独处理,以及计算总的免费金币和总的金币数放在一块
  178. const totalResult = await API({
  179. url: '/goldDetail/getTotal',
  180. method: 'post',
  181. data: {
  182. goldDetail: { ...goldDetail.value },
  183. /* jwcode: goldDetail.value.jwcode || '',
  184. payPlatform: goldDetail.value.payPlatform || '',
  185. type: goldDetail.value.type || '',
  186. market: goldDetail.value.market || '',
  187. startTime: goldDetail.value.startTime || '',
  188. endTime: goldDetail.value.endTime || ''*/
  189. }
  190. })
  191. if (totalResult.code === 200) {
  192. const data = totalResult.data
  193. totalPermanentGold.value = data.permanentGolds
  194. totalFreeGold.value = data.freeGolds
  195. totalTaskGold.value = data.taskGolds
  196. totalGoldTotal.value = data.sumGolds
  197. } else {
  198. ElMessage.error('获取合计数据失败')
  199. }
  200. } catch (error) {
  201. console.log('请求失败', error)
  202. }
  203. }
  204. // 重置
  205. const reset = function () {
  206. delete goldDetail.value.jwcode
  207. delete goldDetail.value.type
  208. delete goldDetail.value.startTime
  209. delete goldDetail.value.endTime
  210. delete goldDetail.value.market
  211. delete sortField.value
  212. delete sortOrder.value
  213. getTime.value = []
  214. delete goldDetail.value.payPlatform
  215. search()
  216. }
  217. // 搜索,点击查询按钮后触发
  218. const search = function () {
  219. trimJwCode();
  220. getObj.value.pageNum = 1
  221. get()
  222. }
  223. // 今天
  224. const getToday = function () {
  225. const today = moment()
  226. const startTime = today.startOf('day').toDate()
  227. const endTime = today.add(1, 'days').startOf('day').toDate()
  228. getTime.value = [startTime, endTime]
  229. search()
  230. }
  231. // 昨天
  232. const getYesterday = function () {
  233. const today = moment()
  234. const yesterday = moment().subtract(1, 'day')
  235. const startTime = yesterday.startOf('day').toDate()
  236. const endTime = today.startOf('day').toDate()
  237. getTime.value = [startTime, endTime]
  238. search()
  239. }
  240. // 近7天
  241. const get7Days = function () {
  242. const startTime = moment().subtract(6, 'day').startOf('day').toDate()
  243. const endTime = moment().add(1, 'days').startOf('day').toDate()
  244. getTime.value = [startTime, endTime]
  245. search()
  246. }
  247. // // 计算所有记录的金币总数
  248. // const sumGoldTotal = computed(() => {
  249. // return tableData.value.reduce((total, row) => {
  250. // return total + (Number(row.sumGold) || 0);
  251. // }, 0);
  252. // });
  253. // 计算每行免费金币6月+12月的方法
  254. const calculateFreeGold = (row) => {
  255. const freeJune = row.freeJune || 0;
  256. const freeDecember = row.freeDecember || 0;
  257. return (freeJune + freeDecember);
  258. };
  259. // 计算总免费金币的计算属性
  260. // const totalFreeGold = computed(() => {
  261. // return tableData.value.reduce((total, row) => {
  262. // return total + calculateFreeGold(row);
  263. // }, 0);
  264. // });
  265. // 新增排序字段和排序方式
  266. const sortField = ref('')
  267. const sortOrder = ref('')
  268. // 处理排序事件
  269. const handleSortChange = (column) => {
  270. if (column.prop === 'sumGold') {//新增金币总数字段排序
  271. sortField.value = 'sum_gold'
  272. } else if (column.prop === 'permanentGold') {
  273. sortField.value = 'permanent_gold'
  274. } else if (column.prop === 'taskGold') {
  275. sortField.value = 'task_gold'
  276. } else if (column.prop === 'freeGold') {
  277. sortField.value = 'free_gold'
  278. } else if (column.prop === 'auditTime') {//删除了更新时间的creatTime
  279. sortField.value = 'audit_time'
  280. }
  281. sortOrder.value = column.order === 'ascending' ? 'ASC' : 'DESC'
  282. //get()要写在handleSortChange方法里面,不然会导致排序失效
  283. get()
  284. }
  285. const exportExcel = async function () {
  286. const params = {
  287. goldDetail: {
  288. jwcode: goldDetail.value.jwcode || '',
  289. payPlatform: goldDetail.value.payPlatform || '',
  290. type: goldDetail.value.type || '',
  291. market: goldDetail.value.market || '',
  292. startTime: goldDetail.value.startTime || '',
  293. endTime: goldDetail.value.endTime || '',
  294. },
  295. page: getObj.value.pageNum,
  296. size: total.value
  297. }
  298. const res = await API({ url: '/goldDetail/export', data: params })
  299. if (res.code === 200) {
  300. ElMessage.success('导出成功')
  301. }
  302. }
  303. const handlePageSizeChange = function (val) {
  304. getObj.value.pageSize = val
  305. get()
  306. }
  307. const handleCurrentChange = function (val) {
  308. getObj.value.pageNum = val
  309. get()
  310. }
  311. // 挂载
  312. onMounted(async function () {
  313. await get()
  314. await getMarket()
  315. await getAdminData()
  316. await getPlatform() // 调用获取平台信息的函数
  317. })
  318. const exportListVisible = ref(false)
  319. // 打开导出列表弹窗
  320. const openExportList = () => {
  321. getExportList()
  322. exportListVisible.value = true
  323. }
  324. // 导出列表数据
  325. const exportList = ref([])
  326. // 导出列表加载状态
  327. const exportListLoading = ref(false)
  328. // 获取导出列表
  329. const getExportList = async () => {
  330. exportListLoading.value = true
  331. try {
  332. const result = await API({ url: '/export/export' })
  333. if (result.code === 200) {
  334. const filteredData = result.data.filter(item => {
  335. return item.type === 0; //0表示客户明细
  336. });
  337. exportList.value = filteredData
  338. } else {
  339. ElMessage.error(result.msg || '获取导出列表失败')
  340. }
  341. } catch (error) {
  342. console.error('获取导出列表出错:', error)
  343. ElMessage.error('获取导出列表失败,请稍后重试')
  344. } finally {
  345. exportListLoading.value = false
  346. }
  347. }
  348. // 下载导出文件
  349. const downloadExportFile = (item) => {
  350. if (item.state === 2) {
  351. const link = document.createElement('a')
  352. link.href = item.url
  353. link.download = item.fileName
  354. link.click()
  355. } else {
  356. ElMessage.warning('文件还在导出中,请稍后再试')
  357. }
  358. }
  359. //根据状态返回对应的标签类型
  360. const getTagType = (state) => {
  361. switch (state) {
  362. case 0:
  363. return 'info';
  364. case 1:
  365. return 'primary';
  366. case 2:
  367. return'success';
  368. case 3:
  369. return 'danger';
  370. default:
  371. return 'info';
  372. }
  373. }
  374. //根据状态返回对应的标签文案
  375. const getTagText = (state) => {
  376. switch (state) {
  377. case 0:
  378. return '待执行';
  379. case 1:
  380. return '执行中';
  381. case 2:
  382. return'执行完成';
  383. case 3:
  384. return '执行出错';
  385. default:
  386. return '未知状态';
  387. }
  388. }
  389. </script>
  390. <template>
  391. <div>
  392. <el-row>
  393. <el-col>
  394. <el-card style="margin-bottom: 20px;margin-top: 10px;">
  395. <el-row style="margin-bottom: 10px">
  396. <el-col :span="5">
  397. <div class="head-card-element">
  398. <el-text class="mx-1" size="large">精网号</el-text>
  399. <el-input v-model="goldDetail.jwcode" style="width: 150px" placeholder="请输入精网号" clearable />
  400. </div>
  401. </el-col>
  402. <el-col :span="6">
  403. <div class="head-card-element">
  404. <el-text class="mx-1" size="large">平台信息</el-text>
  405. <el-select v-model="goldDetail.payPlatform" placeholder="请选择平台信息" style="width: 160px" clearable
  406. :loading="isLoadingPlatform">
  407. <el-option v-for="item in platform" :key="item.value" :label="item.label" :value="item.value" />
  408. </el-select>
  409. </div>
  410. </el-col>
  411. <el-col :span="7">
  412. <div class="head-card-element">
  413. <el-text class="mx-1" size="large">数量更新类型</el-text>
  414. <el-select v-model="goldDetail.type" placeholder="请选择更新类型" style="width: 160px" clearable>
  415. <el-option v-for="item in type" :key="item.value" :label="item.label" :value="item.value" />
  416. </el-select>
  417. </div>
  418. </el-col>
  419. <el-col :span="6">
  420. <div class="head-card-element">
  421. <el-text class="mx-1" size="large">所属地区</el-text>
  422. <el-select v-model="goldDetail.market" placeholder="请选择所属地区" style="width: 180px" clearable
  423. :loading="isLoadingArea">
  424. <el-option v-for="item in market" :key="item.value || item" :label="item.label || item"
  425. :value="item.value || item" />
  426. </el-select>
  427. </div>
  428. </el-col>
  429. </el-row>
  430. <div class="head-card-element">
  431. <el-text class="mx-1" size="large">更新时间</el-text>
  432. <el-date-picker v-model="getTime" type="datetimerange" range-separator="" start-placeholder="起始时间"
  433. end-placeholder="结束时间" style="margin-right: 50px" />
  434. <el-button @click="getToday()"></el-button>
  435. <el-button @click="getYesterday()"></el-button>
  436. <el-button @click="get7Days()">近7天</el-button>
  437. <el-button type="success" @click="reset()">重置</el-button>
  438. <el-button type="primary" @click="search()">查询</el-button>
  439. <el-button type="primary" @click="exportExcel">导出Excel表格</el-button>
  440. <el-button type="primary" @click="openExportList">查看导出列表</el-button>
  441. </div>
  442. </el-card>
  443. </el-col>
  444. </el-row>
  445. <el-row>
  446. <el-col>
  447. <el-card>
  448. <div>
  449. 金币总数{{ (totalGoldTotal) / 100 }}
  450. 永久金币{{ (totalPermanentGold) / 100 }}
  451. 免费金币{{ (totalFreeGold) / 100 }}
  452. 任务金币{{ (totalTaskGold) / 100 }}
  453. </div>
  454. <div style="height: 584px; overflow-y: auto">
  455. <el-table :data="tableData" style="width: 100%" @sort-change="handleSortChange" height="584px">
  456. <el-table-column type="index" label="序号" width="100px" fixed="left">
  457. <template #default="scope">
  458. <span>{{
  459. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  460. }}</span>
  461. </template>
  462. </el-table-column>
  463. <el-table-column fixed="left" prop="name" label="姓名" width="150" />
  464. <el-table-column fixed="left" prop="jwcode" label="精网号" width="120" />
  465. <el-table-column prop="market" label="所属地区" width="120" />
  466. <el-table-column prop="payPlatform" label="平台信息" width="140">
  467. </el-table-column>
  468. <el-table-column prop="type" label="更新类型" width="110">
  469. <template #default="scope">
  470. <span v-if="scope.row.type === 0">充值</span>
  471. <span v-if="scope.row.type === 1">消耗</span>
  472. <span v-if="scope.row.type === 2">退款</span>
  473. <span v-if="scope.row.type === 3">其他</span>
  474. </template>
  475. </el-table-column>
  476. <el-table-column prop="sumGold" sortable="custom" label="金币数量" width="110">
  477. <template #default="scope">
  478. <!-- <span>
  479. {{
  480. scope.row.type === 1 // 消费类型
  481. ? - scope.row.sumGold / 100
  482. : scope.row.sumGold / 100
  483. }}
  484. </span> -->
  485. <span>{{ scope.row.sumGold / 100 }}</span>
  486. </template>
  487. </el-table-column>
  488. <el-table-column prop="permanentGold" sortable="custom" label="永久金币" width="110">
  489. <template #default="scope">
  490. <span>{{ scope.row.permanentGold / 100 }}</span>
  491. </template>
  492. </el-table-column>
  493. <el-table-column prop="freeGold" sortable="custom" label="免费金币" width="110">
  494. <template #default="scope">
  495. <span>{{ calculateFreeGold(scope.row) / 100 }}</span>
  496. </template>
  497. </el-table-column>
  498. <el-table-column prop="taskGold" sortable="custom" label="任务金币" width="110">
  499. <template #default="scope">
  500. <span>{{ scope.row.taskGold / 100 }}</span>
  501. </template>
  502. </el-table-column>
  503. <el-table-column prop="adminName" label="提交人" width="110" />
  504. <el-table-column prop="auditTime" sortable="custom" label="更新时间" width="210" show-overflow-tooltip>
  505. <template #default="scope">
  506. <span>{{
  507. moment(scope.row.auditTime).format('YYYY-MM-DD HH:mm:ss')
  508. }}</span>
  509. </template>
  510. </el-table-column>
  511. </el-table>
  512. </div>
  513. <!-- 此处分页 -->
  514. <div class="pagination" style="margin-top: 20px">
  515. <el-pagination background :page-size="getObj.pageSize" :page-sizes="[5, 10, 20, 50, 100]"
  516. layout="total, sizes, prev, pager, next, jumper" :total="total" @size-change="handlePageSizeChange"
  517. @current-change="handleCurrentChange"></el-pagination>
  518. </div>
  519. </el-card>
  520. </el-col>
  521. </el-row>
  522. <!-- 导出列表弹窗 -->
  523. <el-dialog v-model="exportListVisible" title="导出列表" width="80%">
  524. <el-table :data="exportList" style="width: 100%" :loading="exportListLoading">
  525. <el-table-column prop="fileName" label="文件名" />
  526. <el-table-column prop="state" label="状态">
  527. <template #default="scope">
  528. <el-tag :type="getTagType(scope.row.state)"
  529. :effect="scope.row.state === 3 ? 'light' : 'plain'">
  530. {{ getTagText(scope.row.state) }}
  531. </el-tag>
  532. </template>
  533. </el-table-column>
  534. <el-table-column prop="createTime" label="创建时间">
  535. <template #default="scope">
  536. {{ moment(scope.row.createTime).format('YYYY-MM-DD HH:mm:ss') }}
  537. </template>
  538. </el-table-column>
  539. <el-table-column label="操作">
  540. <template #default="scope">
  541. <el-button type="primary" size="small" @click="downloadExportFile(scope.row)"
  542. :disabled="scope.row.state !== 2">
  543. 下载
  544. </el-button>
  545. </template>
  546. </el-table-column>
  547. </el-table>
  548. <template #footer>
  549. <div class="dialog-footer">
  550. <el-button text @click="exportListVisible = false">关闭</el-button>
  551. </div>
  552. </template>
  553. </el-dialog>
  554. </div>
  555. </template>
  556. <style scoped>
  557. .pagination {
  558. display: flex;
  559. }
  560. .status {
  561. display: flex;
  562. }
  563. .head-card {
  564. display: flex;
  565. }
  566. .info-panel-header {
  567. font-weight: bold;
  568. margin-bottom: 10px;
  569. }
  570. .dialog-footer {
  571. display: flex;
  572. justify-content: flex-end;
  573. }
  574. .export-status {
  575. margin-top: 15px;
  576. text-align: center;
  577. color: #666;
  578. }
  579. .el-progress-bar__inner {
  580. transition: width 0.5s ease;
  581. }
  582. </style>