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

791 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
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
1 month ago
  1. <script setup>
  2. import { ref, onMounted, computed ,nextTick} from 'vue'
  3. import { ElMessage } from 'element-plus'
  4. import axios from 'axios'
  5. import moment from 'moment'
  6. import API from '@/util/http'
  7. import { writeFile, utils } from 'xlsx'
  8. // 变量
  9. const adminData = ref({})
  10. const getAdminData = async function () {
  11. try {
  12. const result = await API({
  13. url: '/admin/userinfo',
  14. method: 'post',
  15. data: {}
  16. })
  17. adminData.value = result
  18. console.log('请求成功', result)
  19. console.log('用户信息', adminData.value)
  20. } catch (error) {
  21. console.log('请求失败', error)
  22. }
  23. }
  24. // 定义加载状态
  25. const isLoadingArea = ref(false);
  26. const area = ref([])
  27. const getArea = async () => {
  28. isLoadingArea.value = true;
  29. try {
  30. const result = await API({
  31. url: '/detailY/getarea'
  32. });
  33. // 假设后端返回的是字符串数组,转换为 { value, label } 格式
  34. if (Array.isArray(result.data) && typeof result.data[0] === 'string') {
  35. area.value = result.data.map(item => ({ value: item, label: item }));
  36. } else {
  37. area.value = result.data;
  38. }
  39. } catch (error) {
  40. console.error('获取地区数据失败:', error);
  41. ElMessage.error('获取地区数据失败,请稍后重试');
  42. // 可以提供默认数据
  43. area.value = [];
  44. } finally {
  45. isLoadingArea.value = false;
  46. }
  47. };
  48. // 充值明细表格
  49. const tableData = ref([])
  50. // 各金币总数
  51. const rechargeCoin = ref(0)
  52. const freeCoin = ref(0)
  53. const taskCoin = ref(0)
  54. // 搜索===========================================
  55. //分页总条目
  56. const total = ref(100)
  57. // 搜索对象时间
  58. const getTime = ref([])
  59. // 搜索detailY
  60. const detailY = ref({})
  61. // 搜索对象
  62. const getObj = ref({
  63. pageNum: 1,
  64. pageSize: 50
  65. })
  66. // 开启条件筛选导出excel
  67. const getPutEX = ref(false)
  68. // 支付方式选项
  69. const num = [
  70. {
  71. value: '1',
  72. label: '增加'
  73. },
  74. {
  75. value: '2',
  76. label: '减少'
  77. }
  78. ]
  79. // 方法
  80. // 搜索===========================================================================
  81. // 搜索方法
  82. const get = async function (val) {
  83. try {
  84. // 地区赋值
  85. if (adminData.value.area === '泰国') {
  86. detailY.value.areas = ['泰国', '越南']
  87. } else if (adminData.value.area !== '总部') {
  88. detailY.value.area = adminData.value.area
  89. }
  90. // 搜索参数页码赋值
  91. if (typeof val === 'number') {
  92. getObj.value.pageNum = val
  93. }
  94. if (getTime.value.length === 2) {
  95. detailY.value.startDate = getTime.value[0]
  96. detailY.value.endDate = getTime.value[1]
  97. } else {
  98. detailY.value.startDate = ''
  99. detailY.value.endDate = ''
  100. }
  101. // 添加排序字段和排序方式到请求参数
  102. detailY.value.sortField = sortField.value
  103. detailY.value.sortOrder = sortOrder.value
  104. console.log('搜索参数', getObj.value)
  105. // 发送POST请求
  106. const result = await API({
  107. url: '/detailY',
  108. method: 'post',
  109. data: { ...getObj.value, detailY: { ...detailY.value } }
  110. })
  111. tableData.value = result.data.list
  112. total.value = result.data.total
  113. } catch (error) {
  114. console.log('请求失败', error)
  115. }
  116. }
  117. // 精网号去空格,处理 detailY 中的 jwcode
  118. const trimJwCode = () => {
  119. if (detailY.value.jwcode) {
  120. detailY.value.jwcode = detailY.value.jwcode.replace(/\s/g, '');
  121. }
  122. }
  123. // 搜索
  124. const search = function () {
  125. trimJwCode();
  126. getObj.value.pageNum = 1
  127. get()
  128. }
  129. // 重置
  130. const reset = function () {
  131. delete detailY.value.jwcode
  132. delete detailY.value.num
  133. delete detailY.value.startDate
  134. delete detailY.value.endDate
  135. delete detailY.value.area
  136. delete sortField.value
  137. delete sortOrder.value
  138. getTime.value = []
  139. delete detailY.value.consumePlatform
  140. }
  141. // 今天
  142. const getToday = function () {
  143. const today = moment()
  144. const startDate = today.startOf('day').toDate()
  145. const endDate = today.endOf('day').toDate()
  146. getTime.value = [startDate, endDate]
  147. search()
  148. }
  149. // 昨天
  150. const getYesterday = function () {
  151. const yesterday = moment().subtract(1, 'day')
  152. const startDate = yesterday.startOf('day').toDate()
  153. const endDate = yesterday.endOf('day').toDate()
  154. getTime.value = [startDate, endDate]
  155. search()
  156. }
  157. // 近7天
  158. const get7Days = function () {
  159. const startDate = moment().subtract(6, 'day').startOf('day').toDate()
  160. const endDate = moment().endOf('day').toDate()
  161. getTime.value = [startDate, endDate]
  162. search()
  163. }
  164. // 验证跳转输入框的数字是否合法
  165. const checkNumber = function () {
  166. if (typeof parseInt(getObj.value.pageNum) === 'number') {
  167. console.log('总共有多少页' + Math.ceil(total.value / getObj.value.pageSize))
  168. if (
  169. getObj.value.pageNum > 0 &&
  170. getObj.value.pageNum <= Math.ceil(total.value / getObj.value.pageSize)
  171. ) {
  172. getObj.value.pageNum = parseInt(getObj.value.pageNum)
  173. console.log('输入的数字合法')
  174. get()
  175. } else {
  176. //提示
  177. ElMessage({
  178. type: 'error',
  179. message: '请检查输入内容'
  180. })
  181. }
  182. } else {
  183. //提示
  184. ElMessage({
  185. type: 'error',
  186. message: '请检查输入内容'
  187. })
  188. }
  189. }
  190. // 挂载
  191. onMounted(async function () {
  192. await getArea()
  193. await getAdminData()
  194. await get()
  195. })
  196. // 导出Excel的方法
  197. const headers = [
  198. '序号',
  199. '姓名',
  200. '精网号',
  201. '所属地区',
  202. '平台信息',
  203. '更新数量',
  204. '更新类型',
  205. '永久金币',
  206. '免费金币',
  207. '任务金币',
  208. '提交人',
  209. '更新时间'
  210. ]
  211. const showExportInfoPanel = ref(false)
  212. // 点击导出按钮直接显示信息面板
  213. const exportExcel = async () => {
  214. try {
  215. console.log('点击导出按钮,尝试显示信息面板');
  216. showExportInfoPanel.value = true;
  217. await nextTick();//组件更新显示信息面板
  218. } catch (error) {
  219. console.error('显示信息面板失败:', error);
  220. ElMessage.error('显示信息面板失败,请稍后重试');
  221. }
  222. };
  223. // 新增导出状态相关变量
  224. const exportProgress = ref(0)
  225. const isExporting = ref(false)
  226. const exportCancelToken = ref(null)
  227. // 优化后的导出方法
  228. const doExportExcel = async () => {
  229. try {
  230. isExporting.value = true
  231. exportProgress.value = 0
  232. showExportInfoPanel.value = false
  233. // 初始化Excel
  234. const wb = utils.book_new()
  235. const ws = utils.aoa_to_sheet([headers])
  236. utils.book_append_sheet(wb, ws, 'Sheet1')
  237. // 流式写入配置
  238. const writer = {
  239. write: (d, o) => {
  240. if (!d) return
  241. utils.sheet_add_aoa(ws, d, { origin: -1 })
  242. }
  243. }
  244. let page = 1
  245. let totalExported = 0
  246. const pageSize = 5000 // 每次请求5000条
  247. let totalRecords = 0
  248. // 首次请求获取总记录数
  249. const firstResult = await API({
  250. url: '/detailY',
  251. method: 'post',
  252. data: {
  253. pageNum: 1,
  254. pageSize,
  255. detailY: { ...detailY.value }
  256. }
  257. })
  258. totalRecords = firstResult.data.total
  259. // 创建取消令牌
  260. const CancelToken = axios.CancelToken
  261. exportCancelToken.value = CancelToken.source()
  262. // 平台信息映射
  263. const platformMap = {
  264. 0: '初始化金币',
  265. 1: 'ERP系统',
  266. 2: 'Homily Chart',
  267. 3: 'Homily Link',
  268. 4: '金币系统'
  269. };
  270. // 更新类型映射
  271. const updateTypeMap = {
  272. 0: '充值',
  273. 1: '消费',
  274. 2: '退款',
  275. 3: '其他'
  276. };
  277. // 处理首次请求的数据
  278. const firstData = firstResult.data.list
  279. if (firstData.length) {
  280. const rows = firstData.map((row, index) => [
  281. totalExported + index + 1,
  282. row.username || '',
  283. row.jwcode || '',
  284. row.area || '',
  285. platformMap[row.consumePlatform] || '',
  286. (row.gold).toFixed(2) || '0.00',
  287. updateTypeMap[row.updateType] || '',
  288. (row.rechargeCoin / 100).toFixed(2) || '0.00',
  289. (row.freeCoin / 100).toFixed(2) || '0.00',
  290. (row.taskCoin / 100).toFixed(2) || '0.00',
  291. row.name || '',
  292. moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') || ''
  293. ])
  294. writer.write(rows)
  295. totalExported += firstData.length
  296. exportProgress.value = Math.round((totalExported / totalRecords) * 100)
  297. page++
  298. }
  299. while (totalExported < totalRecords) {
  300. const result = await API({
  301. url: '/detailY',
  302. method: 'post',
  303. data: {
  304. pageNum: page,
  305. pageSize,
  306. detailY: { ...detailY.value }
  307. },
  308. cancelToken: exportCancelToken.value.token
  309. })
  310. const data = result.data.list
  311. if (!data.length) break
  312. // 转换数据
  313. const rows = data.map((row, index) => [
  314. totalExported + index + 1,
  315. row.username || '',
  316. row.jwcode || '',
  317. row.area || '',
  318. platformMap[row.consumePlatform] || '',
  319. (row.gold / 100).toFixed(2) || '0.00',
  320. updateTypeMap[row.updateType] || '',
  321. (row.rechargeCoin / 100).toFixed(2) || '0.00',
  322. (row.freeCoin / 100).toFixed(2) || '0.00',
  323. (row.taskCoin / 100).toFixed(2) || '0.00',
  324. row.name || '',
  325. moment(row.createTime).format('YYYY-MM-DD HH:mm:ss') || ''
  326. ])
  327. // 流式写入
  328. writer.write(rows)
  329. totalExported += data.length
  330. exportProgress.value = Math.round((totalExported / totalRecords) * 100)
  331. // 内存控制:每500页释放内存
  332. if (page % 500 === 0) {
  333. await new Promise(resolve => setTimeout(resolve, 0))
  334. }
  335. page++
  336. }
  337. // 生成最终文件
  338. writeFile(wb, '客户金币明细.xlsx')
  339. ElMessage.success(`导出成功,共${totalExported}条数据`)
  340. } catch (error) {
  341. if (!axios.isCancel(error)) {
  342. ElMessage.error(`导出失败: ${error.message}`)
  343. }
  344. } finally {
  345. isExporting.value = false
  346. exportCancelToken.value = null
  347. }
  348. }
  349. // 新增取消导出方法
  350. const cancelExport = () => {
  351. if (exportCancelToken.value) {
  352. exportCancelToken.value.cancel('用户取消导出')
  353. ElMessage.warning('导出已取消')
  354. isExporting.value = false
  355. }
  356. }
  357. const putExcel = ref({
  358. startDate: new Date(),
  359. endDate: new Date(new Date().setDate(new Date().getDate() + 1))
  360. })
  361. // 新增校验精网号的方法
  362. const checkJwCode = async (jwcode) => {
  363. try {
  364. const result = await API({
  365. url: '/recharge/user',
  366. method: 'post',
  367. data: {
  368. jwcode,
  369. area: adminData.value.area
  370. }
  371. })
  372. // 根据后端返回的 code 判断精网号是否存在
  373. return result.code !== 0
  374. } catch (error) {
  375. console.log('校验精网号失败', error)
  376. return false
  377. }
  378. }
  379. // 选消费平台
  380. const platform = [
  381. {
  382. value: '4',
  383. label: '金币系统'
  384. },
  385. {
  386. value: '1',
  387. label: 'ERP系统'
  388. },
  389. {
  390. value: '2',
  391. label: 'Homily Chart'
  392. },
  393. {
  394. value: '3',
  395. label: 'Homily Link'
  396. },
  397. {
  398. value: '0',
  399. label: '初始化金币'
  400. }
  401. ]
  402. // 新增排序字段和排序方式
  403. const sortField = ref('')
  404. const sortOrder = ref('')
  405. // 处理排序事件
  406. const handleSortChange = (column) => {
  407. if (column.prop === 'rechargeCoin') {
  408. sortField.value = 'recharge_coin'
  409. } else if (column.prop === 'taskCoin') {
  410. sortField.value = 'task_coin'
  411. } else if (column.prop === 'freeCoin') {
  412. sortField.value = 'free_coin'
  413. } else if (column.prop === 'createTime') {
  414. sortField.value = 'create_time'
  415. } else if (column.prop === 'gold') {
  416. sortField.value = 'gold'
  417. }
  418. sortOrder.value = column.order === 'ascending' ? 'ASC' : 'DESC'
  419. }
  420. get()
  421. const handlePageSizeChange = function (val) {
  422. getObj.value.pageSize = val
  423. get()
  424. }
  425. const handleCurrentChange = function (val) {
  426. getObj.value.pageNum = val
  427. get()
  428. }
  429. </script>
  430. <template>
  431. <!-- 导出excel提前展示的信息面板 -->
  432. <el-dialog
  433. v-model="showExportInfoPanel"
  434. title="导出信息确认"
  435. width="400px"
  436. :close-on-click-modal="false"
  437. >
  438. <div class="info-panel-header">导出信息</div>
  439. <!-- 直接使用 detailY 显示信息添加可选链操作符 -->
  440. <!-- detailY是一个ref所以在模板中应该直接使用detailY.consumePlatform
  441. 而不是detailY.value.consumePlatform
  442. 因为在模板中ref变量会自动解包不需要.value
  443. 例如在代码中我们可能错误地在模板中使用了detailY.value但实际上应该直接使用detailY -->
  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. <!-- 直接使用 getTime 而非 getTime.value -->
  457. <span v-if="Array.isArray(getTime) && getTime.length >= 2">
  458. {{ moment(getTime[0]).format('YYYY-MM-DD') }} {{ moment(getTime[1]).format('YYYY-MM-DD') }}
  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 type="success" @click="getToday()"></el-button>
  572. <el-button type="success" @click="getYesterday()"></el-button>
  573. <el-button type="success" @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>