|
@@ -7,314 +7,268 @@
|
|
|
</div>
|
|
|
|
|
|
<!-- 顶部统计卡片 -->
|
|
|
- <statistics-panel />
|
|
|
+ <statistics-panel :statistics-data="statisticsPanelData" />
|
|
|
|
|
|
- <!-- 生产概况表格 -->
|
|
|
- <production-table
|
|
|
- :table-data="tableData"
|
|
|
+ <!-- 按订单查询表格 -->
|
|
|
+ <order-production-table
|
|
|
+ v-if="queryType === 'order'"
|
|
|
:update-time="updateTime"
|
|
|
- @unassigned-click="handleUnassignedOrders"
|
|
|
@progress-click="handleProgressClick"
|
|
|
- />
|
|
|
+ @data-loaded="handleDataLoaded"
|
|
|
+ >
|
|
|
+ <template #query-buttons>
|
|
|
+ <el-button-group>
|
|
|
+ <el-button
|
|
|
+ type="primary"
|
|
|
+ size="small"
|
|
|
+ :plain="queryType !== 'order'"
|
|
|
+ @click="switchQueryType('order')"
|
|
|
+ >
|
|
|
+ 按订单查询
|
|
|
+ </el-button>
|
|
|
+ <el-button
|
|
|
+ type="primary"
|
|
|
+ size="small"
|
|
|
+ :plain="queryType !== 'product'"
|
|
|
+ @click="switchQueryType('product')"
|
|
|
+ >
|
|
|
+ 按货品查询
|
|
|
+ </el-button>
|
|
|
+ </el-button-group>
|
|
|
+ </template>
|
|
|
+ </order-production-table>
|
|
|
|
|
|
- <!-- 未接单弹窗 -->
|
|
|
- <unassigned-dialog
|
|
|
- :visible.sync="dialogVisible"
|
|
|
+ <!-- 按货品查询表格 -->
|
|
|
+ <product-production-table
|
|
|
+ v-else
|
|
|
:update-time="updateTime"
|
|
|
- :unassigned-orders="unassignedOrders"
|
|
|
- />
|
|
|
+ @progress-click="handleProgressClick"
|
|
|
+ @data-loaded="handleDataLoaded"
|
|
|
+ >
|
|
|
+ <template #query-buttons>
|
|
|
+ <el-button-group>
|
|
|
+ <el-button
|
|
|
+ type="primary"
|
|
|
+ size="small"
|
|
|
+ :plain="queryType !== 'order'"
|
|
|
+ @click="switchQueryType('order')"
|
|
|
+ >
|
|
|
+ 按订单查询
|
|
|
+ </el-button>
|
|
|
+ <el-button
|
|
|
+ type="primary"
|
|
|
+ size="small"
|
|
|
+ :plain="queryType !== 'product'"
|
|
|
+ @click="switchQueryType('product')"
|
|
|
+ >
|
|
|
+ 按货品查询
|
|
|
+ </el-button>
|
|
|
+ </el-button-group>
|
|
|
+ </template>
|
|
|
+ </product-production-table>
|
|
|
+
|
|
|
+ <!-- 发货状态表格 -->
|
|
|
+ <delivery-status-table />
|
|
|
|
|
|
<!-- 货品生产情况 -->
|
|
|
- <production-status-chart
|
|
|
+ <!-- <production-status-chart
|
|
|
:chart-data="chartData"
|
|
|
@filter-change="handleFilterChange"
|
|
|
- />
|
|
|
+ />-->
|
|
|
</div>
|
|
|
</template>
|
|
|
|
|
|
<script>
|
|
|
-import StatisticsPanel from './components/StatisticsPanel'
|
|
|
-import ProductionTable from './components/ProductionTable'
|
|
|
-import UnassignedDialog from './components/UnassignedDialog'
|
|
|
-import ProductionStatusChart from './components/ProductionStatusChart'
|
|
|
+import StatisticsPanel from "./components/StatisticsPanel";
|
|
|
+import OrderProductionTable from "./components/OrderProductionTable.vue";
|
|
|
+import ProductProductionTable from "./components/ProductProductionTable.vue";
|
|
|
+import DeliveryStatusTable from "./components/DeliveryStatusTable.vue";
|
|
|
+import { GetDataByName, GetDataByNames } from "@/api/common";
|
|
|
|
|
|
export default {
|
|
|
- name: 'ProductionSummary',
|
|
|
+ name: "ProductionSummary",
|
|
|
components: {
|
|
|
StatisticsPanel,
|
|
|
- ProductionTable,
|
|
|
- UnassignedDialog,
|
|
|
- ProductionStatusChart
|
|
|
+ OrderProductionTable,
|
|
|
+ ProductProductionTable,
|
|
|
+ DeliveryStatusTable,
|
|
|
},
|
|
|
data() {
|
|
|
return {
|
|
|
- currentTime: '',
|
|
|
- updateTime: '',
|
|
|
+ currentTime: "",
|
|
|
+ updateTime: "",
|
|
|
dialogVisible: false,
|
|
|
- tableData: [
|
|
|
- {
|
|
|
- orderNo: 'DD20240001',
|
|
|
- productName: '包装材料A型',
|
|
|
- status: '进行中',
|
|
|
- progress: 900,
|
|
|
- total: 4000,
|
|
|
- finished: 900,
|
|
|
- unfinished: 3100,
|
|
|
- rate: '22.5%',
|
|
|
- remainingDays: 7
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240002',
|
|
|
- productName: '塑料外壳B型',
|
|
|
- status: '已超期',
|
|
|
- progress: 1500,
|
|
|
- total: 2000,
|
|
|
- finished: 1500,
|
|
|
- unfinished: 500,
|
|
|
- rate: '75%',
|
|
|
- remainingDays: -3
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240003',
|
|
|
- productName: '金属零件C型',
|
|
|
- status: '未开始',
|
|
|
- progress: 0,
|
|
|
- total: 3000,
|
|
|
- finished: 0,
|
|
|
- unfinished: 3000,
|
|
|
- rate: '0%',
|
|
|
- remainingDays: 15
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240004',
|
|
|
- productName: '电子元件D型',
|
|
|
- status: '进行中',
|
|
|
- progress: 2000,
|
|
|
- total: 5000,
|
|
|
- finished: 2000,
|
|
|
- unfinished: 3000,
|
|
|
- rate: '40%',
|
|
|
- remainingDays: 10
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240005',
|
|
|
- productName: '橡胶配件E型',
|
|
|
- status: '已超期',
|
|
|
- progress: 800,
|
|
|
- total: 1000,
|
|
|
- finished: 800,
|
|
|
- unfinished: 200,
|
|
|
- rate: '80%',
|
|
|
- remainingDays: -5
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240006',
|
|
|
- productName: '五金配件F型',
|
|
|
- status: '进行中',
|
|
|
- progress: 1200,
|
|
|
- total: 3000,
|
|
|
- finished: 1200,
|
|
|
- unfinished: 1800,
|
|
|
- rate: '40%',
|
|
|
- remainingDays: 8
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240007',
|
|
|
- productName: '包装材料G型',
|
|
|
- status: '未开始',
|
|
|
- progress: 0,
|
|
|
- total: 2500,
|
|
|
- finished: 0,
|
|
|
- unfinished: 2500,
|
|
|
- rate: '0%',
|
|
|
- remainingDays: 12
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240008',
|
|
|
- productName: '电路板H型',
|
|
|
- status: '进行中',
|
|
|
- progress: 300,
|
|
|
- total: 1000,
|
|
|
- finished: 300,
|
|
|
- unfinished: 700,
|
|
|
- rate: '30%',
|
|
|
- remainingDays: 5
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240009',
|
|
|
- productName: '机械零件I型',
|
|
|
- status: '已超期',
|
|
|
- progress: 600,
|
|
|
- total: 800,
|
|
|
- finished: 600,
|
|
|
- unfinished: 200,
|
|
|
- rate: '75%',
|
|
|
- remainingDays: -2
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: 'DD20240010',
|
|
|
- productName: '塑料外壳J型',
|
|
|
- status: '进行中',
|
|
|
- progress: 1500,
|
|
|
- total: 3000,
|
|
|
- finished: 1500,
|
|
|
- unfinished: 1500,
|
|
|
- rate: '50%',
|
|
|
- remainingDays: 6
|
|
|
- }
|
|
|
- ],
|
|
|
- unassignedOrders: [
|
|
|
- {
|
|
|
- orderNo: '00000021',
|
|
|
- planNum: 2000,
|
|
|
- goods: '智能膜环',
|
|
|
- orderPerson: '孙XX',
|
|
|
- orderTime: '2024-10-01',
|
|
|
- priority: '一般',
|
|
|
- contract: 'HT00000001 泰三农场'
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: '00000022',
|
|
|
- planNum: 2000,
|
|
|
- goods: '智能喷淋',
|
|
|
- orderPerson: '孙XX',
|
|
|
- orderTime: '2024-10-12',
|
|
|
- priority: '一般',
|
|
|
- contract: 'HT00000002 和祥牧场'
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: '00000023',
|
|
|
- planNum: 2000,
|
|
|
- goods: '佳沃评分',
|
|
|
- orderPerson: '孙XX',
|
|
|
- orderTime: '2024-10-15',
|
|
|
- priority: '一般',
|
|
|
- contract: 'HT00000003 五河牧场'
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: '00000024',
|
|
|
- planNum: 2000,
|
|
|
- goods: '大屏',
|
|
|
- orderPerson: '孙XX',
|
|
|
- orderTime: '2024-10-18',
|
|
|
- priority: '一般',
|
|
|
- contract: 'HT00000004 云养牛'
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: '00000025',
|
|
|
- planNum: 2000,
|
|
|
- goods: '耳标',
|
|
|
- orderPerson: '孙XX',
|
|
|
- orderTime: '2024-10-20',
|
|
|
- priority: '高',
|
|
|
- contract: 'HT00000005 宝塔牧场\nHT00000006 马鞍山牧场'
|
|
|
- },
|
|
|
- {
|
|
|
- orderNo: '00000026',
|
|
|
- planNum: 2000,
|
|
|
- goods: '智能称重',
|
|
|
- orderPerson: '孙XX',
|
|
|
- orderTime: '2024-10-30',
|
|
|
- priority: '一般',
|
|
|
- contract: 'HT00000007 海丰牧场'
|
|
|
- }
|
|
|
- ],
|
|
|
- chartData: {
|
|
|
- labels: ['智能喷淋', '智能膜环', '佳沃评分', '精准阉割', '奶牛称重', '车载控制器', '大屏', '耳标'],
|
|
|
- datasets: [
|
|
|
- {
|
|
|
- label: '计划量',
|
|
|
- data: [600, 2000, 700, 970, 680, 840, 500, 1600],
|
|
|
- backgroundColor: '#409EFF',
|
|
|
- order: 2
|
|
|
- },
|
|
|
- {
|
|
|
- label: '完成量',
|
|
|
- data: [550, 1590, 500, 800, 630, 800, 500, 1600],
|
|
|
- backgroundColor: '#FF9F43',
|
|
|
- order: 2
|
|
|
- },
|
|
|
- {
|
|
|
- label: '完成率',
|
|
|
- data: [91.7, 79.5, 71.4, 82.5, 92.6, 95.2, 100, 100],
|
|
|
- type: 'line',
|
|
|
- borderColor: '#67C23A',
|
|
|
- borderWidth: 2,
|
|
|
- fill: false,
|
|
|
- yAxisID: 'percentage',
|
|
|
- order: 1
|
|
|
- }
|
|
|
- ]
|
|
|
- }
|
|
|
- }
|
|
|
+ statisticsPanelData: [],
|
|
|
+ queryType: "order", // 默认按订单查询
|
|
|
+ autoPlayTimer: null, // 自动轮播定时器
|
|
|
+ manualTimer: null, // 手动模式定时器
|
|
|
+ isAutoPlay: true, // 是否为自动轮播模式
|
|
|
+ };
|
|
|
},
|
|
|
created() {
|
|
|
- this.updateCurrentTime()
|
|
|
- setInterval(this.updateCurrentTime, 1000)
|
|
|
- this.getUpdateTime()
|
|
|
- setInterval(this.getUpdateTime, 60000)
|
|
|
- this.getTableData()
|
|
|
+ this.updateCurrentTime();
|
|
|
+ setInterval(this.updateCurrentTime, 1000);
|
|
|
+ this.getTableData();
|
|
|
+ this.startAutoPlay(); // 启动自动轮播
|
|
|
+ },
|
|
|
+ beforeDestroy() {
|
|
|
+ // 组件销毁前清除定时器
|
|
|
+ this.clearTimers();
|
|
|
},
|
|
|
methods: {
|
|
|
updateCurrentTime() {
|
|
|
- const now = new Date()
|
|
|
- const year = now.getFullYear()
|
|
|
- const month = (now.getMonth() + 1).toString().padStart(2, '0')
|
|
|
- const day = now.getDate().toString().padStart(2, '0')
|
|
|
- const hours = now.getHours().toString().padStart(2, '0')
|
|
|
- const minutes = now.getMinutes().toString().padStart(2, '0')
|
|
|
- const seconds = now.getSeconds().toString().padStart(2, '0')
|
|
|
- this.currentTime = `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`
|
|
|
+ const now = new Date();
|
|
|
+ const year = now.getFullYear();
|
|
|
+ const month = (now.getMonth() + 1).toString().padStart(2, "0");
|
|
|
+ const day = now.getDate().toString().padStart(2, "0");
|
|
|
+ const hours = now.getHours().toString().padStart(2, "0");
|
|
|
+ const minutes = now.getMinutes().toString().padStart(2, "0");
|
|
|
+ const seconds = now.getSeconds().toString().padStart(2, "0");
|
|
|
+ this.currentTime = `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
|
|
|
},
|
|
|
- async getUpdateTime() {
|
|
|
- try {
|
|
|
- const now = new Date()
|
|
|
- const month = (now.getMonth() + 1).toString().padStart(2, '0')
|
|
|
- const day = now.getDate().toString().padStart(2, '0')
|
|
|
- const hours = now.getHours().toString().padStart(2, '0')
|
|
|
- const minutes = now.getMinutes().toString().padStart(2, '0')
|
|
|
- this.updateTime = `${month}/${day} ${hours}:${minutes}更新`
|
|
|
- } catch (error) {
|
|
|
- console.error('获取更新时间失败:', error)
|
|
|
- }
|
|
|
+ handleDataLoaded() {
|
|
|
+ // 数据加载完成后更新时间
|
|
|
+ const now = new Date();
|
|
|
+ const month = (now.getMonth() + 1).toString().padStart(2, "0");
|
|
|
+ const day = now.getDate().toString().padStart(2, "0");
|
|
|
+ const hours = now.getHours().toString().padStart(2, "0");
|
|
|
+ const minutes = now.getMinutes().toString().padStart(2, "0");
|
|
|
+ const seconds = now.getSeconds().toString().padStart(2, "0");
|
|
|
+ this.updateTime = `${month}/${day} ${hours}:${minutes}:${seconds}更新`;
|
|
|
},
|
|
|
handleUnassignedOrders() {
|
|
|
- this.getUnassignedOrders()
|
|
|
- this.dialogVisible = true
|
|
|
+ this.getUnassignedOrders();
|
|
|
+ this.dialogVisible = true;
|
|
|
},
|
|
|
async getUnassignedOrders() {
|
|
|
try {
|
|
|
- console.log('获取未接单数据')
|
|
|
+ console.log("获取未接单数据");
|
|
|
} catch (error) {
|
|
|
- console.error('获取未接单数据失败:', error)
|
|
|
- this.$message.error('获取未接单数据失败')
|
|
|
+ console.error("获取未接单数据失败:", error);
|
|
|
+ this.$message.error("获取未接单数据失败");
|
|
|
}
|
|
|
},
|
|
|
handleProgressClick(row) {
|
|
|
- console.log('进度条被点击:', row)
|
|
|
+ console.log("进度条被点击:", row);
|
|
|
},
|
|
|
- async getTableData() {
|
|
|
- try {
|
|
|
- console.log('获取生产概况表格数据')
|
|
|
- } catch (error) {
|
|
|
- console.error('获取表格数据失败:', error)
|
|
|
- this.$message.error('获取表格数据失败')
|
|
|
- }
|
|
|
+ getTableData() {
|
|
|
+ const send_select_list = [
|
|
|
+ {
|
|
|
+ name: "getStatisticsPanelNums",
|
|
|
+ offset: 0,
|
|
|
+ pagecount: 0,
|
|
|
+ parammaps: {},
|
|
|
+ },
|
|
|
+ {
|
|
|
+ name: "getDictListSelect",
|
|
|
+ offset: 0,
|
|
|
+ pagecount: 0,
|
|
|
+ parammaps: {
|
|
|
+ pid: "79",
|
|
|
+ },
|
|
|
+ },
|
|
|
+ {
|
|
|
+ name: "getInstallationOrderToDoNum",
|
|
|
+ offset: 0,
|
|
|
+ pagecount: 0,
|
|
|
+ parammaps: {},
|
|
|
+ },
|
|
|
+ ];
|
|
|
+
|
|
|
+ GetDataByNames(send_select_list)
|
|
|
+ .then((response) => {
|
|
|
+ // 处理服务人员数据
|
|
|
+ this.statisticsPanelData =
|
|
|
+ response.data.getStatisticsPanelNums.list || [];
|
|
|
+ })
|
|
|
+ .catch((error) => {
|
|
|
+ console.error("获取下拉框数据失败:", error);
|
|
|
+ });
|
|
|
},
|
|
|
handleFilterChange({ month, product }) {
|
|
|
- console.log('筛选条件变更:', month, product)
|
|
|
- this.getChartData(month, product)
|
|
|
+ console.log("筛选条件变更:", month, product);
|
|
|
+ this.getChartData(month, product);
|
|
|
},
|
|
|
async getChartData(month, product) {
|
|
|
try {
|
|
|
- console.log('获取图表数据:', month, product)
|
|
|
+ console.log("获取图表数据:", month, product);
|
|
|
} catch (error) {
|
|
|
- console.error('获取图表数据失败:', error)
|
|
|
- this.$message.error('获取图表数据失败')
|
|
|
+ console.error("获取图表数据失败:", error);
|
|
|
+ this.$message.error("获取图表数据失败");
|
|
|
}
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
+ },
|
|
|
+ // 切换查询类型
|
|
|
+ switchQueryType(type) {
|
|
|
+ if (this.queryType === type) return;
|
|
|
+
|
|
|
+ this.queryType = type;
|
|
|
+ this.clearTimers(); // 清除现有定时器
|
|
|
+ this.isAutoPlay = false;
|
|
|
+
|
|
|
+ // 提示进入手动模式
|
|
|
+ this.$message({
|
|
|
+ message: "已切换为手动模式,5分钟后恢复自动轮播",
|
|
|
+ type: "info",
|
|
|
+ duration: 3000,
|
|
|
+ });
|
|
|
+
|
|
|
+ // 设置5分钟后恢复自动轮播
|
|
|
+ this.manualTimer = setTimeout(() => {
|
|
|
+ this.startAutoPlay();
|
|
|
+ // 提示恢复自动模式
|
|
|
+ this.$message({
|
|
|
+ message: "已恢复自动轮播模式",
|
|
|
+ type: "success",
|
|
|
+ duration: 3000,
|
|
|
+ });
|
|
|
+ }, 5 * 60 * 1000);
|
|
|
+ },
|
|
|
+
|
|
|
+ // 启动自动轮播
|
|
|
+ startAutoPlay() {
|
|
|
+ this.clearTimers(); // 先清除现有定时器
|
|
|
+ this.isAutoPlay = true;
|
|
|
+
|
|
|
+ if (this.isAutoPlay) {
|
|
|
+ // 首次启动时提示
|
|
|
+ this.$message({
|
|
|
+ message: "自动轮播模式:每1分钟自动切换视图",
|
|
|
+ type: "success",
|
|
|
+ duration: 3000,
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ // 每分钟切换一次
|
|
|
+ this.autoPlayTimer = setInterval(() => {
|
|
|
+ this.queryType = this.queryType === "order" ? "product" : "order";
|
|
|
+ }, 60 * 1000);
|
|
|
+ },
|
|
|
+
|
|
|
+ // 清除所有定时器
|
|
|
+ clearTimers() {
|
|
|
+ if (this.autoPlayTimer) {
|
|
|
+ clearInterval(this.autoPlayTimer);
|
|
|
+ this.autoPlayTimer = null;
|
|
|
+ }
|
|
|
+ if (this.manualTimer) {
|
|
|
+ clearTimeout(this.manualTimer);
|
|
|
+ this.manualTimer = null;
|
|
|
+ }
|
|
|
+ },
|
|
|
+ },
|
|
|
+ watch: {
|
|
|
+ queryType(newVal) {
|
|
|
+ // 视图切换时的提示
|
|
|
+ if (this.isAutoPlay) {
|
|
|
+ this.$message({
|
|
|
+ message: `自动切换至${newVal === "order" ? "按订单" : "按货品"}视图`,
|
|
|
+ type: "info",
|
|
|
+ duration: 2000,
|
|
|
+ });
|
|
|
+ }
|
|
|
+ },
|
|
|
+ },
|
|
|
+};
|
|
|
</script>
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
@@ -335,16 +289,16 @@ export default {
|
|
|
color: #1f2d3d;
|
|
|
position: relative;
|
|
|
padding-left: 16px;
|
|
|
-
|
|
|
+
|
|
|
&::before {
|
|
|
- content: '';
|
|
|
+ content: "";
|
|
|
position: absolute;
|
|
|
left: 0;
|
|
|
top: 50%;
|
|
|
transform: translateY(-50%);
|
|
|
width: 4px;
|
|
|
height: 22px;
|
|
|
- background: #409EFF;
|
|
|
+ background: #409eff;
|
|
|
border-radius: 4px;
|
|
|
}
|
|
|
}
|
|
@@ -354,4 +308,67 @@ export default {
|
|
|
font-size: 14px;
|
|
|
}
|
|
|
}
|
|
|
+
|
|
|
+// 添加按钮组样式
|
|
|
+:deep(.el-button-group) {
|
|
|
+ .el-button {
|
|
|
+ padding: 8px 16px;
|
|
|
+ font-size: 13px;
|
|
|
+ border-radius: 4px;
|
|
|
+ transition: all 0.3s ease;
|
|
|
+
|
|
|
+ &:first-child {
|
|
|
+ border-top-right-radius: 0;
|
|
|
+ border-bottom-right-radius: 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:last-child {
|
|
|
+ border-top-left-radius: 0;
|
|
|
+ border-bottom-left-radius: 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ &:not(.is-plain) {
|
|
|
+ background: linear-gradient(135deg, #409eff 0%, #3a8ee6 100%);
|
|
|
+ border-color: #3a8ee6;
|
|
|
+ box-shadow: 0 2px 6px rgba(64, 158, 255, 0.2);
|
|
|
+
|
|
|
+ &:hover {
|
|
|
+ background: linear-gradient(135deg, #66b1ff 0%, #409eff 100%);
|
|
|
+ border-color: #409eff;
|
|
|
+ box-shadow: 0 4px 12px rgba(64, 158, 255, 0.3);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ &.is-plain {
|
|
|
+ color: #409eff;
|
|
|
+ background: #ecf5ff;
|
|
|
+ border-color: #b3d8ff;
|
|
|
+
|
|
|
+ &:hover {
|
|
|
+ color: #fff;
|
|
|
+ background: linear-gradient(135deg, #66b1ff 0%, #409eff 100%);
|
|
|
+ border-color: #409eff;
|
|
|
+ box-shadow: 0 4px 12px rgba(64, 158, 255, 0.3);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+// 添加表格切换动画
|
|
|
+.app-container {
|
|
|
+ .el-card {
|
|
|
+ animation: fadeInScale 0.3s ease-out;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+@keyframes fadeInScale {
|
|
|
+ from {
|
|
|
+ opacity: 0;
|
|
|
+ transform: scale(0.98);
|
|
|
+ }
|
|
|
+ to {
|
|
|
+ opacity: 1;
|
|
|
+ transform: scale(1);
|
|
|
+ }
|
|
|
+}
|
|
|
</style>
|