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.

531 lines
17 KiB

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