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.

767 lines
23 KiB

1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month 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. import {reverseMarketMapping} from "@/utils/marketMap.js";
  8. // 之后整理一下
  9. /*
  10. ====================工具方法==============================
  11. */
  12. const trimJwCode = () => {
  13. if (consumeUser.value.jwcode) {
  14. consumeUser.value.jwcode = consumeUser.value.jwcode.replace(/\s/g, '');
  15. }
  16. }
  17. const format3 = (num) => {
  18. if (!num) return '0';
  19. const parts = Number(num).toFixed(2).split('.');
  20. // 添加逗号
  21. parts[0] = parts[0].replace(/\B(?=(\d{3})+(?!\d))/g, ',');
  22. return parts.join('.');
  23. }
  24. // 时间格式化
  25. const formatTime = (val) => val ? dayjs(val).format('YYYY-MM-DD HH:mm:ss') : ''
  26. const defaultTime = [
  27. new Date(2000, 1, 1, 0, 0, 0),
  28. new Date(2000, 2, 1, 23, 59, 59),
  29. ]
  30. /*
  31. ====================数据=================================
  32. */
  33. //这是获取用户信息的接口
  34. const adminData = ref({})
  35. // 充值明细表格
  36. const tableData = ref([])
  37. // 搜索consumeUser 表单
  38. const consumeUser = ref({
  39. jwcode: null,
  40. payPlatform: "",
  41. market: "",
  42. startTime: '',
  43. endTime: '',
  44. goodsName: '',
  45. })
  46. // 标记当前激活的时间范围按钮
  47. const activeTimeRange = ref('')
  48. // 日期选择器变化时清除按钮激活状态
  49. const handleDatePickerChange = () => {
  50. activeTimeRange.value = ''
  51. }
  52. // 搜索对象
  53. const getObj = ref({
  54. pageNum: 1,
  55. pageSize: 50
  56. })
  57. //分页总条目
  58. const total = ref(100)
  59. // 搜索对象时间
  60. const getTime = ref({
  61. startTime: '',
  62. endTime: ''
  63. })
  64. // 搜索活动列表
  65. const activity = ref([])
  66. // 搜索地区列表
  67. const market = ref("")
  68. // 新增排序字段和排序方式
  69. const sortField = ref('')
  70. const sortOrder = ref('')
  71. // 消费平台选项
  72. const consumePlatform = [
  73. {
  74. value: '金币系统',
  75. label: '金币系统'
  76. },
  77. {
  78. value: 'HomilyChart',
  79. label: 'HomilyChart'
  80. },
  81. {
  82. value: 'HomilyLink',
  83. label: 'HomilyLink'
  84. },
  85. {
  86. value: 'ERP',
  87. label: 'ERP'
  88. },
  89. {
  90. value: '其他',
  91. label: '其他'
  92. },
  93. {
  94. value: '初始化金币',
  95. label: '初始化金币'
  96. },
  97. ]
  98. // 合计数的显示数据
  99. const tableDataTotal = ref({})
  100. const permanentGold = ref(0)
  101. const freeGold = ref(0)
  102. const taskGold = ref(0)
  103. const totalGold = ref(0)
  104. // 定义响应式变量存储金币合计数
  105. const permanentGolds = ref(0)
  106. const freeGolds = ref(0)
  107. const taskGolds = ref(0)
  108. // 查询商品的接口的表单
  109. const goods = ref([])
  110. const totalGoldSearch = ref({
  111. jwcode: null,
  112. payPlatform: "",
  113. market: "",
  114. startTime: "",
  115. endTime: "",
  116. goodsName: ""
  117. })
  118. /*
  119. ====================方法=================================
  120. */
  121. // 获取登录用户信息
  122. const getAdminData = async function () {
  123. try {
  124. const result = await request({
  125. url: '/admin/userinfo',
  126. data: {}
  127. })
  128. adminData.value = result
  129. console.log('请求成功', result)
  130. console.log('用户信息', adminData.value)
  131. } catch (error) {
  132. console.log('请求失败', error)
  133. }
  134. }
  135. const ConsumeSelectBy = async function (val) {
  136. try {
  137. // 搜索参数页码赋值
  138. if (typeof val === 'number') {
  139. getObj.value.pageNum = val
  140. }
  141. // 搜索参数时间赋值
  142. if (getTime.value != null) {
  143. if (getTime.value.startTime != '' && getTime.value.endTime != '') {
  144. consumeUser.value.startTime = formatTime(getTime.value[0])
  145. consumeUser.value.endTime = formatTime(getTime.value[1])
  146. }
  147. } else {
  148. consumeUser.value.startTime = ''
  149. consumeUser.value.endTime = ''
  150. }
  151. consumeUser.value.sortField = sortField.value
  152. consumeUser.value.sortOrder = sortOrder.value
  153. console.log('搜索参数_时间', consumeUser.value.startTime)
  154. console.log('搜索参数1', getObj.value)
  155. console.log('搜索参数2', consumeUser.value)
  156. // 发送POST请求
  157. if (consumeUser.value.market === '9' || consumeUser.value.market === '9999') {
  158. consumeUser.value.market = '';
  159. }
  160. if (consumeUser.value.jwcode) {
  161. // 纯数字
  162. const numberRegex = /^\d{1,9}$/;
  163. // 检查是否不是数字
  164. if (!numberRegex.test(consumeUser.value.jwcode)) {
  165. ElMessage.error('请检查精网号格式')
  166. return
  167. }
  168. }
  169. const result = await request({
  170. url: '/consume/selectBy',
  171. data: {
  172. pageNum: getObj.value.pageNum,
  173. pageSize: getObj.value.pageSize,
  174. consumeUser: {
  175. ...consumeUser.value,
  176. sortField: sortField.value,
  177. sortOrder: sortOrder.value
  178. }
  179. }
  180. })
  181. console.log('请求成功2', sortField)
  182. // 合计数的接口
  183. // 复制一份 detail.value 并移除排序字段和排序方式
  184. const detailWithoutSort = {...consumeUser.value}
  185. delete detailWithoutSort.sortField
  186. delete detailWithoutSort.sortOrder
  187. // 赋值
  188. totalGoldSearch.value.startTime = consumeUser.value.startTime
  189. totalGoldSearch.value.endTime = consumeUser.value.endTime
  190. totalGoldSearch.value.payPlatform = consumeUser.value.payPlatform
  191. totalGoldSearch.value.market = consumeUser.value.market
  192. totalGoldSearch.value.goodsName = consumeUser.value.goodsName
  193. totalGoldSearch.value.jwcode = consumeUser.value.jwcode
  194. //
  195. const resultTotalGold = await request({
  196. // url: '/consume/statsGold',
  197. url: '/consume/statsGold',
  198. data: totalGoldSearch.value
  199. })
  200. console.log("总计", resultTotalGold)
  201. if (resultTotalGold.code === 200 && resultTotalGold.data) {
  202. const data = resultTotalGold.data
  203. console.log('获取到的金币数据:', data)
  204. permanentGolds.value = Number(data.permanentGolds) || 0
  205. freeGolds.value = Number(data.freeGolds) || 0
  206. taskGolds.value = Number(data.taskGolds) || 0
  207. }
  208. // 存储表格数据
  209. tableData.value = result.data.list
  210. tableDataTotal.value = resultTotalGold.data
  211. if (resultTotalGold.data == null) {
  212. console.log('请求成功2', resultTotalGold)
  213. // 这是啥东西啊
  214. tableDataTotal.value = resultTotalGold.data
  215. }
  216. const sumGoldList = tableData.value.map(item => item.sumGold);
  217. console.log("sumGold", sumGoldList); // 输出包含所有 sumGold 值的数组
  218. console.log('@@@@@@@@@@tableDataT', tableDataTotal.value.list)
  219. // totalGold.value = tableData.value.sum
  220. // // 修改为保留两位小数
  221. // permanentGold.value = parseFloat(
  222. // (tableDataTotal.value.list.sumGold / 100).toFixed(2)
  223. // )
  224. // freeGold.value = parseFloat(
  225. // (tableDataTotal.value.sumFcion / 100).toFixed(2)
  226. // )
  227. // taskGold.value = parseFloat(
  228. // (tableDataTotal.value.sumTcion / 100).toFixed(2)
  229. // )
  230. // totalGold.value = parseFloat(
  231. // (tableDataTotal.value.sumcion / 100).toFixed(2)
  232. // )
  233. console.log('tableData', tableData.value)
  234. // 存储分页总数
  235. total.value = result.data.total
  236. console.log('total', total.value)
  237. } catch (error) {
  238. console.log('请求失败', error)
  239. // 在这里可以处理错误逻辑,比如显示错误提示等
  240. }
  241. }
  242. // 搜索
  243. const search = function () {
  244. trimJwCode()
  245. getObj.value.pageNum = 1
  246. ConsumeSelectBy()
  247. }
  248. // 重置
  249. const reset = function () {
  250. console.log('兄弟,你点了重置')
  251. consumeUser.value.goodsName = ''
  252. consumeUser.value.market = ""
  253. consumeUser.value.payPlatform = ''
  254. consumeUser.value.startTime = ''
  255. consumeUser.value.endTime = ''
  256. consumeUser.value.jwcode = null
  257. sortField.value = ''
  258. sortOrder.value = ''
  259. getTime.value = {}
  260. activeTimeRange.value = '' // 清除激活状态
  261. selectedMarketPath.value = []
  262. // 点完重置后,重新请求数据
  263. ConsumeSelectBy()
  264. console.log('consumeUser', consumeUser.value)
  265. }
  266. // 今天
  267. const getToday = function () {
  268. const today = dayjs()
  269. const startTime = today.startOf('day').format('YYYY-MM-DD HH:mm:ss')
  270. const endTime = today.endOf('day').format('YYYY-MM-DD HH:mm:ss')
  271. getTime.value = [startTime, endTime]
  272. console.log('getTime', getTime.value)
  273. activeTimeRange.value = 'today' // 标记当前激活状态
  274. ConsumeSelectBy()
  275. }
  276. // 昨天
  277. const getYesterday = function () {
  278. const today = dayjs()
  279. const startTime = today.subtract(1, 'day').startOf('day').format('YYYY-MM-DD HH:mm:ss')
  280. const endTime = today.subtract(1, 'day').endOf('day').format('YYYY-MM-DD HH:mm:ss')
  281. getTime.value = [startTime, endTime]
  282. console.log('getTime', getTime.value)
  283. activeTimeRange.value = 'yesterday' // 标记当前激活状态
  284. ConsumeSelectBy()
  285. }
  286. // 近7天
  287. const get7Days = function () {
  288. const today = dayjs()
  289. const startTime = today.subtract(6, 'day').startOf('day').format('YYYY-MM-DD HH:mm:ss')
  290. const endTime = today.endOf('day').format('YYYY-MM-DD HH:mm:ss')
  291. getTime.value = [startTime, endTime]
  292. console.log('getTime', getTime.value)
  293. activeTimeRange.value = '7days' // 标记当前激活状态
  294. ConsumeSelectBy()
  295. }
  296. // 获取商品列表
  297. const getGoods = async function () {
  298. try {
  299. // 发送POST请求
  300. const result = await request({
  301. url: '/general/goods',
  302. data: {}
  303. })
  304. // 将响应结果存储到响应式数据中
  305. console.log('请求成功product', result)
  306. // 存储全部数据
  307. goods.value = result.data
  308. console.log('goods 数据', goods.value) // 修改日志输出
  309. } catch (error) {
  310. console.log('请求失败', error)
  311. // 在这里可以处理错误逻辑,比如显示错误提示等
  312. }
  313. }
  314. // 处理排序事件
  315. const handleSortChange = (column) => {
  316. console.log('排序字段:', column.prop)
  317. console.log('排序方式:', column.order)
  318. if (column.prop === 'permanentGold') {
  319. sortField.value = 'permanentGold'
  320. } else if (column.prop === 'taskGold') {
  321. sortField.value = 'taskGold'
  322. } else if (column.prop === 'freeGold') {
  323. sortField.value = 'freeGold'
  324. } else if (column.prop === 'createTime') {
  325. sortField.value = 'createTime'
  326. }
  327. sortOrder.value = column.order === 'ascending' ? 'DESC' : 'ASC'
  328. ConsumeSelectBy()
  329. }
  330. const handlePageSizeChange = function (val) {
  331. getObj.value.pageSize = val
  332. ConsumeSelectBy()
  333. }
  334. const handleCurrentChange = function (val) {
  335. getObj.value.pageNum = val
  336. ConsumeSelectBy()
  337. }
  338. /*
  339. ====================计算属性=================================
  340. */
  341. // 计算总金币数
  342. const sumGold = computed(() => permanentGolds.value + freeGolds.value + taskGolds.value)
  343. /*
  344. ====================监听=================================
  345. */
  346. /*
  347. ====================挂载=================================
  348. */
  349. onMounted(async function () {
  350. await getAdminData()
  351. await ConsumeSelectBy()
  352. // getActivity();
  353. await getMarket()
  354. await getGoods()
  355. })
  356. const exportExcel = async function () {
  357. const params = {
  358. consumeUser: {
  359. jwcode: consumeUser.value.jwcode || '',
  360. payPlatform: consumeUser.value.payPlatform || '',
  361. marketss: consumeUser.value.market || "",
  362. startTime: consumeUser.value.startTime || '',
  363. endTime: consumeUser.value.endTime || '',
  364. goodsName: consumeUser.value.goodsName || '',
  365. sortField: sortField.value || '',
  366. sortOrder: sortOrder.value || ''
  367. }
  368. }
  369. const res = await API({url: '/export/exportConsume', data: params})
  370. if (res.code === 200) {
  371. ElMessage.success('导出成功')
  372. }
  373. }
  374. const exportListVisible = ref(false)
  375. // 打开导出列表弹窗
  376. const openExportList = () => {
  377. getExportList()
  378. exportListVisible.value = true
  379. }
  380. // 导出列表数据
  381. const exportList = ref([])
  382. // 导出列表加载状态
  383. const exportListLoading = ref(false)
  384. // 获取导出列表
  385. const getExportList = async () => {
  386. exportListLoading.value = true
  387. try {
  388. const result = await API({url: '/export/export'})
  389. if (result.code === 200) {
  390. const filteredData = result.data.filter(item => {
  391. return item.type === 4; //4表示金币消耗列表
  392. });
  393. exportList.value = filteredData
  394. } else {
  395. ElMessage.error(result.msg || '获取导出列表失败')
  396. }
  397. } catch (error) {
  398. console.error('获取导出列表出错:', error)
  399. ElMessage.error('获取导出列表失败,请稍后重试')
  400. } finally {
  401. exportListLoading.value = false
  402. }
  403. }
  404. // 下载导出文件
  405. const downloadExportFile = (item) => {
  406. if (item.state === 2) {
  407. const link = document.createElement('a')
  408. link.href = item.url
  409. link.download = item.fileName
  410. link.click()
  411. } else {
  412. ElMessage.warning('文件还在导出中,请稍后再试')
  413. }
  414. }
  415. //根据状态返回对应的标签类型
  416. const getTagType = (state) => {
  417. switch (state) {
  418. case 0:
  419. return 'info';
  420. case 1:
  421. return 'primary';
  422. case 2:
  423. return 'success';
  424. case 3:
  425. return 'danger';
  426. default:
  427. return 'info';
  428. }
  429. }
  430. //根据状态返回对应的标签文案
  431. const getTagText = (state) => {
  432. switch (state) {
  433. case 0:
  434. return '待执行';
  435. case 1:
  436. return '执行中';
  437. case 2:
  438. return '执行完成';
  439. case 3:
  440. return '执行出错';
  441. default:
  442. return '未知状态';
  443. }
  444. }
  445. // 存储地区选择变化
  446. const selectedMarketPath = ref("")
  447. const handleMarketChange = (value) => {
  448. if (value && value.length > 0) {
  449. const lastValue = value[value.length - 1]
  450. consumeUser.value.market = reverseMarketMapping[lastValue]
  451. } else {
  452. consumeUser.value.market = ''
  453. }
  454. }
  455. // 获取地区,修改为级联下拉框
  456. const getMarket = async function () {
  457. try {
  458. // 发送POST请求
  459. const result = await API({
  460. url: '/market/selectMarket',
  461. });
  462. // 将响应结果存储到响应式数据中
  463. console.log('请求成功', result)
  464. // 递归转换树形结构为级联选择器需要的格式(跳过第一级节点)
  465. const transformTree = (nodes) => {
  466. // 直接处理第一级节点的子节点
  467. const allChildren = nodes.flatMap(node => node.children || []);
  468. return allChildren.map(child => {
  469. const grandchildren = child.children && child.children.length
  470. ? transformTree([child]) // 递归处理子节点
  471. : null;
  472. return {
  473. value: child.name,
  474. label: child.name,
  475. children: grandchildren
  476. };
  477. });
  478. };
  479. // 存储地区信息
  480. market.value = transformTree(result.data)
  481. console.log('转换后的地区树==============', market.value)
  482. } catch (error) {
  483. console.log('请求失败', error)
  484. }
  485. }
  486. </script>
  487. <template>
  488. <el-row>
  489. <el-col>
  490. <el-card style="margin-bottom: 20px;margin-top:10px">
  491. <el-row style="margin-bottom: 10px">
  492. <el-col :span="5">
  493. <div class="head-card-element">
  494. <el-text class="mx-1">精网号</el-text>
  495. <el-input v-model="consumeUser.jwcode" placeholder="请输入精网号" style="width: 150px" clearable/>
  496. </div>
  497. </el-col>
  498. <el-col :span="6">
  499. <div class="head-card-element">
  500. <el-text class="mx-1" size="large">商品名称</el-text>
  501. <el-select v-model="consumeUser.goodsName" placeholder="请选择商品名称" style="width: 180px"
  502. clearable>
  503. <!-- 修改 v-for 绑定逻辑 -->
  504. <el-option v-for="(item, index) in goods" :key="index" :label="item" :value="item"/>
  505. </el-select>
  506. </div>
  507. </el-col>
  508. <el-col :span="6">
  509. <el-text class="mx-1" size="large">所属地区</el-text>
  510. <el-cascader
  511. v-model="selectedMarketPath"
  512. :options="market"
  513. placeholder="请选择所属地区"
  514. clearable
  515. style="width:180px"
  516. @change="handleMarketChange"
  517. />
  518. </el-col>
  519. <el-col :span="6">
  520. <div class="head-card-element">
  521. <el-text class="mx-1" size="large">消耗平台</el-text>
  522. <el-select v-model="consumeUser.payPlatform" placeholder="请选择消耗平台"
  523. style="width: 180px"
  524. clearable>
  525. <el-option v-for="item in consumePlatform" :key="item.value" :label="item.label" :value="item.value"/>
  526. </el-select>
  527. </div>
  528. </el-col>
  529. </el-row>
  530. <el-row>
  531. <el-col :span="24">
  532. <div class="head-card-element">
  533. <el-text class="mx-1" size="large">消耗时间</el-text>
  534. <el-date-picker v-model="getTime" type="datetimerange" range-separator="" start-placeholder="起始时间"
  535. end-placeholder="结束时间" style="width: 400px" @change="handleDatePickerChange"
  536. :default-time="defaultTime"/>
  537. <el-button @click="getToday()" style="margin-left: 10px"
  538. :type="activeTimeRange === 'today' ? 'primary' : ''">
  539. </el-button>
  540. <el-button @click="getYesterday()" style="margin-left: 10px"
  541. :type="activeTimeRange === 'yesterday' ? 'primary' : ''">
  542. </el-button>
  543. <el-button @click="get7Days()" style="margin-left: 10px"
  544. :type="activeTimeRange === '7days' ? 'primary' : ''"> 近7天
  545. </el-button>
  546. <!-- </div>
  547. </el-col>
  548. <el-col :span="3">
  549. <div class="head-card-btn"> -->
  550. <el-button type="success" @click="reset()">重置</el-button>
  551. <el-button type="primary" @click="search()">查询</el-button>
  552. <el-button type="primary" @click="exportExcel()">导出excel</el-button>
  553. <el-button type="primary" @click="openExportList">查看导出列表</el-button>
  554. </div>
  555. </el-col>
  556. </el-row>
  557. </el-card>
  558. </el-col>
  559. </el-row>
  560. <el-row>
  561. <el-col>
  562. <el-card>
  563. <div>
  564. 消耗总金额{{ format3(Math.abs(permanentGolds) / 100) }}新币&nbsp;&nbsp;&nbsp;&nbsp;
  565. 永久金币{{ format3(Math.abs(permanentGolds) / 100) }}&nbsp;&nbsp;&nbsp;&nbsp;
  566. 免费金币{{ format3(Math.abs(freeGolds) / 100) }}&nbsp;&nbsp;&nbsp;&nbsp;
  567. 任务金币{{ format3(Math.abs(taskGolds) / 100) }}
  568. </div>
  569. <!-- 设置表格容器的高度和滚动样式 -->
  570. <div style="height: 576px; overflow-y: auto">
  571. <el-table :data="tableData" style="width: 100%" height="576px" @sort-change="handleSortChange">
  572. <el-table-column type="index" label="序号" width="80px" fixed="left">
  573. <template #default="scope">
  574. <span>{{
  575. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  576. }}</span>
  577. </template>
  578. </el-table-column>
  579. <!-- 固定姓名列 -->
  580. <el-table-column prop="name" label="姓名" width="150px" fixed="left"/>
  581. <!-- 固定精网号列 -->
  582. <el-table-column prop="jwcode" label="精网号" width="110px" fixed="left"/>
  583. <el-table-column prop="market" label="所属地区" width="110px"/>
  584. <el-table-column prop="goodsName" label="商品" width="160px" show-overflow-tooltip/>
  585. <el-table-column prop="payPlatform" label="消耗平台" width="120px">
  586. <template #default="scope">
  587. <!-- 使用非严格相等比较 -->
  588. {{ scope.row.payPlatform }}
  589. <!-- <span scope.row.consumePlatform == 1">Homily Chart</span>-->
  590. </template>
  591. </el-table-column>
  592. <!-- <el-table-column
  593. prop="consumeType"
  594. label="消费类型"
  595. width="120px"
  596. /> -->
  597. <el-table-column prop="sumGold" label="消耗金币总数" width="120px">
  598. <template #default="scope">
  599. {{
  600. (scope.row.taskGold +
  601. scope.row.freeGold +
  602. scope.row.permanentGold) / 100
  603. }}
  604. </template>
  605. </el-table-column>
  606. <el-table-column prop="permanentGold" label="永久金币" sortable="“custom”" width="110px">
  607. <template #default="scope">
  608. {{ scope.row.permanentGold / 100 }}
  609. </template>
  610. </el-table-column>
  611. <el-table-column prop="freeGold" label="免费金币" sortable="“custom”" width="110px">
  612. <template #default="scope">
  613. {{ scope.row.freeGold / 100 }}
  614. </template>
  615. </el-table-column>
  616. <el-table-column prop="taskGold" label="任务金币" sortable="“custom”" width="110px">
  617. <template #default="scope">
  618. {{ scope.row.taskGold / 100 }}
  619. </template>
  620. </el-table-column>
  621. <el-table-column prop="remark" label="备注" width="200px" show-overflow-tooltip/>
  622. <el-table-column prop="adminName" label="提交人" width="110px"/>
  623. <el-table-column prop="createTime" label="消耗时间" sortable="custom" width="180px"/>
  624. </el-table>
  625. </div>
  626. <!-- 分页 -->
  627. <div class="pagination">
  628. <el-pagination background :page-size="getObj.pageSize" :page-sizes="[5, 10, 20, 50, 100]"
  629. layout="total, sizes, prev, pager, next, jumper" :total="total"
  630. @size-change="handlePageSizeChange"
  631. @current-change="handleCurrentChange"></el-pagination>
  632. </div>
  633. </el-card>
  634. </el-col>
  635. </el-row>
  636. <!-- 导出弹窗 -->
  637. <el-dialog v-model="exportListVisible" title="导出列表" width="80%">
  638. <el-table :data="exportList" style="width: 100% ;height: 60vh;" :loading="exportListLoading">
  639. <el-table-column prop="fileName" label="文件名"/>
  640. <el-table-column prop="state" label="状态">
  641. <template #default="scope">
  642. <el-tag :type="getTagType(scope.row.state)" :effect="scope.row.state === 3 ? 'light' : 'plain'">
  643. {{ getTagText(scope.row.state) }}
  644. </el-tag>
  645. </template>
  646. </el-table-column>
  647. <el-table-column prop="createTime" label="创建时间">
  648. <template #default="scope">
  649. {{ moment(scope.row.createTime).format('YYYY-MM-DD HH:mm:ss') }}
  650. </template>
  651. </el-table-column>
  652. <el-table-column label="操作">
  653. <template #default="scope">
  654. <el-button type="primary" size="small" @click="downloadExportFile(scope.row)"
  655. :disabled="scope.row.state !== 2">
  656. 下载
  657. </el-button>
  658. </template>
  659. </el-table-column>
  660. </el-table>
  661. <template #footer>
  662. <div class="dialog-footer">
  663. <el-button text @click="exportListVisible = false">关闭</el-button>
  664. </div>
  665. </template>
  666. </el-dialog>
  667. </template>
  668. <style scoped>
  669. .status {
  670. display: flex;
  671. }
  672. .head-card {
  673. display: flex;
  674. }
  675. .head-card-element {
  676. margin-right: 20px;
  677. }
  678. .head-card-btn {
  679. margin-left: auto;
  680. }
  681. .pagination {
  682. display: flex;
  683. margin-top: 20px;
  684. }
  685. </style>