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.

668 lines
20 KiB

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