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.

618 lines
18 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. // 之后整理一下
  7. /*
  8. ====================工具方法==============================
  9. */
  10. // 时间格式化
  11. const formatTime = (val) => val ? dayjs(val).format('YYYY-MM-DD HH:mm:ss') : ''
  12. /*
  13. ====================数据=================================
  14. */
  15. //这是获取用户信息的接口
  16. const adminData = ref({})
  17. // 充值明细表格
  18. const tableData = ref([])
  19. // 搜索consumeUser 表单
  20. const consumeUser = ref({
  21. jwcode: null,
  22. payPlatform: "",
  23. market: "",
  24. startTime: '',
  25. endTime: '',
  26. goodsName: '',
  27. })
  28. // 标记当前激活的时间范围按钮
  29. const activeTimeRange = ref('')
  30. // 日期选择器变化时清除按钮激活状态
  31. const handleDatePickerChange = () => {
  32. activeTimeRange.value = ''
  33. }
  34. // 搜索对象
  35. const getObj = ref({
  36. pageNum: 1,
  37. pageSize: 50
  38. })
  39. //分页总条目
  40. const total = ref(100)
  41. // 搜索对象时间
  42. const getTime = ref({
  43. startTime: '',
  44. endTime: ''
  45. })
  46. // 搜索活动列表
  47. const activity = ref([])
  48. // 搜索地区列表
  49. const market = ref([])
  50. // 新增排序字段和排序方式
  51. const sortField = ref('')
  52. const sortOrder = ref('')
  53. // 消费平台选项
  54. const consumePlatform = [
  55. {
  56. value: '金币系统',
  57. label: '金币系统'
  58. },
  59. {
  60. value: 'HomilyChart',
  61. label: 'HomilyChart'
  62. },
  63. {
  64. value: 'HomilyLink',
  65. label: 'HomilyLink'
  66. },
  67. {
  68. value: 'ERP',
  69. label: 'ERP'
  70. },
  71. {
  72. value: '其他',
  73. label: '其他'
  74. },
  75. ]
  76. // 合计数的显示数据
  77. const tableDataTotal = ref({})
  78. const permanentGold = ref(0)
  79. const freeGold = ref(0)
  80. const taskGold = ref(0)
  81. const totalGold = ref(0)
  82. // 定义响应式变量存储金币合计数
  83. const permanentGolds = ref(0)
  84. const freeGolds = ref(0)
  85. const taskGolds = ref(0)
  86. // 查询商品的接口的表单
  87. const goods = ref([])
  88. const totalGoldSearch = ref({
  89. jwcode: null,
  90. payPlatform: "",
  91. market: "",
  92. startTime: "",
  93. endTime: "",
  94. goodsName: ""
  95. })
  96. /*
  97. ====================方法=================================
  98. */
  99. // 获取登录用户信息
  100. const getAdminData = async function () {
  101. try {
  102. const result = await request({
  103. url: '/admin/userinfo',
  104. data: {}
  105. })
  106. adminData.value = result
  107. console.log('请求成功', result)
  108. console.log('用户信息', adminData.value)
  109. } catch (error) {
  110. console.log('请求失败', error)
  111. }
  112. }
  113. const ConsumeSelectBy = async function (val) {
  114. try {
  115. // 搜索参数页码赋值
  116. if (typeof val === 'number') {
  117. getObj.value.pageNum = val
  118. }
  119. // 搜索参数时间赋值
  120. if (getTime.value != null) {
  121. if (getTime.value.startTime != '' && getTime.value.endTime != '') {
  122. consumeUser.value.startTime = formatTime(getTime.value[0])
  123. consumeUser.value.endTime = formatTime(getTime.value[1])
  124. }
  125. } else {
  126. consumeUser.value.startTime = ''
  127. consumeUser.value.endTime = ''
  128. }
  129. consumeUser.value.sortField = sortField.value
  130. consumeUser.value.sortOrder = sortOrder.value
  131. console.log('搜索参数_时间', consumeUser.value.startTime)
  132. console.log('搜索参数1', getObj.value)
  133. console.log('搜索参数2', consumeUser.value)
  134. // 发送POST请求
  135. const result = await request({
  136. url: '/consume/selectBy',
  137. data: {
  138. pageNum: getObj.value.pageNum,
  139. pageSize: getObj.value.pageSize,
  140. consumeUser: {
  141. ...consumeUser.value,
  142. sortField: sortField.value,
  143. sortOrder: sortOrder.value
  144. }
  145. }
  146. })
  147. console.log('请求成功2', sortField)
  148. // 合计数的接口
  149. // 复制一份 detail.value 并移除排序字段和排序方式
  150. const detailWithoutSort = {...consumeUser.value}
  151. delete detailWithoutSort.sortField
  152. delete detailWithoutSort.sortOrder
  153. // 赋值
  154. totalGoldSearch.value.startTime = consumeUser.value.startTime
  155. totalGoldSearch.value.endTime = consumeUser.value.endTime
  156. totalGoldSearch.value.payPlatform = consumeUser.value.payPlatform
  157. totalGoldSearch.value.market = consumeUser.value.market
  158. totalGoldSearch.value.goodsName = consumeUser.value.goodsName
  159. totalGoldSearch.value.jwcode = consumeUser.value.jwcode
  160. //
  161. const resultTotalGold = await request({
  162. // url: '/consume/statsGold',
  163. url: '/consume/statsGold',
  164. data: totalGoldSearch.value
  165. })
  166. console.log("总计", resultTotalGold)
  167. if (resultTotalGold.code === 200 && resultTotalGold.data) {
  168. const data = resultTotalGold.data
  169. console.log('获取到的金币数据:', data)
  170. permanentGolds.value = Number(data.permanentGolds) || 0
  171. freeGolds.value = Number(data.freeGolds) || 0
  172. taskGolds.value = Number(data.taskGolds) || 0
  173. }
  174. // 存储表格数据
  175. tableData.value = result.data.list
  176. tableDataTotal.value = resultTotalGold.data
  177. if (resultTotalGold.data == null) {
  178. console.log('请求成功2', resultTotalGold)
  179. // 这是啥东西啊
  180. tableDataTotal.value = resultTotalGold.data
  181. }
  182. const sumGoldList = tableData.value.map(item => item.sumGold);
  183. console.log("sumGold", sumGoldList); // 输出包含所有 sumGold 值的数组
  184. console.log('@@@@@@@@@@tableDataT', tableDataTotal.value.list)
  185. // totalGold.value = tableData.value.sum
  186. // // 修改为保留两位小数
  187. // permanentGold.value = parseFloat(
  188. // (tableDataTotal.value.list.sumGold / 100).toFixed(2)
  189. // )
  190. // freeGold.value = parseFloat(
  191. // (tableDataTotal.value.sumFcion / 100).toFixed(2)
  192. // )
  193. // taskGold.value = parseFloat(
  194. // (tableDataTotal.value.sumTcion / 100).toFixed(2)
  195. // )
  196. // totalGold.value = parseFloat(
  197. // (tableDataTotal.value.sumcion / 100).toFixed(2)
  198. // )
  199. console.log('tableData', tableData.value)
  200. // 存储分页总数
  201. total.value = result.data.total
  202. console.log('total', total.value)
  203. } catch (error) {
  204. console.log('请求失败', error)
  205. // 在这里可以处理错误逻辑,比如显示错误提示等
  206. }
  207. }
  208. // 搜索
  209. const search = function () {
  210. getObj.value.pageNum = 1
  211. ConsumeSelectBy()
  212. }
  213. // 重置
  214. const reset = function () {
  215. console.log('兄弟,你点了重置')
  216. consumeUser.value.goodsName = ''
  217. consumeUser.value.market = ''
  218. consumeUser.value.payPlatform = ''
  219. consumeUser.value.startTime = ''
  220. consumeUser.value.endTime = ''
  221. consumeUser.value.jwcode = null
  222. sortField.value = ''
  223. sortOrder.value = ''
  224. getTime.value = {}
  225. activeTimeRange.value = '' // 清除激活状态
  226. // 点完重置后,重新请求数据
  227. ConsumeSelectBy()
  228. console.log('consumeUser', consumeUser.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 getGoods = async function () {
  287. try {
  288. // 发送POST请求
  289. const result = await request({
  290. url: '/general/goods',
  291. data: {}
  292. })
  293. // 将响应结果存储到响应式数据中
  294. console.log('请求成功product', result)
  295. // 存储全部数据
  296. goods.value = result.data
  297. console.log('goods 数据', goods.value) // 修改日志输出
  298. } catch (error) {
  299. console.log('请求失败', error)
  300. // 在这里可以处理错误逻辑,比如显示错误提示等
  301. }
  302. }
  303. // 获取地区列表的方法
  304. const getMarket = async function () {
  305. try {
  306. // 发送请求获取地区列表
  307. const result = await request({
  308. // url: '/general/market',
  309. url: '/general/market', // todo 换成实际接口地址
  310. data: {}
  311. })
  312. console.log('请求地区列表成功', result)
  313. // 存储地区数据
  314. market.value = result.data
  315. console.log('地区数据', market.value)
  316. } catch (error) {
  317. console.log('请求地区列表失败', error)
  318. ElMessage({
  319. type: 'error',
  320. message: '获取地区列表失败,请稍后重试'
  321. })
  322. }
  323. }
  324. // 处理排序事件
  325. const handleSortChange = (column) => {
  326. console.log('排序字段:', column.prop)
  327. console.log('排序方式:', column.order)
  328. if (column.prop === 'permanentGold') {
  329. sortField.value = 'permanentGold'
  330. } else if (column.prop === 'taskGold') {
  331. sortField.value = 'taskGold'
  332. } else if (column.prop === 'freeGold') {
  333. sortField.value = 'freeGold'
  334. } else if (column.prop === 'createTime') {
  335. sortField.value = 'createTime'
  336. }
  337. sortOrder.value = column.order === 'ascending' ? 'DESC' : 'ASC'
  338. ConsumeSelectBy()
  339. }
  340. const exportExcel = async function () {
  341. const params = {
  342. consumUser: {
  343. jwcode: consumeUser.value.jwcode || '',
  344. payPlatform: consumeUser.value.payPlatform || '',
  345. market: consumeUser.value.market || '',
  346. startTime: consumeUser.value.startTime || '',
  347. endTime: consumeUser.value.endTime || '',
  348. goodsName: consumeUser.value.goodsName || ''
  349. },
  350. page: getObj.pageNum,
  351. size: getObj.pageSize
  352. }
  353. const res = await API({url: '/export/exportConsume', data: params})
  354. if (res.code === 200) {
  355. ElMessage.success('导出成功')
  356. }
  357. }
  358. const handlePageSizeChange = function (val) {
  359. getObj.value.pageSize = val
  360. ConsumeSelectBy()
  361. }
  362. const handleCurrentChange = function (val) {
  363. getObj.value.pageNum = val
  364. ConsumeSelectBy()
  365. }
  366. /*
  367. ====================计算属性=================================
  368. */
  369. // 计算总金币数
  370. const sumGold = computed(() => permanentGolds.value + freeGolds.value + taskGolds.value)
  371. /*
  372. ====================监听=================================
  373. */
  374. /*
  375. ====================挂载=================================
  376. */
  377. onMounted(async function () {
  378. await getAdminData()
  379. await ConsumeSelectBy()
  380. // getActivity();
  381. await getMarket()
  382. await getGoods()
  383. })
  384. </script>
  385. <template>
  386. <el-row>
  387. <el-col>
  388. <el-card style="margin-bottom: 20px;margin-top:10px">
  389. <el-row style="margin-bottom: 10px">
  390. <el-col :span="6">
  391. <div class="head-card-element">
  392. <el-text size="large">精网号</el-text>
  393. <el-input v-model="consumeUser.jwcode" placeholder="请输入精网号" size="large" style="width: 240px"
  394. clearable>
  395. </el-input>
  396. </div>
  397. </el-col>
  398. <el-col :span="6">
  399. <div class="head-card-element">
  400. <el-text class="mx-1" size="large">商品名称</el-text>
  401. <el-select v-model="consumeUser.goodsName" placeholder="请选择商品名称" size="large" style="width: 180px"
  402. clearable>
  403. <!-- 修改 v-for 绑定逻辑 -->
  404. <el-option v-for="(item, index) in goods" :key="index" :label="item" :value="item"/>
  405. </el-select>
  406. </div>
  407. </el-col>
  408. <el-col :span="6">
  409. <div class="head-card-element">
  410. <el-text class="mx-1" size="large">所属地区</el-text>
  411. <el-select v-model="consumeUser.market" placeholder="请选择所属地区" size="large" style="width: 180px"
  412. clearable>
  413. <el-option v-for="(item, index) in market" :key="index" :label="item" :value="item"/>
  414. </el-select>
  415. </div>
  416. </el-col>
  417. <el-col :span="6">
  418. <div class="head-card-element">
  419. <el-text class="mx-1" size="large">消耗平台</el-text>
  420. <el-select v-model="consumeUser.payPlatform" placeholder="请选择消耗平台" size="large"
  421. style="width: 180px"
  422. clearable>
  423. <el-option v-for="item in consumePlatform" :key="item.value" :label="item.label" :value="item.value"/>
  424. </el-select>
  425. </div>
  426. </el-col>
  427. </el-row>
  428. <el-row>
  429. <el-col :span="21">
  430. <div class="head-card-element">
  431. <el-text class="mx-1" size="large">消耗时间</el-text>
  432. <el-date-picker v-model="getTime" type="datetimerange" range-separator="" start-placeholder="起始时间"
  433. end-placeholder="结束时间" style="width: 400px" @change="handleDatePickerChange"/>
  434. <el-button @click="getToday()" style="margin-left: 10px" :type="activeTimeRange === 'today' ? 'primary' : ''"> </el-button>
  435. <el-button @click="getYesterday()" style="margin-left: 10px" :type="activeTimeRange === 'yesterday' ? 'primary' : ''"> </el-button>
  436. <el-button @click="get7Days()" style="margin-left: 10px" :type="activeTimeRange === '7days' ? 'primary' : ''"> 近7天</el-button>
  437. <!-- </div>
  438. </el-col>
  439. <el-col :span="3">
  440. <div class="head-card-btn"> -->
  441. <el-button type="success" @click="reset()">重置</el-button>
  442. <el-button type="primary" @click="search()">查询</el-button>
  443. <el-button type="primary" @click="exportExcel()">导出excel</el-button>
  444. </div>
  445. </el-col>
  446. </el-row>
  447. </el-card>
  448. </el-col>
  449. </el-row>
  450. <el-row>
  451. <el-col>
  452. <el-card>
  453. <div>
  454. 消耗金额{{ Math.abs(sumGold) / 100 }}新币永久金币{{ Math.abs(permanentGolds) / 100 }}免费金币{{
  455. Math.abs(freeGolds) / 100
  456. }}任务金币{{ Math.abs(taskGolds) / 100 }}
  457. </div>
  458. <!-- 设置表格容器的高度和滚动样式 -->
  459. <div style="height: 576px; overflow-y: auto">
  460. <el-table :data="tableData" style="width: 100%" height="576px" @sort-change="handleSortChange">
  461. <el-table-column type="index" label="序号" width="80px" fixed="left">
  462. <template #default="scope">
  463. <span>{{
  464. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  465. }}</span>
  466. </template>
  467. </el-table-column>
  468. <!-- 固定姓名列 -->
  469. <el-table-column prop="name" label="姓名" width="150px" fixed="left"/>
  470. <!-- 固定精网号列 -->
  471. <el-table-column prop="jwcode" label="精网号" width="110px" fixed="left"/>
  472. <el-table-column prop="market" label="所属地区" width="110px"/>
  473. <el-table-column prop="goodsName" label="商品" width="160px" show-overflow-tooltip/>
  474. <el-table-column prop="payPlatform" label="消耗平台" width="120px">
  475. <template #default="scope">
  476. <!-- 使用非严格相等比较 -->
  477. {{ scope.row.payPlatform }}
  478. <!-- <span scope.row.consumePlatform == 1">Homily Chart</span>-->
  479. </template>
  480. </el-table-column>
  481. <!-- <el-table-column
  482. prop="consumeType"
  483. label="消费类型"
  484. width="120px"
  485. /> -->
  486. <el-table-column prop="sumGold" label="消耗金币总数" width="120px">
  487. <template #default="scope">
  488. {{
  489. (scope.row.taskGold +
  490. scope.row.freeGold +
  491. scope.row.permanentGold) / 100
  492. }}
  493. </template>
  494. </el-table-column>
  495. <el-table-column prop="permanentGold" label="永久金币" sortable="“custom”" width="110px">
  496. <template #default="scope">
  497. {{ scope.row.permanentGold / 100 }}
  498. </template>
  499. </el-table-column>
  500. <el-table-column prop="freeGold" label="免费金币" sortable="“custom”" width="110px">
  501. <template #default="scope">
  502. {{ scope.row.freeGold / 100 }}
  503. </template>
  504. </el-table-column>
  505. <el-table-column prop="taskGold" label="任务金币" sortable="“custom”" width="110px">
  506. <template #default="scope">
  507. {{ scope.row.taskGold / 100 }}
  508. </template>
  509. </el-table-column>
  510. <el-table-column prop="remark" label="备注" width="200px" show-overflow-tooltip/>
  511. <el-table-column prop="adminName" label="提交人" width="110px"/>
  512. <el-table-column prop="createTime" label="消耗时间" sortable="custom" width="180px"/>
  513. </el-table>
  514. </div>
  515. <!-- 分页 -->
  516. <div class="pagination">
  517. <el-pagination background :page-size="getObj.pageSize" :page-sizes="[5, 10, 20, 50, 100]"
  518. layout="total, sizes, prev, pager, next, jumper" :total="total"
  519. @size-change="handlePageSizeChange"
  520. @current-change="handleCurrentChange"></el-pagination>
  521. </div>
  522. </el-card>
  523. </el-col>
  524. </el-row>
  525. </template>
  526. <style scoped>
  527. .status {
  528. display: flex;
  529. }
  530. .head-card {
  531. display: flex;
  532. }
  533. .head-card-element {
  534. margin-right: 20px;
  535. }
  536. .head-card-btn {
  537. margin-left: auto;
  538. }
  539. .pagination {
  540. display: flex;
  541. margin-top: 20px;
  542. }
  543. </style>