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.

554 lines
16 KiB

5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
5 months ago
  1. <script setup>
  2. import { ref, onMounted, reactive, computed, nextTick } from 'vue'
  3. import ElementPlus from 'element-plus'
  4. import { ElMessage, ElMessageBox } from 'element-plus'
  5. import { AiFillRead } from 'vue-icons-plus/ai'
  6. import axios from 'axios'
  7. import moment from 'moment'
  8. import API from '@/util/http.js'
  9. import {useI18n} from 'vue-i18n'
  10. const { t } = useI18n()
  11. const defaultTime = [
  12. new Date(2000, 1, 1, 0, 0, 0),
  13. new Date(2000, 2, 1, 23, 59, 59),
  14. ]
  15. const selectData = ref({
  16. jwcode: '',
  17. market: '',
  18. startTime: '',
  19. endTime: '',
  20. })
  21. const permanentBeans = ref(0)
  22. const num = ref(0)
  23. const money = ref(0)
  24. const tableRef = ref(null)
  25. const scrollTableTop = () => {
  26. tableRef.value?.setScrollTop?.(0)
  27. }
  28. //获取当齐全条件的金豆总数
  29. const getTotalBeans = async () => {
  30. try {
  31. const result = await API({
  32. url: '/beanRecharge/statsOnlineBean',
  33. data: {
  34. ...selectData.value,
  35. }
  36. })
  37. if (result.code == 200) {
  38. if (result.data) {
  39. permanentBeans.value = result.data.permanentBean
  40. num.value = result.data.num
  41. money.value = result.data.money
  42. } else {
  43. permanentBeans.value = 0
  44. num.value = 0
  45. money.value = 0
  46. }
  47. console.log('金豆总数获取成功');
  48. }
  49. console.log('获取金豆总数失败:', result.msg);
  50. } catch (error) {
  51. console.log('获取金豆总数出错');
  52. }
  53. }
  54. //时间格式化
  55. const formatTime = (val) => val ? moment(val).format('YYYY-MM-DD HH:mm:ss') : ''
  56. const tableData = ref([])
  57. //存储所有的地区数据
  58. const market = ref()
  59. const getTime = ref([])
  60. //记录当前选中时间按钮状态
  61. const activeTimeRange = ref('')
  62. const handleDatePickerChange = () => {
  63. activeTimeRange.value = ''
  64. console.log('当前选中时间范围', getTime.value);
  65. }
  66. //获取所有地区
  67. const getArea = async () => {
  68. const result = await API({
  69. url: '/beanRecharge/onlineMarket',
  70. data: {}
  71. })
  72. if (result.code == 200) {
  73. market.value = result.data
  74. console.log('线上充值地区获取成功', market.value)
  75. } else {
  76. ElMessage.error(t('elmessage.onlineRechargeAreaError'))
  77. }
  78. }
  79. //获取今天日期
  80. const getToday = () => {
  81. const today = new Date()
  82. const startTime = new Date(
  83. today.getFullYear(),
  84. today.getMonth(),
  85. today.getDate(),
  86. )
  87. const endTime = new Date(
  88. today.getFullYear(),
  89. today.getMonth(),
  90. today.getDate(),
  91. 23, 59, 59
  92. )
  93. getTime.value = [startTime, endTime]
  94. activeTimeRange.value = 'today'
  95. search();
  96. }
  97. const getYesterday = () => {
  98. const yesterday = new Date()
  99. yesterday.setDate(yesterday.getDate() - 1)
  100. const startTime = new Date(
  101. yesterday.getFullYear(),
  102. yesterday.getMonth(),
  103. yesterday.getDate()
  104. )
  105. const endTime = new Date(
  106. yesterday.getFullYear(),
  107. yesterday.getMonth(),
  108. yesterday.getDate(),
  109. 23, 59, 59
  110. )
  111. getTime.value = [startTime, endTime]
  112. activeTimeRange.value = 'yesterday'
  113. search();
  114. }
  115. const get7Days = function () {
  116. const today = new Date()
  117. const startTime = new Date(
  118. today.getFullYear(),
  119. today.getMonth(),
  120. today.getDate() - 6
  121. )
  122. const endTime = new Date(
  123. today.getFullYear(),
  124. today.getMonth(),
  125. today.getDate(),
  126. 23, 59, 59
  127. )
  128. getTime.value = [startTime, endTime]
  129. activeTimeRange.value = '7days'
  130. search();
  131. }
  132. const reset = () => {
  133. selectData.value = {
  134. jwcode: '',
  135. market: '',
  136. startTime: '',
  137. endTime: '',
  138. }
  139. // 重置页码
  140. getObj.value.pageNum = 1
  141. getTime.value = []
  142. activeTimeRange.value = ''
  143. search()
  144. }
  145. //查询
  146. const search = async () => {
  147. getObj.value.pageNum = 1
  148. //判断精网号查询是否为数字
  149. getObj.value.pageNum = 1
  150. if (selectData.value.jwcode) {
  151. const numRef = /^\d{1,9}$/;
  152. if (!numRef.test(selectData.value.jwcode)) {
  153. ElMessage.error(t('elmessage.checkJwcodeFormat'))
  154. return
  155. }
  156. }
  157. await get()
  158. await getTotalBeans()
  159. }
  160. //获取表格数据
  161. const get = async () => {
  162. try {
  163. if (getTime.value != null) {
  164. selectData.value.startTime = formatTime(getTime.value[0])
  165. selectData.value.endTime = formatTime(getTime.value[1])
  166. } else {
  167. selectData.value.startTime = ''
  168. selectData.value.endTime = ''
  169. }
  170. const data = {
  171. ...getObj.value,
  172. beanOnlineRechargeInfo: {
  173. ...selectData.value,
  174. sortField: sortField.value,
  175. sortOrder: sortOrder.value,
  176. },
  177. }
  178. console.log('请求参数:', data);
  179. const result = await API({
  180. url: '/beanRecharge/selectByOnline',
  181. data: data
  182. })
  183. if (result.code == 200) {
  184. tableData.value = result.data.list
  185. total.value = result.data.total
  186. await nextTick()
  187. scrollTableTop()
  188. } else {
  189. ElMessage.error(result.message)
  190. }
  191. } catch (error) {
  192. console.log('搜索失败', error);
  193. }
  194. }
  195. //充值平台
  196. const platform = [
  197. {
  198. value: 1,
  199. label: 'PC'
  200. },
  201. {
  202. value: 2,
  203. label: t('recharge.mobile')
  204. }
  205. ]
  206. //排序字段
  207. //处理表格排序事件
  208. const sortField = ref('')
  209. const sortOrder = ref('')
  210. const handleSortChange = (column) => {
  211. console.log('排序字段:', column.prop)
  212. console.log('排序方式:', column.order)
  213. // 直接使用允许的排序字段列表进行过滤
  214. const allowedFields = ['money', 'num'];
  215. if (allowedFields.includes(column.prop)) {
  216. sortField.value = column.prop;
  217. }
  218. sortOrder.value = column.order === 'ascending' ? 'ASC' : 'DESC';
  219. console.log('传递给后端的排序字段:', sortField.value)
  220. console.log('传递给后端的排序方式:', sortOrder.value)
  221. get();
  222. }
  223. //========================分页===========================
  224. // 搜索对象
  225. const total = ref(0)
  226. const getObj = ref({
  227. pageNum: 1,
  228. pageSize: 50
  229. })
  230. const handlePageSizeChange = (value) => {
  231. getObj.value.pageSize = value
  232. get()
  233. }
  234. const handleCurrentChange = (value) => {
  235. getObj.value.pageNum = value
  236. get()
  237. }
  238. //======================导出=======================
  239. const exportExcel = async () => {
  240. const params = {
  241. ...getObj.value,
  242. beanOnlineRechargeInfo: {
  243. ...selectData.value,
  244. sortField: sortField.value,
  245. sortOrder: sortOrder.value,
  246. },
  247. }
  248. try {
  249. const res = await API({ url: '/export/exportol', data: params })
  250. if (res.code === 200) {
  251. ElMessage.success(t('elmessage.exportSuccess'))
  252. } else {
  253. ElMessage.error(res.message || t('elmessage.exportFailed'))
  254. }
  255. } catch (error) {
  256. console.log('请求失败', error)
  257. ElMessage.error(t('elmessage.exportFailed'))
  258. }
  259. }
  260. const exportListVisible = ref(false)
  261. // 打开导出列表弹窗
  262. const openExportList = () => {
  263. getExportList()
  264. exportListVisible.value = true
  265. }
  266. // 导出列表数据
  267. const exportList = ref([])
  268. // 导出列表加载状态
  269. const exportListLoading = ref(false)
  270. // 获取导出列表
  271. const getExportList = async () => {
  272. exportListLoading.value = true
  273. try {
  274. const result = await API({ url: '/export/export' })
  275. if (result.code === 200) {
  276. const filteredData = result.data.filter(item => {
  277. return item.type === 10;
  278. });
  279. exportList.value = filteredData
  280. } else {
  281. ElMessage.error(result.msg || '获取导出列表失败')
  282. }
  283. } catch (error) {
  284. console.error('获取导出列表出错:', error)
  285. ElMessage.error(t('elmessage.getExportListError'))
  286. } finally {
  287. exportListLoading.value = false
  288. }
  289. }
  290. // 下载导出文件
  291. const downloadExportFile = (item) => {
  292. if (item.state === 2) {
  293. const link = document.createElement('a')
  294. link.href = item.url
  295. link.download = item.fileName
  296. link.click()
  297. } else {
  298. ElMessage.warning(t('elmessage.exportingInProgress'))
  299. }
  300. }
  301. //根据状态返回对应的标签类型
  302. const getTagType = (state) => {
  303. switch (state) {
  304. case 0:
  305. return 'info';
  306. case 1:
  307. return 'primary';
  308. case 2:
  309. return 'success';
  310. case 3:
  311. return 'danger';
  312. default:
  313. return 'info';
  314. }
  315. }
  316. //根据状态返回对应的标签文案
  317. const getTagText = (state) => {
  318. switch (state) {
  319. case 0:
  320. return t('elmessage.pendingExecution');
  321. case 1:
  322. return t('elmessage.executing');
  323. case 2:
  324. return t('elmessage.executed');
  325. case 3:
  326. return t('elmessage.errorExecution');
  327. default:
  328. return t('elmessage.unknownStatus');
  329. }
  330. }
  331. const format3 = (num) => {
  332. // 每三位添加逗号
  333. return num.toLocaleString('en-US')
  334. }
  335. onMounted(async function () {
  336. await get()
  337. await getArea()
  338. await getTotalBeans()
  339. })
  340. </script>
  341. <template>
  342. <el-card class="card1" style="margin-bottom: 0.5vh;">
  343. <el-col style="margin-bottom: 1vh">
  344. <div class="select">
  345. <div class="selectRow">
  346. <el-text class="text" size="large">{{ t('common.jwcode') }}</el-text>
  347. <el-input class="selectContent" v-model="selectData.jwcode" :placeholder="t('common.jwcodePlaceholder')" clearable />
  348. </div>
  349. <div class="selectRow">
  350. <el-text class="text" size="large">{{ t('common.market') }}</el-text>
  351. <el-select class="selectContent" v-model="selectData.market" :placeholder="t('common.marketPlaceholder')" clearable>
  352. <el-option v-for="item in market" :key="item" :label="item" :value="item" />
  353. </el-select>
  354. </div>
  355. <div class="selectRow" style="width: 15vw;">
  356. <el-text class="text" size="large">{{ t('common.orderNo') }}</el-text>
  357. <el-input class="selectContent" v-model="selectData.orderNo" :placeholder="t('common.orderNoPlaceholder')" clearable />
  358. </div>
  359. <div class="selectRow" style="width: 15vw;">
  360. <el-text class="text" size="large">{{ t('common.rechargePlatform') }}</el-text>
  361. <el-select class="selectContent" v-model="selectData.platform" :placeholder="t('common.rechargePlatformPlaceholder')" clearable>
  362. <el-option v-for="item in platform" :key="item.value" :label="item.label" :value="item.value" />
  363. </el-select>
  364. </div>
  365. </div>
  366. </el-col>
  367. <el-col>
  368. <div class="select">
  369. <div class="selectRow" style="width: 36vw;">
  370. <el-text class="text" size="large">{{ t('common.rechargeTime') }}</el-text>
  371. <el-date-picker class="selectContent" v-model="getTime" type="datetimerange" :range-separator="t('common.to')"
  372. :start-placeholder="t('common.startTime')" :end-placeholder="t('common.endTime')" style="margin-right:1vw;width:25vw"
  373. @change="handleDatePickerChange" :default-time="defaultTime" />
  374. <div v-if="false">
  375. <el-button @click="getToday()" style="margin-left: 10px"
  376. :type="activeTimeRange === 'today' ? 'primary' : ''">
  377. </el-button>
  378. <el-button @click="getYesterday()" style="margin-left: 10px"
  379. :type="activeTimeRange === 'yesterday' ? 'primary' : ''">
  380. </el-button>
  381. <el-button @click="get7Days()" style="margin-left: 10px"
  382. :type="activeTimeRange === '7days' ? 'primary' : ''">
  383. 近7天
  384. </el-button>
  385. </div>
  386. </div>
  387. <div class="selectRow" style="justify-content: flex-start;">
  388. <el-button type="primary" @click="search()">{{ t('common.search') }}</el-button>
  389. <el-button type="primary" @click="exportExcel()">{{ t('common.exportExcel') }}</el-button>
  390. <el-button type="primary" @click="openExportList">{{ t('common.viewExportList') }}</el-button>
  391. <el-button type="success" @click="reset()">{{ t('common.reset') }}</el-button>
  392. </div>
  393. </div>
  394. </el-col>
  395. </el-card>
  396. <el-card class="card2">
  397. <div class="goldStatistics">
  398. {{ t('common.rechargeGoldBean') }}{{ format3(num) }}&nbsp;&nbsp;&nbsp;&nbsp;{{ t('common.totalRechargeSGD') }}{{ format3(money) }}
  399. </div>
  400. <!-- 设置表格容器的高度和滚动样式 -->
  401. <div>
  402. <el-table ref="tableRef" :data="tableData" style="width:80vw;" height="65vh" @sort-change="handleSortChange"
  403. :row-style="{ height: '50px' }">
  404. <el-table-column type="index" :label="t('common_list.id')" width="80px" fixed="left">
  405. <template #default="scope">
  406. <span>{{
  407. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  408. }}</span>
  409. </template>
  410. </el-table-column>
  411. <el-table-column fixed="left" prop="name" :label="t('common_list.name')" min-width="120" show-overflow-tooltip />
  412. <el-table-column fixed="left" prop="jwcode" :label="t('common_list.jwcode')" min-width="110px" />
  413. <el-table-column prop="market" :label="t('common_list.market')" min-width="100px" />
  414. <el-table-column prop="orderNo" header-align="center" align="center" :label="t('common_list.orderNo')" min-width="210px" />
  415. <el-table-column prop="num" :label="t('common_list.number')" sortable="custom" min-width="110px" />
  416. <el-table-column prop="money" :label="t('common_list.money')" sortable="custom" min-width="150px" />
  417. <el-table-column prop="platform" :label="t('common_list.rechargePlatform')" min-width="150px" show-overflow-tooltip>
  418. <template #default=scope>
  419. <span v-if="scope.row.platform == 1">PC</span>
  420. <span v-else-if="scope.row.platform == 2">{{ t('common_list.mobile') }}</span>
  421. <span v-else>{{ t('common_list.other') }}</span>
  422. </template>
  423. </el-table-column>
  424. <el-table-column prop="rechargeTime" :label="t('common_list.rechargeTime')" min-width="200px">
  425. <template #default="scope">
  426. {{ moment(scope.row.rechargeTime).format('YYYY-MM-DD HH:mm:ss') }}
  427. </template>
  428. </el-table-column>
  429. </el-table>
  430. </div>
  431. <!-- 分页 -->
  432. <div class="pagination" style="margin-top: 20px">
  433. <el-pagination background :current-page="getObj.pageNum" :page-size="getObj.pageSize" :page-sizes="[5, 10, 20, 50, 100]"
  434. layout="total, sizes, prev, pager, next, jumper" :total="total" @size-change="handlePageSizeChange"
  435. @current-change="handleCurrentChange"></el-pagination>
  436. </div>
  437. </el-card>
  438. <!-- 导出弹窗 -->
  439. <el-dialog v-model="exportListVisible" :title="t('common_export.exportList')" width="80vw" class="custom-height-dialog">
  440. <el-table :data="exportList" style="width:80vw;height: 60vh;" :loading="exportListLoading">
  441. <el-table-column prop="fileName" :label="t('common_export.fileName')" />
  442. <el-table-column prop="state" :label="t('common_export.status')">
  443. <template #default="scope">
  444. <el-tag :type="getTagType(scope.row.state)" :effect="scope.row.state === 3 ? 'light' : 'plain'">
  445. {{ getTagText(scope.row.state) }}
  446. </el-tag>
  447. </template>
  448. </el-table-column>
  449. <el-table-column prop="createTime" :label="t('common_export.createTime')">
  450. <template #default="scope">
  451. {{ moment(scope.row.createTime).format('YYYY-MM-DD HH:mm:ss') }}
  452. </template>
  453. </el-table-column>
  454. <el-table-column :label="t('common_export.operation')">
  455. <template #default="scope">
  456. <el-button type="primary" size="small" @click="downloadExportFile(scope.row)"
  457. :disabled="scope.row.state !== 2">
  458. {{ t('common_export.download') }}
  459. </el-button>
  460. </template>
  461. </el-table-column>
  462. </el-table>
  463. <template #footer>
  464. <div class="dialog-footer">
  465. <el-button text @click="exportListVisible = false">{{ t('common_export.close') }}</el-button>
  466. </div>
  467. </template>
  468. </el-dialog>
  469. </template>
  470. <style scoped lang="scss">
  471. // 搜索的卡片样式
  472. .card1 {
  473. background: #F3FAFE;
  474. }
  475. // 表单的卡片样式
  476. .card2 {
  477. background: #E7F4FD;
  478. }
  479. // 新币总数等等
  480. .goldStatistics {
  481. margin-left: 1vw;
  482. margin-bottom: 1vh;
  483. color: #000000;
  484. font-family: "PingFang SC";
  485. font-size: 16px;
  486. font-style: normal;
  487. font-weight: 700;
  488. line-height: 20px;
  489. }
  490. // 表头背景等
  491. :deep(.el-table__header-wrapper),
  492. :deep(.el-table__body-wrapper),
  493. :deep(.el-table__cell),
  494. /* 表格 */
  495. :deep(.el-table__body td) {
  496. background-color: #F3FAFE !important;
  497. }
  498. /* 表头 */
  499. :deep(.el-table__header th) {
  500. background-color: #F3FAFE !important;
  501. }
  502. /* 鼠标悬停 */
  503. :deep(.el-table__row:hover > .el-table__cell) {
  504. background-color: #E5EBFE !important;
  505. }
  506. /** 搜索的样式 */
  507. .select {
  508. display: flex;
  509. .selectRow {
  510. width: 17vw;
  511. display: flex;
  512. align-items: center;
  513. justify-content: center;
  514. padding: 0 0.5vw;
  515. .text {
  516. width: 5vw;
  517. font-size: 15px;
  518. }
  519. .selectContent {
  520. flex: 1;
  521. }
  522. }
  523. }
  524. </style>