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.

581 lines
17 KiB

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