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.

504 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. }
  187. // 搜索,点击查询按钮后触发
  188. const search = function () {
  189. trimJwCode();
  190. getObj.value.pageNum = 1
  191. get()
  192. }
  193. // 今天
  194. const getToday = function () {
  195. const today = moment()
  196. const startTime = today.startOf('day').toDate()
  197. const endTime = today.add(1, 'days').startOf('day').toDate()
  198. getTime.value = [startTime, endTime]
  199. search()
  200. }
  201. // 昨天
  202. const getYesterday = function () {
  203. const today = moment()
  204. const yesterday = moment().subtract(1, 'day')
  205. const startTime = yesterday.startOf('day').toDate()
  206. const endTime = today.startOf('day').toDate()
  207. getTime.value = [startTime, endTime]
  208. search()
  209. }
  210. // 近7天
  211. const get7Days = function () {
  212. const startTime = moment().subtract(6, 'day').startOf('day').toDate()
  213. const endTime = moment().add(1, 'days').startOf('day').toDate()
  214. getTime.value = [startTime, endTime]
  215. search()
  216. }
  217. // 计算所有记录的金币总数
  218. const sumGoldTotal = computed(() => {
  219. return tableData.value.reduce((total, row) => {
  220. return total + (Number(row.sumGold) || 0);
  221. }, 0);
  222. });
  223. // 计算每行免费金币6月+12月的方法
  224. const calculateFreeGold = (row) => {
  225. const freeJune = row.freeJune || 0;
  226. const freeDecember = row.freeDecember || 0;
  227. return (freeJune + freeDecember);
  228. };
  229. // 计算总免费金币的计算属性
  230. const totalFreeGold = computed(() => {
  231. return tableData.value.reduce((total, row) => {
  232. return total + calculateFreeGold(row);
  233. }, 0);
  234. });
  235. // 验证页码跳转输入框的数字是否合法
  236. const checkNumber = function () {
  237. if (typeof parseInt(getObj.value.pageNum) === 'number') {
  238. console.log('总共有多少页' + Math.ceil(total.value / getObj.value.pageSize))
  239. if (
  240. getObj.value.pageNum > 0 &&
  241. getObj.value.pageNum <= Math.ceil(total.value / getObj.value.pageSize)
  242. ) {
  243. getObj.value.pageNum = parseInt(getObj.value.pageNum)
  244. console.log('输入的数字合法')
  245. get()
  246. } else {
  247. //提示
  248. ElMessage({
  249. type: 'error',
  250. message: '请检查输入内容'
  251. })
  252. }
  253. } else {
  254. //提示
  255. ElMessage({
  256. type: 'error',
  257. message: '输入非法'
  258. })
  259. }
  260. }
  261. // 新增排序字段和排序方式
  262. const sortField = ref('')
  263. const sortOrder = ref('')
  264. // 处理排序事件
  265. const handleSortChange = (column) => {
  266. if (column.prop === 'sumGold') {//新增金币总数字段排序
  267. sortField.value = 'sum_gold'
  268. } else if (column.prop === ' permanentGold') {
  269. sortField.value = 'permanent_gold'
  270. } else if (column.prop === 'taskGold') {
  271. sortField.value = 'task_gold'
  272. } else if (column.prop === 'freeGold') {
  273. sortField.value = 'free_gold'
  274. } else if (column.prop === 'auditTime') {//删除了更新时间的creatTime
  275. sortField.value = 'audit_time'
  276. } else if (column.prop === 'gold') {
  277. sortField.value = 'gold'
  278. }
  279. sortOrder.value = column.order === 'ascending' ? 'ASC' : 'DESC'
  280. //get()要写在handleSortChange方法里面,不然会导致排序失效
  281. get()
  282. }
  283. const handlePageSizeChange = function (val) {
  284. getObj.value.pageSize = val
  285. get()
  286. }
  287. const handleCurrentChange = function (val) {
  288. getObj.value.pageNum = val
  289. get()
  290. }
  291. // 挂载
  292. onMounted(async function () {
  293. await get()
  294. await getArea()
  295. await getAdminData()
  296. await getPlatform() // 调用获取平台信息的函数
  297. })
  298. </script>
  299. <template>
  300. <div>
  301. <el-row>
  302. <el-col>
  303. <el-card style="margin-bottom: 20px;margin-top: 10px;">
  304. <el-row style="margin-bottom: 10px">
  305. <el-col :span="5">
  306. <div class="head-card-element">
  307. <el-text class="mx-1" size="large">精网号</el-text>
  308. <el-input v-model="goldDetail.jwcode" style="width: 150px" placeholder="请输入精网号" clearable />
  309. </div>
  310. </el-col>
  311. <el-col :span="6">
  312. <div class="head-card-element">
  313. <el-text class="mx-1" size="large">平台信息</el-text>
  314. <el-select v-model="goldDetail.payPlatform" placeholder="请选择平台信息" style="width: 160px" clearable
  315. :loading="isLoadingPlatform">
  316. <el-option v-for="item in platform" :key="item.value" :label="item.label" :value="item.value" />
  317. </el-select>
  318. </div>
  319. </el-col>
  320. <el-col :span="7">
  321. <div class="head-card-element">
  322. <el-text class="mx-1" size="large">数量更新类型</el-text>
  323. <el-select v-model="goldDetail.type" placeholder="请选择更新类型" style="width: 160px" clearable>
  324. <el-option v-for="item in type" :key="item.value" :label="item.label" :value="item.value" />
  325. </el-select>
  326. </div>
  327. </el-col>
  328. <el-col :span="6">
  329. <div class="head-card-element">
  330. <el-text class="mx-1" size="large">所属地区</el-text>
  331. <el-select v-model="goldDetail.market" placeholder="请选择所属地区" style="width: 180px" clearable
  332. :loading="isLoadingArea">
  333. <el-option v-for="item in market" :key="item.value || item" :label="item.label || item"
  334. :value="item.value || item" />
  335. </el-select>
  336. </div>
  337. </el-col>
  338. </el-row>
  339. <div class="head-card-element">
  340. <el-text class="mx-1" size="large">更新时间</el-text>
  341. <el-date-picker v-model="getTime" type="datetimerange" range-separator="" start-placeholder="起始时间"
  342. end-placeholder="结束时间" style="margin-right: 50px" />
  343. <el-button @click="getToday()"></el-button>
  344. <el-button @click="getYesterday()"></el-button>
  345. <el-button @click="get7Days()">近7天</el-button>
  346. <el-button type="success" @click="reset()">重置</el-button>
  347. <el-button type="primary" @click="search()">查询</el-button>
  348. <el-button type="primary" @click="exportExcel">导出Excel表格</el-button>
  349. </div>
  350. </el-card>
  351. </el-col>
  352. </el-row>
  353. <el-row>
  354. <el-col>
  355. <el-card>
  356. <div>
  357. 金币总数{{ Math.abs(sumGoldTotal) / 100 }}
  358. 永久金币{{ Math.abs(permanentGold) / 100 }}
  359. 免费金币{{ Math.abs(totalFreeGold) / 100 }}
  360. 任务金币{{ Math.abs(taskGold) / 100 }}
  361. </div>
  362. <div style="height: 584px; overflow-y: auto">
  363. <el-table :data="tableData" style="width: 100%" @sort-change="handleSortChange" height="584px">
  364. <el-table-column type="index" label="序号" width="100px" fixed="left">
  365. <template #default="scope">
  366. <span>{{
  367. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  368. }}</span>
  369. </template>
  370. </el-table-column>
  371. <el-table-column fixed="left" prop="name" label="姓名" width="150" />
  372. <el-table-column fixed="left" prop="jwcode" label="精网号" width="120" />
  373. <el-table-column prop="market" label="所属地区" width="120" />
  374. <el-table-column prop="payPlatform" label="平台信息" width="140">
  375. </el-table-column>
  376. <el-table-column prop="type" label="更新类型" width="110">
  377. <template #default="scope">
  378. <span v-if="scope.row.type === 0">充值</span>
  379. <span v-if="scope.row.type === 1">消耗</span>
  380. <span v-if="scope.row.type === 2">退款</span>
  381. </template>
  382. </el-table-column>
  383. <el-table-column prop="sumGold" sortable="custom" label="金币数量" width="110">
  384. <template #default="scope">
  385. <span>
  386. {{
  387. scope.row.type === 1 // 消费类型
  388. ? - scope.row.sumGold / 100
  389. : scope.row.sumGold / 100
  390. }}
  391. </span>
  392. </template>
  393. </el-table-column>
  394. <el-table-column prop=" permanentGold" sortable="custom" label="永久金币" width="110">
  395. <template #default="scope">
  396. <span>{{ scope.row.permanentGold / 100 }}</span>
  397. </template>
  398. </el-table-column>
  399. <el-table-column sortable="custom" label="免费金币" width="110">
  400. <template #default="scope">
  401. <span>{{ calculateFreeGold(scope.row) / 100 }}</span>
  402. </template>
  403. </el-table-column>
  404. <el-table-column prop="taskGold" sortable="custom" label="任务金币" width="110">
  405. <template #default="scope">
  406. <span>{{ scope.row.taskGold / 100 }}</span>
  407. </template>
  408. </el-table-column>
  409. <el-table-column prop="adminName" label="提交人" width="110" />
  410. <el-table-column prop="auditTime" sortable="custom" label="更新时间" width="210" show-overflow-tooltip>
  411. <template #default="scope">
  412. <span>{{
  413. moment(scope.row.auditTime).format('YYYY-MM-DD HH:mm:ss')
  414. }}</span>
  415. </template>
  416. </el-table-column>
  417. </el-table>
  418. </div>
  419. <!-- 此处分页 -->
  420. <div class="pagination" style="margin-top: 20px">
  421. <el-pagination background :page-size="getObj.pageSize" :page-sizes="[5, 10, 20, 50, 100]"
  422. layout="total, sizes, prev, pager, next, jumper" :total="total" @size-change="handlePageSizeChange"
  423. @current-change="handleCurrentChange"></el-pagination>
  424. </div>
  425. </el-card>
  426. </el-col>
  427. </el-row>
  428. </div>
  429. </template>
  430. <style scoped>
  431. .pagination {
  432. display: flex;
  433. }
  434. .status {
  435. display: flex;
  436. }
  437. .head-card {
  438. display: flex;
  439. }
  440. .info-panel-header {
  441. font-weight: bold;
  442. margin-bottom: 10px;
  443. }
  444. .dialog-footer {
  445. display: flex;
  446. justify-content: flex-end;
  447. }
  448. .export-status {
  449. margin-top: 15px;
  450. text-align: center;
  451. color: #666;
  452. }
  453. .el-progress-bar__inner {
  454. transition: width 0.5s ease;
  455. }
  456. </style>