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.

505 lines
16 KiB

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