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.

693 lines
22 KiB

2 weeks ago
2 weeks ago
2 weeks ago
1 week ago
1 week ago
  1. <script setup>
  2. import { computed, onMounted, ref } from 'vue'
  3. import { dayjs, ElMessage } from 'element-plus'
  4. import request from '@/util/http.js'
  5. import API from '@/util/http.js'
  6. import moment from 'moment'
  7. // 之后整理一下
  8. /*
  9. ====================工具方法==============================
  10. */
  11. const defaultTime = [
  12. new Date(2000, 1, 1, 0, 0, 0),
  13. new Date(2000, 2, 1,23 , 59, 59),
  14. ]
  15. const format3 = (num) => {
  16. // 每三位添加逗号
  17. return num.toLocaleString('en-US')
  18. }
  19. // 时间格式化
  20. const formatTime = (val) => val ? dayjs(val).format('YYYY-MM-DD HH:mm:ss') : ''
  21. /*
  22. ====================数据=================================
  23. */
  24. //这是获取用户信息的接口
  25. const adminData = ref({})
  26. // 充值明细表格
  27. const tableData = ref([])
  28. // 搜索beanConsumeLive 表单
  29. const beanConsumeLive = ref({
  30. jwcode: null,
  31. dept: "",
  32. type: "",
  33. gift: "",
  34. liveChannel: "",
  35. liveName: "",
  36. startTime: '',
  37. endTime: '',
  38. sortField: '',
  39. sortOrder: ''
  40. })
  41. // 礼物列表
  42. const gifts = ref([])
  43. // 获取礼物列表的方法
  44. const getGift = async function () {
  45. try {
  46. const result = await request({
  47. url: '/beanConsume/getLiveGift', // todo 换成实际接口地址
  48. data: {account:adminData.value.account}
  49. })
  50. console.log('请求礼物列表成功', result)
  51. // 存储礼物数据
  52. gifts.value = result.data
  53. console.log('礼物数据', gifts.value)
  54. } catch (error) {
  55. console.log('请求礼物列表失败', error)
  56. ElMessage({
  57. type: 'error',
  58. message: '获取礼物列表失败,请稍后重试'
  59. })
  60. }
  61. }
  62. // 频道列表
  63. const channels = ref([])
  64. // 获取频道列表的方法
  65. const getChannel = async function () {
  66. try {
  67. const result = await request({
  68. url: '/beanConsume/getLiveChannel', // todo 换成实际接口地址
  69. data: {account:adminData.value.account}
  70. })
  71. console.log('请求频道列表成功', result)
  72. // 存储频道数据
  73. channels.value = result.data
  74. console.log('频道数据', channels.value)
  75. } catch (error) {
  76. console.log('请求频道列表失败', error)
  77. ElMessage({
  78. type: 'error',
  79. message: '获取频道列表失败,请稍后重试'
  80. })
  81. }
  82. }
  83. // 抽离类型选项到响应式数组
  84. const consumeTypes = ref([
  85. { label: '发礼物', value: 1 },
  86. { label: '发红包', value: 2 },
  87. { label: '发福袋', value: 3 },
  88. { label: '付费直播', value: 4 },
  89. { label: '加入粉丝团', value: 5 },
  90. { label: '发弹幕', value: 6 }
  91. ])
  92. // 处理类型选择变化
  93. const handleTypeChange = (value) => {
  94. if (value !== 1) {
  95. beanConsumeLive.value.gift = ''
  96. }
  97. }
  98. //------------------------
  99. // 标记当前激活的时间范围按钮
  100. const activeTimeRange = ref('')
  101. // 日期选择器变化时清除按钮激活状态
  102. const handleDatePickerChange = () => {
  103. activeTimeRange.value = ''
  104. }
  105. // 搜索对象
  106. const getObj = ref({
  107. pageNum: 1,
  108. pageSize: 50
  109. })
  110. //分页总条目
  111. const total = ref(100)
  112. // 搜索对象时间
  113. const getTime = ref({
  114. startTime: '',
  115. endTime: ''
  116. })
  117. // 搜索地区列表
  118. const dept = ref([])
  119. // 获取地区列表的方法
  120. const getDept = async function () {
  121. try {
  122. // 发送请求获取地区列表
  123. const result = await request({
  124. // url: '/general/dept',
  125. url: '/beanConsume/getDept', // todo 换成实际接口地址
  126. data: {account:adminData.value.account}
  127. })
  128. console.log('请求地区列表成功', result)
  129. // 存储地区数据
  130. dept.value = result.data
  131. console.log('地区数据', dept.value)
  132. } catch (error) {
  133. console.log('请求地区列表失败', error)
  134. ElMessage({
  135. type: 'error',
  136. message: '获取地区列表失败,请稍后重试'
  137. })
  138. }
  139. }
  140. // 新增排序字段和排序方式
  141. const sortField = ref('')
  142. const sortOrder = ref('')
  143. // 合计数
  144. const permanentBean = ref(0)
  145. const freeBean = ref(0)
  146. const totalNum = ref(0)
  147. /*
  148. ====================方法=================================
  149. */
  150. // 获取登录用户信息
  151. const getAdminData = async function () {
  152. try {
  153. const result = await request({
  154. url: '/admin/userinfo',
  155. data: {}
  156. })
  157. adminData.value = result
  158. console.log('请求成功', result)
  159. console.log('用户信息', adminData.value)
  160. } catch (error) {
  161. console.log('请求失败', error)
  162. }
  163. }
  164. const selectLiveBy = async function (val) {
  165. try {
  166. // 搜索参数页码赋值
  167. if (typeof val === 'number') {
  168. getObj.value.pageNum = val
  169. }
  170. // 搜索参数时间赋值
  171. // 修复时间范围处理
  172. if (Array.isArray(getTime.value) && getTime.value.length === 2) {
  173. beanConsumeLive.value.startTime = formatTime(getTime.value[0])
  174. beanConsumeLive.value.endTime = formatTime(getTime.value[1])
  175. } else {
  176. beanConsumeLive.value.startTime = ''
  177. beanConsumeLive.value.endTime = ''
  178. }
  179. // 设置排序参数
  180. beanConsumeLive.value.sortField = sortField.value
  181. beanConsumeLive.value.sortOrder = sortOrder.value
  182. console.log('搜索参数_时间', beanConsumeLive.value.startTime)
  183. console.log('搜索参数1', getObj.value)
  184. console.log('搜索参数2', beanConsumeLive.value)
  185. // 发送POST请求
  186. const result = await request({
  187. url: '/beanConsume/selectLiveBy',
  188. data: {
  189. pageNum: getObj.value.pageNum,
  190. pageSize: getObj.value.pageSize,
  191. beanConsumeLive: {
  192. ...beanConsumeLive.value,
  193. jwcode: beanConsumeLive.value.jwcode ? String(beanConsumeLive.value.jwcode) : '',
  194. dept: beanConsumeLive.value.dept || '',
  195. type: beanConsumeLive.value.type || '',
  196. gift: beanConsumeLive.value.gift || '',
  197. beanNum: beanConsumeLive.value.beanNum || '',
  198. isBackpack: beanConsumeLive.value.isBackpack || '',
  199. liveChannel: beanConsumeLive.value.liveChannel || '',
  200. liveName: beanConsumeLive.value.liveName || '',
  201. startTime: beanConsumeLive.value.startTime || '',
  202. endTime: beanConsumeLive.value.endTime || '',
  203. sortField: beanConsumeLive.value.sortField || 'consumeTime',
  204. sortOrder: beanConsumeLive.value.sortOrder || 'desc'
  205. }
  206. }
  207. })
  208. console.log('请求成功2', sortField)
  209. console.log('接口响应结果', result); // 打印接口响应结果
  210. if (result.code === 200 && result.data && result.data.list) {
  211. tableData.value = result.data.list;
  212. total.value = result.data.total;
  213. }
  214. // 复制一份 beanConsumeLive.value 并设置固定的 payType 值 1是直播
  215. const sumConsumeParams = {
  216. payType: 1, // 固定传入 payType 值 1 是直播
  217. beanConsumeLive: {
  218. ...beanConsumeLive.value,
  219. }
  220. };
  221. // 发送 POST 请求获取合计数
  222. const resultTotalGold = await request({
  223. url: '/beanConsume/sumConsumeGold',
  224. data: sumConsumeParams
  225. });
  226. console.log("总计", resultTotalGold);
  227. const data = resultTotalGold.data || resultTotalGold;
  228. console.log('请求成功1', resultTotalGold.data) //undifined
  229. console.log('permanentBean1',data.permanentBean)
  230. // 返回字段为 permanentBean、freeBean、totalNum
  231. permanentBean.value = Number(data.permanentBean) || 0;
  232. freeBean.value = Number(data.freeBean) || 0;
  233. totalNum.value = Number(data.totalNum) || 0;
  234. // 存储分页总数
  235. total.value = result.data.total
  236. console.log('total', total.value)
  237. } catch (error) {
  238. console.log('请求失败', error)
  239. }
  240. }
  241. // 搜索
  242. const search = function () {
  243. getObj.value.pageNum = 1
  244. if(beanConsumeLive.value.jwcode){
  245. const numRef = /^\d{1,9}$/;
  246. if(!numRef.test(beanConsumeLive.value.jwcode)){
  247. ElMessage.error('请检查精网号格式')
  248. return
  249. }
  250. }
  251. selectLiveBy()
  252. }
  253. // 重置
  254. const reset = function () {
  255. console.log('直播的重置')
  256. beanConsumeLive.value.jwcode = null
  257. beanConsumeLive.value.type = ''
  258. beanConsumeLive.value.gift = ''
  259. beanConsumeLive.value.liveChannel = ''
  260. beanConsumeLive.value.liveName = ''
  261. beanConsumeLive.value.dept = ''
  262. beanConsumeLive.value.startTime = ''
  263. beanConsumeLive.value.endTime = ''
  264. sortField.value = ''
  265. sortOrder.value = ''
  266. getTime.value = {}
  267. activeTimeRange.value = '' // 清除激活状态
  268. // 点完重置后,重新请求数据
  269. selectLiveBy()
  270. console.log(' beanConsumeLive', beanConsumeLive.value)
  271. }
  272. // 今天
  273. const getToday = function () {
  274. const today = dayjs()
  275. const startTime = today.startOf('day').format('YYYY-MM-DD HH:mm:ss')
  276. const endTime =today.endOf('day').format('YYYY-MM-DD HH:mm:ss')
  277. getTime.value = [startTime, endTime]
  278. console.log('getTime', getTime.value)
  279. activeTimeRange.value = 'today' // 标记当前激活状态
  280. selectLiveBy()
  281. }
  282. // 昨天
  283. const getYesterday = function () {
  284. const today = dayjs()
  285. const startTime = today.subtract(1, 'day').startOf('day').format('YYYY-MM-DD HH:mm:ss')
  286. const endTime = today.subtract(1, 'day').endOf('day').format('YYYY-MM-DD HH:mm:ss')
  287. getTime.value = [startTime, endTime]
  288. console.log('getTime', getTime.value)
  289. activeTimeRange.value = 'yesterday' // 标记当前激活状态
  290. selectLiveBy()
  291. }
  292. // 近7天
  293. const get7Days = function () {
  294. const today = dayjs()
  295. const startTime = today.subtract(6, 'day').startOf('day').format('YYYY-MM-DD HH:mm:ss')
  296. const endTime = today.endOf('day').format('YYYY-MM-DD HH:mm:ss')
  297. getTime.value = [startTime, endTime]
  298. console.log('getTime', getTime.value)
  299. activeTimeRange.value = '7days' // 标记当前激活状态
  300. selectLiveBy()
  301. }
  302. // 处理排序事件
  303. const handleSortChange = (column) => {
  304. console.log('排序字段:', column.prop)
  305. console.log('排序方式:', column.order)
  306. if (column.prop === 'beanNum') {
  307. sortField.value = 'beanNum'
  308. } else if (column.prop === 'consumeTime') {
  309. sortField.value = 'consumeTime'
  310. } else if(column.prop === 'buyBean') {
  311. sortField.value = 'buyBean'
  312. } else if(column.prop === 'freeBean') {
  313. sortField.value = 'freeBean'
  314. }
  315. sortOrder.value = column.order === 'ascending' ? 'DESC' : 'ASC'
  316. selectLiveBy()
  317. }
  318. const handlePageSizeChange = function (val) {
  319. getObj.value.pageSize = val
  320. selectLiveBy()
  321. }
  322. const handleCurrentChange = function (val) {
  323. getObj.value.pageNum = val
  324. selectLiveBy()
  325. }
  326. /*
  327. ====================计算属性=================================
  328. */
  329. // 计算总金币数
  330. // const totalBean = computed(() => permanentBean.value + freeBean.value)
  331. /*
  332. ====================监听=================================
  333. */
  334. /*
  335. ====================挂载=================================
  336. */
  337. onMounted(async function () {
  338. await getAdminData()
  339. await selectLiveBy()
  340. await getDept()
  341. await getGift()
  342. await getChannel()
  343. })
  344. const exportExcel = async function () {
  345. console.log('1')
  346. const params = {
  347. ...getObj.value,
  348. "beanConsumeLive": {
  349. ...beanConsumeLive.value,
  350. jwcode: beanConsumeLive.value.jwcode ? String(beanConsumeLive.value.jwcode) : '',
  351. dept: beanConsumeLive.value.dept || '',
  352. type: beanConsumeLive.value.type || '',
  353. gift: beanConsumeLive.value.gift || '',
  354. liveChannel: beanConsumeLive.value.liveChannel || '',
  355. liveName: beanConsumeLive.value.liveName || '',
  356. startTime: beanConsumeLive.value.startTime || '',
  357. endTime: beanConsumeLive.value.endTime || '',
  358. sortField: sortField.value || 'consumeTime',
  359. sortOrder: sortOrder.value || 'desc'
  360. }
  361. }
  362. // 打印请求参数,方便调试
  363. console.log('导出请求参数:', params);
  364. try {
  365. console.log('2')
  366. const res = await API({ url: '/export/exportLive', data: params });
  367. console.log('导出请求响应:', res);
  368. if (res.code === 200) {
  369. ElMessage.success('导出成功');
  370. } else {
  371. ElMessage.error(res.message || '导出失败,请稍后重试');
  372. }
  373. } catch (error) {
  374. console.error('导出请求出错:', error);
  375. ElMessage.error('导出失败,请稍后重试');
  376. }
  377. }
  378. const exportListVisible = ref(false)
  379. // 打开导出列表弹窗
  380. const openExportList = () => {
  381. getExportList()
  382. exportListVisible.value = true
  383. }
  384. // 导出列表数据
  385. const exportList = ref([])
  386. // 导出列表加载状态
  387. const exportListLoading = ref(false)
  388. // 获取导出列表
  389. const getExportList = async () => {
  390. exportListLoading.value = true
  391. try {
  392. const result = await API({ url: '/export/export' })
  393. if (result.code === 200) {
  394. const filteredData = result.data.filter(item => {
  395. return item.type === 6; //4表示金币消耗列表 // todo 修改对应type 6是直播
  396. });
  397. exportList.value = filteredData
  398. } else {
  399. ElMessage.error(result.msg || '获取导出列表失败')
  400. }
  401. } catch (error) {
  402. console.error('获取导出列表出错:', error)
  403. ElMessage.error('获取导出列表失败,请稍后重试')
  404. } finally {
  405. exportListLoading.value = false
  406. }
  407. }
  408. // 下载导出文件
  409. const downloadExportFile = (item) => {
  410. if (item.state === 2) {
  411. const link = document.createElement('a')
  412. link.href = item.url
  413. link.download = item.fileName
  414. link.click()
  415. } else {
  416. ElMessage.warning('文件还在导出中,请稍后再试')
  417. }
  418. }
  419. //根据状态返回对应的标签类型
  420. const getTagType = (state) => {
  421. switch (state) {
  422. case 0:
  423. return 'info';
  424. case 1:
  425. return 'primary';
  426. case 2:
  427. return 'success';
  428. case 3:
  429. return 'danger';
  430. default:
  431. return 'info';
  432. }
  433. }
  434. //根据状态返回对应的标签文案
  435. const getTagText = (state) => {
  436. switch (state) {
  437. case 0:
  438. return '待执行';
  439. case 1:
  440. return '执行中';
  441. case 2:
  442. return '执行完成';
  443. case 3:
  444. return '执行出错';
  445. default:
  446. return '未知状态';
  447. }
  448. }
  449. </script>
  450. <template>
  451. <el-row>
  452. <el-col>
  453. <el-card style="margin-bottom: 20px;margin-top:10px">
  454. <el-row style="margin-bottom: 10px">
  455. <el-col :span="4">
  456. <div class="head-card-element">
  457. <el-text class="mx-1">精网号</el-text>
  458. <el-input v-model="beanConsumeLive.jwcode" placeholder="请输入精网号" style="width: 140px" clearable />
  459. </div>
  460. </el-col>
  461. <el-col :span="4">
  462. <div class="head-card-element">
  463. <el-text class="mx-1">地区</el-text>
  464. <el-select v-model="beanConsumeLive.dept" placeholder="请选择地区" style="width: 140px" clearable>
  465. <el-option v-for="(item, index) in dept" :key="index" :label="item" :value="item" />
  466. </el-select>
  467. </div>
  468. </el-col>
  469. <!-- <el-col :span="4">
  470. <div class="head-card-element">
  471. <el-text class="mx-1">类型</el-text>
  472. <el-select v-model="beanConsumeLive.type" placeholder="请选择类型" style="width: 140px" clearable @change="handleTypeChange">
  473. <el-option v-for="(item, index) in consumeTypes" :key="index" :label="item.label" :value="item.value" />
  474. </el-select>
  475. </div>
  476. </el-col> -->
  477. <el-col :span="4" >
  478. <div class="head-card-element">
  479. <el-text class="mx-1">礼物名称</el-text>
  480. <el-select v-model="beanConsumeLive.gift" placeholder="请选择礼物名称" style="width: 140px" clearable
  481. filterable
  482. allow-create
  483. default-first-option
  484. >
  485. <el-option v-for="(item, index) in gifts" :key="index" :label="item" :value="item" />
  486. </el-select>
  487. </div>
  488. </el-col>
  489. <el-col :span="4">
  490. <div class="head-card-element">
  491. <el-text class="mx-1">频道</el-text>
  492. <el-select
  493. v-model="beanConsumeLive.liveChannel"
  494. placeholder="请选择频道"
  495. style="width: 150px"
  496. clearable
  497. filterable
  498. allow-create
  499. default-first-option
  500. >
  501. <el-option v-for="(item, index) in channels" :key="index" :label="item" :value="item" />
  502. </el-select>
  503. </div>
  504. </el-col>
  505. <el-col :span="4">
  506. <div class="head-card-element">
  507. <el-text class="mx-1">直播间</el-text>
  508. <el-input v-model="beanConsumeLive.liveName" placeholder="请输入直播间" style="width: 150px" clearable />
  509. </div>
  510. </el-col>
  511. </el-row>
  512. <el-row>
  513. <el-col :span="24">
  514. <div class="head-card-element">
  515. <el-text class="mx-1" >消费时间</el-text>
  516. <el-date-picker v-model="getTime" type="datetimerange" range-separator="" start-placeholder="起始时间"
  517. end-placeholder="结束时间" style="width: 400px" @change="handleDatePickerChange" :default-time="defaultTime"/>
  518. <el-button @click="getToday()" style="margin-left: 10px"
  519. :type="activeTimeRange === 'today' ? 'primary' : ''"> </el-button>
  520. <el-button @click="getYesterday()" style="margin-left: 10px"
  521. :type="activeTimeRange === 'yesterday' ? 'primary' : ''"> </el-button>
  522. <el-button @click="get7Days()" style="margin-left: 10px"
  523. :type="activeTimeRange === '7days' ? 'primary' : ''"> 近7天</el-button>
  524. <el-button type="success" @click="reset()">重置</el-button>
  525. <el-button type="primary" @click="search()">查询</el-button>
  526. <el-button type="primary" @click="exportExcel()">导出excel</el-button>
  527. <el-button type="primary" @click="openExportList">查看导出列表</el-button>
  528. </div>
  529. </el-col>
  530. </el-row>
  531. </el-card>
  532. </el-col>
  533. </el-row>
  534. <el-row>
  535. <el-col>
  536. <el-card>
  537. <div>
  538. 金豆总数{{ format3(Math.abs(permanentBean+freeBean)) }}&nbsp;&nbsp;&nbsp;&nbsp;
  539. 付费金豆数{{ format3(Math.abs(permanentBean)) }}&nbsp;&nbsp;&nbsp;&nbsp;
  540. 免费金豆数{{ format3(Math.abs(freeBean)) }}
  541. </div>
  542. <!-- 设置表格容器的高度和滚动样式 -->
  543. <div style="height: 576px; overflow-y: auto">
  544. <el-table :data="tableData" style="width: 100%" height="576px" @sort-change="handleSortChange">
  545. <el-table-column type="index" label="序号" width="80px" fixed="left">
  546. <template #default="scope">
  547. <span>{{
  548. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  549. }}</span>
  550. </template>
  551. </el-table-column>
  552. <!-- 固定姓名列 -->
  553. <el-table-column prop="name" label="姓名" width="150px" fixed="left" />
  554. <!-- 固定精网号列 -->
  555. <el-table-column prop="jwcode" label="精网号" width="110px" fixed="left" />
  556. <el-table-column prop="dept" label="地区" width="110px" />
  557. <el-table-column prop="gift" label="礼物" width="140px" >
  558. </el-table-column>
  559. <el-table-column prop="beanNum" label="金豆数量" sortable="custom" width="120px"/>
  560. <el-table-column prop="isBackpack" label="背包礼物" width="120px">
  561. <template #default="scope">
  562. {{ scope.row.isBackpack == 1 ? '是' : '否' }}
  563. </template>
  564. </el-table-column>
  565. <el-table-column prop="buyBean" label="付费金豆数" sortable="custom" width="120px"/>
  566. <el-table-column prop="freeBean" label="免费金豆数" sortable="custom" width="120px"/>
  567. <el-table-column prop="liveChannel" label="频道" width="180px" />
  568. <el-table-column prop="liveName" label="直播间名称" width="160px" show-overflow-tooltip />
  569. <el-table-column prop="consumeTime" label="消费时间" sortable="custom" width="180px" />
  570. </el-table>
  571. </div>
  572. <!-- 分页 -->
  573. <div class="pagination">
  574. <el-pagination background :page-size="getObj.pageSize" :page-sizes="[5, 10, 20, 50, 100]"
  575. layout="total, sizes, prev, pager, next, jumper" :total="total" @size-change="handlePageSizeChange"
  576. @current-change="handleCurrentChange"></el-pagination>
  577. </div>
  578. </el-card>
  579. </el-col>
  580. </el-row>
  581. <!-- 导出弹窗 -->
  582. <el-dialog v-model="exportListVisible" title="导出列表" width="80%">
  583. <el-table :data="exportList" style="width: 100% ;height: 60vh;" :loading="exportListLoading">
  584. <el-table-column prop="fileName" label="文件名" />
  585. <el-table-column prop="state" label="状态">
  586. <template #default="scope">
  587. <el-tag :type="getTagType(scope.row.state)" :effect="scope.row.state === 3 ? 'light' : 'plain'">
  588. {{ getTagText(scope.row.state) }}
  589. </el-tag>
  590. </template>
  591. </el-table-column>
  592. <el-table-column prop="createTime" label="创建时间">
  593. <template #default="scope">
  594. {{ moment(scope.row.createTime).format('YYYY-MM-DD HH:mm:ss') }}
  595. </template>
  596. </el-table-column>
  597. <el-table-column label="操作">
  598. <template #default="scope">
  599. <el-button type="primary" size="small" @click="downloadExportFile(scope.row)"
  600. :disabled="scope.row.state !== 2">
  601. 下载
  602. </el-button>
  603. </template>
  604. </el-table-column>
  605. </el-table>
  606. <template #footer>
  607. <div class="dialog-footer">
  608. <el-button text @click="exportListVisible = false">关闭</el-button>
  609. </div>
  610. </template>
  611. </el-dialog>
  612. </template>
  613. <style scoped>
  614. .status {
  615. display: flex;
  616. }
  617. .head-card {
  618. display: flex;
  619. }
  620. .head-card-element {
  621. margin-right: 20px;
  622. }
  623. .head-card-btn {
  624. margin-left: auto;
  625. }
  626. .pagination {
  627. display: flex;
  628. margin-top: 20px;
  629. }
  630. </style>