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.

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