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.

737 lines
22 KiB

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