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.

627 lines
18 KiB

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