金币系统前端
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.

731 lines
20 KiB

1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
3 weeks ago
3 weeks ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
3 weeks ago
3 weeks ago
1 month ago
1 month ago
1 month ago
3 weeks ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
3 weeks ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
1 month 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. // 定义加载状态
  25. const isLoadingArea = ref(false);
  26. const area = ref([])
  27. const getArea = async () => {
  28. isLoadingArea.value = true;
  29. try {
  30. const result = await API({
  31. url: '/detailY/getarea'
  32. });
  33. // 假设后端返回的是字符串数组,转换为 { value, label } 格式
  34. if (Array.isArray(result.data) && typeof result.data[0] === 'string') {
  35. area.value = result.data.map(item => ({ value: item, label: item }));
  36. } else {
  37. area.value = result.data;
  38. }
  39. } catch (error) {
  40. console.error('获取地区数据失败:', error);
  41. ElMessage.error('获取地区数据失败,请稍后重试');
  42. // 可以提供默认数据
  43. area.value = [];
  44. } finally {
  45. isLoadingArea.value = false;
  46. }
  47. };
  48. // 充值明细表格
  49. const tableData = ref([])
  50. // 各金币总数
  51. const rechargeCoin = ref(0)
  52. const freeCoin = ref(0)
  53. const taskCoin = ref(0)
  54. // 搜索===========================================
  55. //分页总条目
  56. const total = ref(100)
  57. // 搜索对象时间
  58. const getTime = ref([])
  59. // 搜索detailY
  60. const detailY = ref({})
  61. // 搜索对象
  62. const getObj = ref({
  63. pageNum: 1,
  64. pageSize: 50
  65. })
  66. // 开启条件筛选导出excel
  67. const getPutEX = ref(false)
  68. // 支付方式选项
  69. const num = [
  70. {
  71. value: '1',
  72. label: '增加'
  73. },
  74. {
  75. value: '2',
  76. label: '减少'
  77. }
  78. ]
  79. // 方法
  80. // 搜索===========================================================================
  81. // 搜索方法
  82. const get = async function (val) {
  83. try {
  84. // 地区赋值
  85. if (adminData.value.area === '泰国') {
  86. detailY.value.areas = ['泰国', '越南']
  87. } else if (adminData.value.area !== '总部') {
  88. detailY.value.area = adminData.value.area
  89. }
  90. // 搜索参数页码赋值
  91. if (typeof val === 'number') {
  92. getObj.value.pageNum = val
  93. }
  94. if (getTime.value.length === 2) {
  95. detailY.value.startDate = getTime.value[0]
  96. detailY.value.endDate = getTime.value[1]
  97. } else {
  98. detailY.value.startDate = ''
  99. detailY.value.endDate = ''
  100. }
  101. // 添加排序字段和排序方式到请求参数
  102. detailY.value.sortField = sortField.value
  103. detailY.value.sortOrder = sortOrder.value
  104. console.log('搜索参数', getObj.value)
  105. // 发送POST请求
  106. const result = await API({
  107. url: '/detailY',
  108. method: 'post',
  109. data: { ...getObj.value, detailY: { ...detailY.value } }
  110. })
  111. tableData.value = result.data.list
  112. total.value = result.data.total
  113. } catch (error) {
  114. console.log('请求失败', error)
  115. }
  116. }
  117. // 精网号去空格,处理 detailY 中的 jwcode
  118. const trimJwCode = () => {
  119. if (detailY.value.jwcode) {
  120. detailY.value.jwcode = detailY.value.jwcode.replace(/\s/g, '');
  121. }
  122. }
  123. // 搜索
  124. const search = function () {
  125. trimJwCode();
  126. getObj.value.pageNum = 1
  127. get()
  128. }
  129. // 重置
  130. const reset = function () {
  131. delete detailY.value.jwcode
  132. delete detailY.value.num
  133. delete detailY.value.startDate
  134. delete detailY.value.endDate
  135. delete detailY.value.area
  136. delete sortField.value
  137. delete sortOrder.value
  138. getTime.value = []
  139. delete detailY.value.consumePlatform
  140. }
  141. // 挂载
  142. onMounted(async function () {
  143. await getArea()
  144. await getAdminData()
  145. await get()
  146. })
  147. // 导出Excel的方法
  148. const headers = [
  149. '序号',
  150. '姓名',
  151. '精网号',
  152. '所属地区',
  153. '平台信息',
  154. '更新数量',
  155. '更新类型',
  156. '永久金币',
  157. '免费金币',
  158. '任务金币',
  159. '提交人',
  160. '更新时间'
  161. ]
  162. const showExportInfoPanel = ref(false)
  163. // 点击导出按钮直接显示信息面板
  164. const exportExcel = async () => {
  165. try {
  166. console.log('点击导出按钮,尝试显示信息面板');
  167. showExportInfoPanel.value = true;
  168. await nextTick();//组件更新显示信息面板
  169. } catch (error) {
  170. console.error('显示信息面板失败:', error);
  171. ElMessage.error('显示信息面板失败,请稍后重试');
  172. }
  173. };
  174. // 新增导出状态相关变量
  175. const exportProgress = ref(0)
  176. const isExporting = ref(false)
  177. const exportCancelToken = ref(null)
  178. // 优化后的导出方法
  179. const doExportExcel = async () => {
  180. try {
  181. isExporting.value = true
  182. exportProgress.value = 0
  183. showExportInfoPanel.value = false
  184. // 初始化Excel
  185. const wb = utils.book_new()
  186. const ws = utils.aoa_to_sheet([headers])
  187. utils.book_append_sheet(wb, ws, 'Sheet1')
  188. // 流式写入配置
  189. const writer = {
  190. write: (d, o) => {
  191. if (!d) return
  192. utils.sheet_add_aoa(ws, d, { origin: -1 })
  193. }
  194. }
  195. let page = 1
  196. let totalExported = 0
  197. const pageSize = 5000 // 每次请求5000条
  198. let totalRecords = 0
  199. // 首次请求获取总记录数
  200. const firstResult = await API({
  201. url: '/detailY',
  202. method: 'post',
  203. data: {
  204. pageNum: 1,
  205. pageSize,
  206. detailY: { ...detailY.value }
  207. }
  208. })
  209. totalRecords = firstResult.data.total
  210. // 创建取消令牌
  211. const CancelToken = axios.CancelToken
  212. exportCancelToken.value = CancelToken.source()
  213. // 平台信息映射
  214. const platformMap = {
  215. 0: '初始化金币',
  216. 1: 'ERP系统',
  217. 2: 'Homily Chart',
  218. 3: 'Homily Link',
  219. 4: '金币系统'
  220. };
  221. // 更新类型映射
  222. const updateTypeMap = {
  223. 0: '充值',
  224. 1: '消费',
  225. 2: '退款',
  226. 3: '其他'
  227. };
  228. // 处理首次请求的数据
  229. const firstData = firstResult.data.list
  230. if (firstData.length) {
  231. const rows = firstData.map((row, index) => [
  232. totalExported + index + 1,
  233. row.username || '',
  234. row.jwcode || '',
  235. row.area || '',
  236. platformMap[row.consumePlatform] || '',
  237. (row.gold).toFixed(2) || '0.00',
  238. updateTypeMap[row.updateType] || '',
  239. (row.rechargeCoin / 100).toFixed(2) || '0.00',
  240. (row.freeCoin / 100).toFixed(2) || '0.00',
  241. (row.taskCoin / 100).toFixed(2) || '0.00',
  242. row.name || '',
  243. moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') || ''
  244. ])
  245. writer.write(rows)
  246. totalExported += firstData.length
  247. exportProgress.value = Math.round((totalExported / totalRecords) * 100)
  248. page++
  249. }
  250. while (totalExported < totalRecords) {
  251. const result = await API({
  252. url: '/detailY',
  253. method: 'post',
  254. data: {
  255. pageNum: page,
  256. pageSize,
  257. detailY: { ...detailY.value }
  258. },
  259. cancelToken: exportCancelToken.value.token
  260. })
  261. const data = result.data.list
  262. if (!data.length) break
  263. // 转换数据
  264. const rows = data.map((row, index) => [
  265. totalExported + index + 1,
  266. row.username || '',
  267. row.jwcode || '',
  268. row.area || '',
  269. platformMap[row.consumePlatform] || '',
  270. (row.gold / 100).toFixed(2) || '0.00',
  271. updateTypeMap[row.updateType] || '',
  272. (row.rechargeCoin / 100).toFixed(2) || '0.00',
  273. (row.freeCoin / 100).toFixed(2) || '0.00',
  274. (row.taskCoin / 100).toFixed(2) || '0.00',
  275. row.name || '',
  276. moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') || ''
  277. ])
  278. // 流式写入
  279. writer.write(rows)
  280. totalExported += data.length
  281. exportProgress.value = Math.round((totalExported / totalRecords) * 100)
  282. // 内存控制:每500页释放内存
  283. if (page % 500 === 0) {
  284. await new Promise(resolve => setTimeout(resolve, 0))
  285. }
  286. page++
  287. }
  288. // 生成最终文件
  289. writeFile(wb, '客户金币明细.xlsx')
  290. ElMessage.success(`导出成功,共${totalExported}条数据`)
  291. } catch (error) {
  292. if (!axios.isCancel(error)) {
  293. ElMessage.error(`导出失败: ${error.message}`)
  294. }
  295. } finally {
  296. isExporting.value = false
  297. exportCancelToken.value = null
  298. }
  299. }
  300. // 新增取消导出方法
  301. const cancelExport = () => {
  302. if (exportCancelToken.value) {
  303. exportCancelToken.value.cancel('用户取消导出')
  304. ElMessage.warning('导出已取消')
  305. isExporting.value = false
  306. }
  307. }
  308. const putExcel = ref({
  309. startDate: new Date(),
  310. endDate: new Date(new Date().setDate(new Date().getDate() + 1))
  311. })
  312. // 新增校验精网号的方法
  313. const checkJwCode = async (jwcode) => {
  314. try {
  315. const result = await API({
  316. url: '/recharge/user',
  317. method: 'post',
  318. data: {
  319. jwcode,
  320. area: adminData.value.area
  321. }
  322. })
  323. // 根据后端返回的 code 判断精网号是否存在
  324. return result.code !== 0
  325. } catch (error) {
  326. console.log('校验精网号失败', error)
  327. return false
  328. }
  329. }
  330. // 选消费平台
  331. const platform = [
  332. {
  333. value: '4',
  334. label: '金币系统'
  335. },
  336. {
  337. value: '1',
  338. label: 'ERP系统'
  339. },
  340. {
  341. value: '2',
  342. label: 'Homily Chart'
  343. },
  344. {
  345. value: '3',
  346. label: 'Homily Link'
  347. },
  348. {
  349. value: '0',
  350. label: '初始化金币'
  351. }
  352. ]
  353. // 新增排序字段和排序方式
  354. const sortField = ref('')
  355. const sortOrder = ref('')
  356. // 处理排序事件
  357. const handleSortChange = (column) => {
  358. if (column.prop === 'rechargeCoin') {
  359. sortField.value = 'recharge_coin'
  360. } else if (column.prop === 'taskCoin') {
  361. sortField.value = 'task_coin'
  362. } else if (column.prop === 'freeCoin') {
  363. sortField.value = 'free_coin'
  364. } else if (column.prop === 'createTime') {
  365. sortField.value = 'create_time'
  366. } else if (column.prop === 'gold') {
  367. sortField.value = 'gold'
  368. }
  369. sortOrder.value = column.order === 'ascending' ? 'ASC' : 'DESC'
  370. }
  371. get()
  372. const handlePageSizeChange = function (val) {
  373. getObj.value.pageSize = val
  374. get()
  375. }
  376. const handleCurrentChange = function (val) {
  377. getObj.value.pageNum = val
  378. get()
  379. }
  380. </script>
  381. <template>
  382. <!-- 导出excel提前展示的信息面板 -->
  383. <el-dialog
  384. v-model="showExportInfoPanel"
  385. title="导出信息确认"
  386. width="400px"
  387. :close-on-click-modal="false"
  388. >
  389. <div class="info-panel-header">导出信息</div>
  390. <div>你正在导出以下数据</div>
  391. <!-- 直接使用 detailY 显示信息添加可选链操作符 -->
  392. <!-- detailY是一个ref所以在模板中应该直接使用detailY.consumePlatform
  393. 而不是detailY.value.consumePlatform
  394. 因为在模板中ref变量会自动解包不需要.value
  395. 例如在代码中用户可能错误地在模板中使用了detailY.value但实际上应该直接使用detailY -->
  396. <div v-if="detailY.jwcode">精网号{{ detailY.jwcode || '' }}</div>
  397. <div v-if="detailY.consumePlatform">平台信息{{ detailY.consumePlatform ? (platform.find(item => item.value === detailY.consumePlatform)?.label) : '' }}</div>
  398. <div v-if="detailY.num">数量更新类型{{ detailY.num ? (num.find(item => item.value === detailY.num)?.label || '') : '' }}</div>
  399. <div v-if="detailY.area">所属地区{{ detailY.area || '' }}</div>
  400. <div v-if="Array.isArray(getTime) && getTime.length >= 2">
  401. <span>更新时间</span>
  402. <!-- 直接使用 getTime 而非 getTime.value -->
  403. <span v-if="Array.isArray(getTime) && getTime.length >= 2">
  404. {{ moment(getTime[0]).format('YYYY-MM-DD') }} {{ moment(getTime[1]).format('YYYY-MM-DD') }}
  405. </span>
  406. <span v-else></span>
  407. </div>
  408. <template #footer>
  409. <span class="dialog-footer">
  410. <el-button @click="showExportInfoPanel = false">取消</el-button>
  411. <el-button type="primary" @click="doExportExcel">导出</el-button>
  412. </span>
  413. </template>
  414. </el-dialog>
  415. <!-- 导出进度弹窗 -->
  416. <el-dialog
  417. v-model="isExporting"
  418. title="正在导出"
  419. width="400px"
  420. :close-on-click-modal="false"
  421. :show-close="false"
  422. >
  423. <el-progress
  424. :percentage="exportProgress"
  425. :stroke-width="15"
  426. striped
  427. animated
  428. />
  429. <div class="export-status">
  430. 已导出 {{ Math.round((exportProgress / 100) * total) }} / {{ total }}
  431. </div>
  432. <template #footer>
  433. <el-button type="danger" @click="cancelExport">取消导出</el-button>
  434. </template>
  435. </el-dialog>
  436. <el-row>
  437. <el-col>
  438. <el-card style="margin-bottom: 20px">
  439. <el-row style="margin-bottom: 10px">
  440. <el-col :span="6">
  441. <div class="head-card-element">
  442. <el-text class="mx-1" size="large">精网号</el-text>
  443. <el-input
  444. v-model="detailY.jwcode"
  445. style="width: 240px"
  446. placeholder="请输入精网号"
  447. clearable
  448. />
  449. </div>
  450. </el-col>
  451. <el-col :span="6">
  452. <div class="head-card-element">
  453. <el-text class="mx-1" size="large">平台信息</el-text>
  454. <el-select
  455. v-model="detailY.consumePlatform"
  456. placeholder="请选择平台信息"
  457. style="width: 200px"
  458. clearable
  459. >
  460. <el-option
  461. v-for="item in platform"
  462. :key="item.value"
  463. :label="item.label"
  464. :value="item.value"
  465. />
  466. </el-select>
  467. </div>
  468. </el-col>
  469. <el-col :span="6">
  470. <div class="head-card-element">
  471. <el-text class="mx-1" size="large">数量更新类型</el-text>
  472. <el-select
  473. v-model="detailY.num"
  474. placeholder="请选择更新类型"
  475. style="width: 200px"
  476. clearable
  477. >
  478. <el-option
  479. v-for="item in num"
  480. :key="item.value"
  481. :label="item.label"
  482. :value="item.value"
  483. />
  484. </el-select>
  485. </div>
  486. </el-col>
  487. <el-col :span="6">
  488. <div class="head-card-element">
  489. <el-text class="mx-1" size="large">所属地区</el-text>
  490. <el-select
  491. v-model="detailY.area"
  492. placeholder="请选择所属地区"
  493. style="width: 240px"
  494. clearable
  495. :loading="isLoadingArea"
  496. >
  497. <el-option
  498. v-for="item in area"
  499. :key="item.value || item"
  500. :label="item.label || item"
  501. :value="item.value || item"
  502. />
  503. </el-select>
  504. </div>
  505. </el-col>
  506. </el-row>
  507. <div class="head-card-element">
  508. <el-text class="mx-1" size="large">更新时间</el-text>
  509. <el-date-picker
  510. v-model="getTime"
  511. type="datetimerange"
  512. range-separator="至"
  513. start-placeholder="起始时间"
  514. end-placeholder="结束时间"
  515. style="margin-right: 700px"
  516. />
  517. <el-button type="success" @click="exportExcel">导出Excel表格</el-button>
  518. <el-button type="success" @click="reset()">重置</el-button>
  519. <el-button type="primary" @click="search()">查询</el-button>
  520. </div>
  521. </el-card>
  522. </el-col>
  523. </el-row>
  524. <el-row>
  525. <el-col>
  526. <el-card>
  527. <div style="height: 584px; overflow-y: auto">
  528. <el-table
  529. :data="tableData"
  530. style="width: 100%"
  531. @sort-change="handleSortChange"
  532. height="584px"
  533. >
  534. <el-table-column
  535. type="index"
  536. label="序号"
  537. width="100px"
  538. fixed="left"
  539. >
  540. <template #default="scope">
  541. <span>{{
  542. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  543. }}</span>
  544. </template>
  545. </el-table-column>
  546. <el-table-column
  547. fixed="left"
  548. prop="username"
  549. label="姓名"
  550. width="150"
  551. />
  552. <el-table-column
  553. fixed="left"
  554. prop="jwcode"
  555. label="精网号"
  556. width="120"
  557. />
  558. <el-table-column prop="area" label="所属地区" width="120" />
  559. <el-table-column
  560. prop="consumePlatform"
  561. label="平台信息"
  562. width="140"
  563. >
  564. <template #default="scope">
  565. <!-- 使用非严格相等比较 -->
  566. <span v-if="scope.row.consumePlatform == 0">初始化金币</span>
  567. <span v-if="scope.row.consumePlatform == 1">ERP系统</span>
  568. <span v-if="scope.row.consumePlatform == 3">Homily Link</span>
  569. <span v-if="scope.row.consumePlatform == 2">Homily Chart</span>
  570. <span v-if="scope.row.consumePlatform == 4">金币系统</span>
  571. </template>
  572. </el-table-column>
  573. <el-table-column
  574. prop="gold"
  575. label="更新数量"
  576. width="120"
  577. sortable="custom"
  578. >
  579. <template #default="scope">
  580. <span>{{ scope.row.gold / 100 }}</span>
  581. </template>
  582. </el-table-column>
  583. <el-table-column prop="updateType" label="更新类型" width="110">
  584. <!-- 模板内容 -->
  585. <template #default="scope">
  586. <span v-if="scope.row.updateType == 1">消费</span>
  587. <span v-if="scope.row.updateType == 0">充值</span>
  588. <span v-if="scope.row.updateType == 2">退款</span>
  589. <span v-if="scope.row.updateType == 3">其他</span>
  590. </template>
  591. </el-table-column>
  592. <el-table-column
  593. prop="rechargeCoin"
  594. sortable="custom"
  595. label="永久金币"
  596. width="110"
  597. >
  598. <template #default="scope">
  599. <span>{{ scope.row.rechargeCoin / 100 }}</span>
  600. </template>
  601. </el-table-column>
  602. <el-table-column
  603. prop="freeCoin"
  604. sortable="custom"
  605. label="免费金币"
  606. width="110"
  607. >
  608. <template #default="scope">
  609. <span>{{ scope.row.freeCoin / 100 }}</span>
  610. </template>
  611. </el-table-column>
  612. <el-table-column
  613. prop="taskCoin"
  614. sortable="custom"
  615. label="任务金币"
  616. width="110"
  617. >
  618. <template #default="scope">
  619. <span>{{ scope.row.taskCoin / 100 }}</span>
  620. </template>
  621. </el-table-column>
  622. <el-table-column prop="name" label="提交人" width="110" />
  623. <el-table-column
  624. prop="createTime"
  625. sortable="custom"
  626. label="更新时间"
  627. width="210"
  628. show-overflow-tooltip
  629. >
  630. <template #default="scope">
  631. <span>{{
  632. moment(scope.row.createTime).format('YYYY-MM-DD HH:mm:ss')
  633. }}</span>
  634. </template>
  635. </el-table-column>
  636. </el-table>
  637. </div>
  638. <!-- 分页 -->
  639. <div class="pagination" style="margin-top: 20px">
  640. <el-pagination
  641. background
  642. :page-size="getObj.pageSize"
  643. :page-sizes="[5, 10, 20, 50, 100]"
  644. layout="total, sizes, prev, pager, next, jumper"
  645. :total="total"
  646. @size-change="handlePageSizeChange"
  647. @current-change="handleCurrentChange"
  648. ></el-pagination>
  649. </div>
  650. </el-card>
  651. </el-col>
  652. </el-row>
  653. </template>
  654. <style scoped>
  655. .pagination {
  656. display: flex;
  657. }
  658. .status {
  659. display: flex;
  660. }
  661. .head-card {
  662. display: flex;
  663. }
  664. .info-panel-header {
  665. font-weight: bold;
  666. margin-bottom: 10px;
  667. }
  668. .dialog-footer {
  669. display: flex;
  670. justify-content: flex-end;
  671. }
  672. .export-status {
  673. margin-top: 15px;
  674. text-align: center;
  675. color: #666;
  676. }
  677. .el-progress-bar__inner {
  678. transition: width 0.5s ease;
  679. }
  680. </style>