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.

651 lines
19 KiB

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