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.

619 lines
17 KiB

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