金币系统前端
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.

790 lines
22 KiB

1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
3 weeks ago
3 weeks 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 month ago
1 month ago
1 month ago
3 weeks ago
3 weeks 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 month ago
3 weeks ago
1 month ago
1 month ago
1 month ago
1 month ago
1 month ago
3 weeks ago
1 month ago
1 month ago
3 weeks 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
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
3 weeks ago
1 month ago
  1. <!-- 这是客户金币明细页面 -->
  2. <script setup>
  3. import { ref, onMounted, computed ,nextTick} from 'vue'
  4. import { ElMessage } from 'element-plus'
  5. import axios from 'axios'
  6. import moment from 'moment'
  7. import API from '@/util/http'
  8. import { writeFile, utils } from 'xlsx'
  9. // 变量
  10. const adminData = ref({})
  11. const getAdminData = async function () {
  12. try {
  13. const result = await API({
  14. url: '/admin/userinfo',
  15. method: 'post',
  16. data: {}
  17. })
  18. adminData.value = result
  19. console.log('请求成功', result)
  20. console.log('用户信息', adminData.value)
  21. } catch (error) {
  22. console.log('请求失败', error)
  23. }
  24. }
  25. // 定义加载状态
  26. const isLoadingArea = ref(false);
  27. const area = ref([])
  28. const getArea = async () => {
  29. isLoadingArea.value = true;
  30. try {
  31. const result = await API({
  32. url: '/detailY/getarea'
  33. });
  34. // 假设后端返回的是字符串数组,转换为 { value, label } 格式
  35. if (Array.isArray(result.data) && typeof result.data[0] === 'string') {
  36. area.value = result.data.map(item => ({ value: item, label: item }));
  37. } else {
  38. area.value = result.data;
  39. }
  40. } catch (error) {
  41. console.error('获取地区数据失败:', error);
  42. ElMessage.error('获取地区数据失败,请稍后重试');
  43. // 可以提供默认数据
  44. area.value = [];
  45. } finally {
  46. isLoadingArea.value = false;
  47. }
  48. };
  49. // 充值明细表格
  50. const tableData = ref([])
  51. // 各金币总数
  52. const rechargeCoin = ref(0)
  53. const freeCoin = ref(0)
  54. const taskCoin = ref(0)
  55. // 搜索===========================================
  56. //分页总条目
  57. const total = ref(100)
  58. // 搜索对象时间
  59. const getTime = ref([])
  60. // 搜索detailY
  61. const detailY = ref({})
  62. // 搜索对象
  63. const getObj = ref({
  64. pageNum: 1,
  65. pageSize: 50
  66. })
  67. // 开启条件筛选导出excel
  68. const getPutEX = ref(false)
  69. // 支付方式选项
  70. const num = [
  71. {
  72. value: '1',
  73. label: '增加'
  74. },
  75. {
  76. value: '2',
  77. label: '减少'
  78. }
  79. ]
  80. // 方法
  81. // 搜索===========================================================================
  82. // 搜索方法
  83. const get = async function (val) {
  84. try {
  85. // 地区赋值
  86. if (adminData.value.area === '泰国') {
  87. detailY.value.areas = ['泰国', '越南']
  88. } else if (adminData.value.area !== '总部') {
  89. detailY.value.area = adminData.value.area
  90. }
  91. // 搜索参数页码赋值
  92. if (typeof val === 'number') {
  93. getObj.value.pageNum = val
  94. }
  95. if (getTime.value.length === 2) {
  96. detailY.value.startDate = getTime.value[0]
  97. detailY.value.endDate = getTime.value[1]
  98. } else {
  99. detailY.value.startDate = ''
  100. detailY.value.endDate = ''
  101. }
  102. // 添加排序字段和排序方式到请求参数
  103. detailY.value.sortField = sortField.value
  104. detailY.value.sortOrder = sortOrder.value
  105. console.log('搜索参数', getObj.value)
  106. // 发送POST请求
  107. const result = await API({
  108. url: '/detailY',
  109. method: 'post',
  110. data: { ...getObj.value, detailY: { ...detailY.value } }
  111. })
  112. tableData.value = result.data.list
  113. total.value = result.data.total
  114. } catch (error) {
  115. console.log('请求失败', error)
  116. }
  117. }
  118. // 精网号去空格,处理 detailY 中的 jwcode
  119. const trimJwCode = () => {
  120. if (detailY.value.jwcode) {
  121. detailY.value.jwcode = detailY.value.jwcode.replace(/\s/g, '');
  122. }
  123. }
  124. // 搜索
  125. const search = function () {
  126. trimJwCode();
  127. getObj.value.pageNum = 1
  128. get()
  129. }
  130. // 重置
  131. const reset = function () {
  132. delete detailY.value.jwcode
  133. delete detailY.value.num
  134. delete detailY.value.startDate
  135. delete detailY.value.endDate
  136. delete detailY.value.area
  137. delete sortField.value
  138. delete sortOrder.value
  139. getTime.value = []
  140. delete detailY.value.consumePlatform
  141. }
  142. // 今天
  143. const getToday = function () {
  144. const today = moment()
  145. const startDate = today.startOf('day').toDate()
  146. const endDate = today.endOf('day').toDate()
  147. getTime.value = [startDate, endDate]
  148. search()
  149. }
  150. // 昨天
  151. const getYesterday = function () {
  152. const yesterday = moment().subtract(1, 'day')
  153. const startDate = yesterday.startOf('day').toDate()
  154. const endDate = yesterday.endOf('day').toDate()
  155. getTime.value = [startDate, endDate]
  156. search()
  157. }
  158. // 近7天
  159. const get7Days = function () {
  160. const startDate = moment().subtract(6, 'day').startOf('day').toDate()
  161. const endDate = moment().endOf('day').toDate()
  162. getTime.value = [startDate, endDate]
  163. search()
  164. }
  165. // 验证跳转输入框的数字是否合法
  166. const checkNumber = function () {
  167. if (typeof parseInt(getObj.value.pageNum) === 'number') {
  168. console.log('总共有多少页' + Math.ceil(total.value / getObj.value.pageSize))
  169. if (
  170. getObj.value.pageNum > 0 &&
  171. getObj.value.pageNum <= Math.ceil(total.value / getObj.value.pageSize)
  172. ) {
  173. getObj.value.pageNum = parseInt(getObj.value.pageNum)
  174. console.log('输入的数字合法')
  175. get()
  176. } else {
  177. //提示
  178. ElMessage({
  179. type: 'error',
  180. message: '请检查输入内容'
  181. })
  182. }
  183. } else {
  184. //提示
  185. ElMessage({
  186. type: 'error',
  187. message: '请检查输入内容'
  188. })
  189. }
  190. }
  191. // 挂载
  192. onMounted(async function () {
  193. await getArea()
  194. await getAdminData()
  195. await get()
  196. })
  197. // 导出Excel的方法
  198. const headers = [
  199. '序号',
  200. '姓名',
  201. '精网号',
  202. '所属地区',
  203. '平台信息',
  204. '更新数量',
  205. '更新类型',
  206. '永久金币',
  207. '免费金币',
  208. '任务金币',
  209. '提交人',
  210. '更新时间'
  211. ]
  212. const showExportInfoPanel = ref(false)
  213. // 点击导出按钮直接显示信息面板
  214. const exportExcel = async () => {
  215. try {
  216. console.log('点击导出按钮,尝试显示信息面板');
  217. showExportInfoPanel.value = true;
  218. await nextTick();//组件更新显示信息面板
  219. } catch (error) {
  220. console.error('显示信息面板失败:', error);
  221. ElMessage.error('显示信息面板失败,请稍后重试');
  222. }
  223. };
  224. // 新增导出状态相关变量
  225. const exportProgress = ref(0)
  226. const isExporting = ref(false)
  227. const exportCancelToken = ref(null)
  228. // 优化后的导出方法
  229. const doExportExcel = async () => {
  230. try {
  231. isExporting.value = true
  232. exportProgress.value = 0
  233. showExportInfoPanel.value = false
  234. // 初始化Excel
  235. const wb = utils.book_new()
  236. const ws = utils.aoa_to_sheet([headers])
  237. utils.book_append_sheet(wb, ws, 'Sheet1')
  238. // 流式写入配置
  239. const writer = {
  240. write: (d, o) => {
  241. if (!d) return
  242. utils.sheet_add_aoa(ws, d, { origin: -1 })
  243. }
  244. }
  245. let page = 1
  246. let totalExported = 0
  247. const pageSize = 5000 // 每次请求5000条
  248. let totalRecords = 0
  249. // 首次请求获取总记录数
  250. const firstResult = await API({
  251. url: '/detailY',
  252. method: 'post',
  253. data: {
  254. pageNum: 1,
  255. pageSize,
  256. detailY: { ...detailY.value }
  257. }
  258. })
  259. totalRecords = firstResult.data.total
  260. // 创建取消令牌
  261. const CancelToken = axios.CancelToken
  262. exportCancelToken.value = CancelToken.source()
  263. // 平台信息映射
  264. const platformMap = {
  265. 0: '初始化金币',
  266. 1: 'ERP系统',
  267. 2: 'Homily Chart',
  268. 3: 'Homily Link',
  269. 4: '金币系统'
  270. };
  271. // 更新类型映射
  272. const updateTypeMap = {
  273. 0: '充值',
  274. 1: '消费',
  275. 2: '退款',
  276. 3: '其他'
  277. };
  278. // 处理首次请求的数据
  279. const firstData = firstResult.data.list
  280. if (firstData.length) {
  281. const rows = firstData.map((row, index) => {
  282. const consumePlatform = parseInt(row.consumePlatform, 10); // 转换为数字类型
  283. const platformInfo = platformMap[consumePlatform] || '';
  284. return [
  285. totalExported + index + 1,
  286. row.username || '',
  287. row.jwcode || '',
  288. row.area || '',
  289. platformInfo,
  290. (row.gold / 100).toFixed(2) || '0.00',
  291. updateTypeMap[row.updateType] || '',
  292. (row.rechargeCoin / 100).toFixed(2) || '0.00',
  293. (row.freeCoin / 100).toFixed(2) || '0.00',
  294. (row.taskCoin / 100).toFixed(2) || '0.00',
  295. row.name || '',
  296. moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') || ''
  297. ]
  298. })
  299. writer.write(rows)
  300. totalExported += firstData.length
  301. exportProgress.value = Math.round((totalExported / totalRecords) * 100)
  302. page++
  303. }
  304. while (totalExported < totalRecords) {
  305. const result = await API({
  306. url: '/detailY',
  307. method: 'post',
  308. data: {
  309. pageNum: page,
  310. pageSize,
  311. detailY: { ...detailY.value }
  312. },
  313. cancelToken: exportCancelToken.value.token
  314. })
  315. const data = result.data.list
  316. if (!data.length) break
  317. // 转换数据
  318. const rows = data.map((row, index) => [
  319. totalExported + index + 1,
  320. row.username || '',
  321. row.jwcode || '',
  322. row.area || '',
  323. platformMap[row.consumePlatform] || '',
  324. (row.gold / 100).toFixed(2) || '0.00',
  325. updateTypeMap[row.updateType] || '',
  326. (row.rechargeCoin / 100).toFixed(2) || '0.00',
  327. (row.freeCoin / 100).toFixed(2) || '0.00',
  328. (row.taskCoin / 100).toFixed(2) || '0.00',
  329. row.name || '',
  330. moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') || ''
  331. ])
  332. // 流式写入
  333. writer.write(rows)
  334. totalExported += data.length
  335. exportProgress.value = Math.round((totalExported / totalRecords) * 100)
  336. // 内存控制:每500页释放内存
  337. if (page % 500 === 0) {
  338. await new Promise(resolve => setTimeout(resolve, 0))
  339. }
  340. page++
  341. }
  342. // 生成最终文件
  343. writeFile(wb, '客户金币明细.xlsx')
  344. ElMessage.success(`导出成功,共${totalExported}条数据`)
  345. } catch (error) {
  346. if (!axios.isCancel(error)) {
  347. ElMessage.error(`导出失败: ${error.message}`)
  348. }
  349. } finally {
  350. isExporting.value = false
  351. exportCancelToken.value = null
  352. }
  353. }
  354. // 新增取消导出方法
  355. const cancelExport = () => {
  356. if (exportCancelToken.value) {
  357. exportCancelToken.value.cancel('用户取消导出')
  358. ElMessage.warning('导出已取消')
  359. isExporting.value = false
  360. }
  361. }
  362. const putExcel = ref({
  363. startDate: new Date(),
  364. endDate: new Date(new Date().setDate(new Date().getDate() + 1))
  365. })
  366. // 新增校验精网号的方法
  367. const checkJwCode = async (jwcode) => {
  368. try {
  369. const result = await API({
  370. url: '/recharge/user',
  371. method: 'post',
  372. data: {
  373. jwcode,
  374. area: adminData.value.area
  375. }
  376. })
  377. // 根据后端返回的 code 判断精网号是否存在
  378. return result.code !== 0
  379. } catch (error) {
  380. console.log('校验精网号失败', error)
  381. return false
  382. }
  383. }
  384. // 选消费平台
  385. const platform = [
  386. {
  387. value: '4',
  388. label: '金币系统'
  389. },
  390. {
  391. value: '1',
  392. label: 'ERP系统'
  393. },
  394. {
  395. value: '2',
  396. label: 'Homily Chart'
  397. },
  398. {
  399. value: '3',
  400. label: 'Homily Link'
  401. },
  402. {
  403. value: '0',
  404. label: '初始化金币'
  405. }
  406. ]
  407. // 新增排序字段和排序方式
  408. const sortField = ref('')
  409. const sortOrder = ref('')
  410. // 处理排序事件
  411. const handleSortChange = (column) => {
  412. if (column.prop === 'rechargeCoin') {
  413. sortField.value = 'recharge_coin'
  414. } else if (column.prop === 'taskCoin') {
  415. sortField.value = 'task_coin'
  416. } else if (column.prop === 'freeCoin') {
  417. sortField.value = 'free_coin'
  418. } else if (column.prop === 'createTime') {
  419. sortField.value = 'create_time'
  420. } else if (column.prop === 'gold') {
  421. sortField.value = 'gold'
  422. }
  423. sortOrder.value = column.order === 'ascending' ? 'ASC' : 'DESC'
  424. }
  425. get()
  426. const handlePageSizeChange = function (val) {
  427. getObj.value.pageSize = val
  428. get()
  429. }
  430. const handleCurrentChange = function (val) {
  431. getObj.value.pageNum = val
  432. get()
  433. }
  434. </script>
  435. <template>
  436. <!-- 导出excel提前展示的信息面板 -->
  437. <el-dialog
  438. v-model="showExportInfoPanel"
  439. title="导出信息确认"
  440. width="400px"
  441. :close-on-click-modal="false"
  442. >
  443. <div class="info-panel-header">导出信息</div>
  444. <div v-if="!detailY.jwcode && !detailY.consumePlatform && !detailY.num && !detailY.area && (getTime.length < 2)">
  445. 你正在导出所有数据
  446. </div>
  447. <div v-else>
  448. 你正在导出以下数据
  449. </div>
  450. <div v-if="detailY.jwcode">精网号{{ detailY.jwcode || '' }}</div>
  451. <div v-if="detailY.consumePlatform">平台信息{{ detailY.consumePlatform ? (platform.find(item => item.value === detailY.consumePlatform)?.label) : '' }}</div>
  452. <div v-if="detailY.num">数量更新类型{{ detailY.num ? (num.find(item => item.value === detailY.num)?.label || '') : '' }}</div>
  453. <div v-if="detailY.area">所属地区{{ detailY.area || '' }}</div>
  454. <div v-if="Array.isArray(getTime) && getTime.length >= 2">
  455. <span>更新时间</span>
  456. <!-- 修改时间格式为精确到秒 -->
  457. <span v-if="Array.isArray(getTime) && getTime.length >= 2">
  458. {{ moment(getTime[0]).format('YYYY-MM-DD HH:mm:ss') }} {{ moment(getTime[1]).format('YYYY-MM-DD HH:mm:ss') }}
  459. </span>
  460. <span v-else></span>
  461. </div>
  462. <template #footer>
  463. <span class="dialog-footer">
  464. <el-button @click="showExportInfoPanel = false">取消</el-button>
  465. <el-button type="primary" @click="doExportExcel">导出</el-button>
  466. </span>
  467. </template>
  468. </el-dialog>
  469. <!-- 导出进度弹窗 -->
  470. <el-dialog
  471. v-model="isExporting"
  472. title="正在导出"
  473. width="400px"
  474. :close-on-click-modal="false"
  475. :show-close="false"
  476. >
  477. <el-progress
  478. :percentage="exportProgress"
  479. :stroke-width="15"
  480. striped
  481. animated
  482. />
  483. <div class="export-status">
  484. 已导出 {{ Math.round((exportProgress / 100) * total) }} / {{ total }}
  485. </div>
  486. <template #footer>
  487. <el-button type="danger" @click="cancelExport">取消导出</el-button>
  488. </template>
  489. </el-dialog>
  490. <el-row>
  491. <el-col>
  492. <el-card style="margin-bottom: 20px">
  493. <el-row style="margin-bottom: 10px">
  494. <el-col :span="6">
  495. <div class="head-card-element">
  496. <el-text class="mx-1" size="large">精网号</el-text>
  497. <el-input
  498. v-model="detailY.jwcode"
  499. style="width: 240px"
  500. placeholder="请输入精网号"
  501. clearable
  502. />
  503. </div>
  504. </el-col>
  505. <el-col :span="6">
  506. <div class="head-card-element">
  507. <el-text class="mx-1" size="large">平台信息</el-text>
  508. <el-select
  509. v-model="detailY.consumePlatform"
  510. placeholder="请选择平台信息"
  511. style="width: 200px"
  512. clearable
  513. >
  514. <el-option
  515. v-for="item in platform"
  516. :key="item.value"
  517. :label="item.label"
  518. :value="item.value"
  519. />
  520. </el-select>
  521. </div>
  522. </el-col>
  523. <el-col :span="6">
  524. <div class="head-card-element">
  525. <el-text class="mx-1" size="large">数量更新类型</el-text>
  526. <el-select
  527. v-model="detailY.num"
  528. placeholder="请选择更新类型"
  529. style="width: 200px"
  530. clearable
  531. >
  532. <el-option
  533. v-for="item in num"
  534. :key="item.value"
  535. :label="item.label"
  536. :value="item.value"
  537. />
  538. </el-select>
  539. </div>
  540. </el-col>
  541. <el-col :span="6">
  542. <div class="head-card-element">
  543. <el-text class="mx-1" size="large">所属地区</el-text>
  544. <el-select
  545. v-model="detailY.area"
  546. placeholder="请选择所属地区"
  547. style="width: 240px"
  548. clearable
  549. :loading="isLoadingArea"
  550. >
  551. <el-option
  552. v-for="item in area"
  553. :key="item.value || item"
  554. :label="item.label || item"
  555. :value="item.value || item"
  556. />
  557. </el-select>
  558. </div>
  559. </el-col>
  560. </el-row>
  561. <div class="head-card-element">
  562. <el-text class="mx-1" size="large">更新时间</el-text>
  563. <el-date-picker
  564. v-model="getTime"
  565. type="datetimerange"
  566. range-separator="至"
  567. start-placeholder="起始时间"
  568. end-placeholder="结束时间"
  569. style="margin-right: 10px"
  570. />
  571. <el-button @click="getToday()"></el-button>
  572. <el-button @click="getYesterday()"></el-button>
  573. <el-button @click="get7Days()">近7天</el-button>
  574. <el-button type="success" @click="exportExcel">导出Excel表格</el-button>
  575. <el-button type="success" @click="reset()">重置</el-button>
  576. <el-button type="primary" @click="search()">查询</el-button>
  577. </div>
  578. </el-card>
  579. </el-col>
  580. </el-row>
  581. <el-row>
  582. <el-col>
  583. <el-card>
  584. <div style="height: 584px; overflow-y: auto">
  585. <el-table
  586. :data="tableData"
  587. style="width: 100%"
  588. @sort-change="handleSortChange"
  589. height="584px"
  590. >
  591. <el-table-column
  592. type="index"
  593. label="序号"
  594. width="100px"
  595. fixed="left"
  596. >
  597. <template #default="scope">
  598. <span>{{
  599. scope.$index + 1 + (getObj.pageNum - 1) * getObj.pageSize
  600. }}</span>
  601. </template>
  602. </el-table-column>
  603. <el-table-column
  604. fixed="left"
  605. prop="username"
  606. label="姓名"
  607. width="150"
  608. />
  609. <el-table-column
  610. fixed="left"
  611. prop="jwcode"
  612. label="精网号"
  613. width="120"
  614. />
  615. <el-table-column prop="area" label="所属地区" width="120" />
  616. <el-table-column
  617. prop="consumePlatform"
  618. label="平台信息"
  619. width="140"
  620. >
  621. <template #default="scope">
  622. <!-- 使用非严格相等比较 -->
  623. <span v-if="scope.row.consumePlatform == 0">初始化金币</span>
  624. <span v-if="scope.row.consumePlatform == 1">ERP系统</span>
  625. <span v-if="scope.row.consumePlatform == 3">Homily Link</span>
  626. <span v-if="scope.row.consumePlatform == 2">Homily Chart</span>
  627. <span v-if="scope.row.consumePlatform == 4">金币系统</span>
  628. </template>
  629. </el-table-column>
  630. <el-table-column
  631. prop="gold"
  632. label="更新数量"
  633. width="120"
  634. sortable="custom"
  635. >
  636. <template #default="scope">
  637. <span>{{ scope.row.gold / 100 }}</span>
  638. </template>
  639. </el-table-column>
  640. <el-table-column prop="updateType" label="更新类型" width="110">
  641. <!-- 模板内容 -->
  642. <template #default="scope">
  643. <span v-if="scope.row.updateType == 1">消费</span>
  644. <span v-if="scope.row.updateType == 0">充值</span>
  645. <span v-if="scope.row.updateType == 2">退款</span>
  646. <span v-if="scope.row.updateType == 3">其他</span>
  647. </template>
  648. </el-table-column>
  649. <el-table-column
  650. prop="rechargeCoin"
  651. sortable="custom"
  652. label="永久金币"
  653. width="110"
  654. >
  655. <template #default="scope">
  656. <span>{{ scope.row.rechargeCoin / 100 }}</span>
  657. </template>
  658. </el-table-column>
  659. <el-table-column
  660. prop="freeCoin"
  661. sortable="custom"
  662. label="免费金币"
  663. width="110"
  664. >
  665. <template #default="scope">
  666. <span>{{ scope.row.freeCoin / 100 }}</span>
  667. </template>
  668. </el-table-column>
  669. <el-table-column
  670. prop="taskCoin"
  671. sortable="custom"
  672. label="任务金币"
  673. width="110"
  674. >
  675. <template #default="scope">
  676. <span>{{ scope.row.taskCoin / 100 }}</span>
  677. </template>
  678. </el-table-column>
  679. <el-table-column prop="name" label="提交人" width="110" />
  680. <el-table-column
  681. prop="createTime"
  682. sortable="custom"
  683. label="更新时间"
  684. width="210"
  685. show-overflow-tooltip
  686. >
  687. <template #default="scope">
  688. <span>{{
  689. moment(scope.row.createTime).format('YYYY-MM-DD HH:mm:ss')
  690. }}</span>
  691. </template>
  692. </el-table-column>
  693. </el-table>
  694. </div>
  695. <!-- 此处分页 -->
  696. <div class="pagination" style="margin-top: 20px">
  697. <el-pagination
  698. background
  699. :page-size="getObj.pageSize"
  700. :page-sizes="[5, 10, 20, 50, 100]"
  701. layout="total, sizes, prev, pager, next, jumper"
  702. :total="total"
  703. @size-change="handlePageSizeChange"
  704. @current-change="handleCurrentChange"
  705. ></el-pagination>
  706. </div>
  707. </el-card>
  708. </el-col>
  709. </el-row>
  710. </template>
  711. <style scoped>
  712. .pagination {
  713. display: flex;
  714. }
  715. .status {
  716. display: flex;
  717. }
  718. .head-card {
  719. display: flex;
  720. }
  721. .info-panel-header {
  722. font-weight: bold;
  723. margin-bottom: 10px;
  724. }
  725. .dialog-footer {
  726. display: flex;
  727. justify-content: flex-end;
  728. }
  729. .export-status {
  730. margin-top: 15px;
  731. text-align: center;
  732. color: #666;
  733. }
  734. .el-progress-bar__inner {
  735. transition: width 0.5s ease;
  736. }
  737. </style>