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.

675 lines
19 KiB

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