251 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			251 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|     <div class="chart-container">
 | |
|         <div class="chart-header">
 | |
|             <h2>效率分析</h2>
 | |
|             <div class="chart-actions">
 | |
|                 <button :class="{ active: timeRange === 'hour' }" @click="changeTimeRange('hour')">小时</button>
 | |
|                 <button :class="{ active: timeRange === 'day' }" @click="changeTimeRange('day')">每天</button>
 | |
|                 <button :class="{ active: timeRange === 'week' }" @click="changeTimeRange('week')">每周</button>
 | |
|             </div>
 | |
|         </div>
 | |
|         <!--图表内容区域 -->
 | |
|         <div ref="chartRef" class="chart-content"></div>
 | |
|     </div>
 | |
| </template>
 | |
| 
 | |
| <script setup>
 | |
| import { ref, onMounted } from 'vue';
 | |
| import * as echarts from 'echarts';
 | |
| 
 | |
| // 图表DOM引用
 | |
| const chartRef = ref(null);
 | |
| // 图表实例
 | |
| let chartInstance = null;
 | |
| 
 | |
| // 当前时间范围状态
 | |
| const timeRange = ref('hour'); // 默认显示小时数据
 | |
| 
 | |
| // 定义props
 | |
| const props = defineProps({
 | |
|   chartData: {
 | |
|     type: Object,
 | |
|     default: () => ({
 | |
|       hour: {
 | |
|         xAxis: [],
 | |
|         efficiencyData: []
 | |
|       },
 | |
|       day: {
 | |
|         xAxis: ['周一', '周二', '周三', '周四', '周五', '周六', '周日'],
 | |
|         efficiencyData: []
 | |
|       },
 | |
|       week: {
 | |
|         xAxis: ['第1周', '第2周', '第3周', '第4周', '第5周', '第6周'],
 | |
|         efficiencyData: []
 | |
|       }
 | |
|     })
 | |
|   }
 | |
| });
 | |
| 
 | |
| // 切换时间范围的函数
 | |
| const changeTimeRange = (range) => {
 | |
|     timeRange.value = range;
 | |
|     updateChartData();
 | |
| };
 | |
| 
 | |
| // 更新图表数据的函数
 | |
| const updateChartData = () => {
 | |
|     if (chartInstance) {
 | |
|         const data = props.chartData[timeRange.value];
 | |
|         chartInstance.setOption({
 | |
|             xAxis: {
 | |
|                 data: data.xAxis
 | |
|             },
 | |
|             series: [
 | |
|                 {
 | |
|                     data: data.efficiencyData
 | |
|                 }
 | |
|             ]
 | |
|         });
 | |
|     }
 | |
| };
 | |
| 
 | |
| // 初始化图表
 | |
| const initChart = () => {
 | |
|     if (chartRef.value && !chartInstance) {
 | |
|         chartInstance = echarts.init(chartRef.value);
 | |
|     }
 | |
| 
 | |
|     const data = props.chartData[timeRange.value];
 | |
| 
 | |
|     const option = {
 | |
|         title: {
 | |
|             text: '',
 | |
|             show: false
 | |
|         },
 | |
|         tooltip: {
 | |
|             trigger: 'axis',
 | |
|             backgroundColor: 'rgba(33,56,77,1)',
 | |
|             borderColor: 'rgba(33,56,77,1)',
 | |
|             textStyle: {
 | |
|                 color: '#fff',
 | |
|                 fontSize: 14
 | |
|             }
 | |
|         },
 | |
|         grid: {
 | |
|             top: '20%',
 | |
|             right: '4%',
 | |
|             bottom: '4%',
 | |
|             left: '6%',
 | |
|             containLabel: true
 | |
|         },
 | |
|         xAxis: {
 | |
|             data: data.xAxis,
 | |
|             type: 'category',
 | |
|             axisTick: {
 | |
|                 show: false
 | |
|             }
 | |
|         },
 | |
|         yAxis: {
 | |
|             type: 'value',
 | |
|             max: 100,
 | |
|             interval: 10,
 | |
|             axisTick: {
 | |
|                 show: false
 | |
|             },
 | |
|             splitLine: {
 | |
|                 show: true,
 | |
|                 lineStyle: {
 | |
|                     color: '#f0f0f0',
 | |
|                     type: 'dashed'
 | |
|                 }
 | |
|             }
 | |
|         },
 | |
|         series: [
 | |
|             {
 | |
|                 name: '转换效率(%)',
 | |
|                 type: 'bar',
 | |
|                 data: data.efficiencyData,
 | |
|                 showBackground: true,
 | |
|                 itemStyle: {
 | |
|                     color: 'rgba(63, 140, 255, 1)',
 | |
|                     borderRadius: [6, 6, 0, 0] // 为顶部两个角设置圆角
 | |
|                 },
 | |
|                 barWidth: 50,
 | |
|                 backgroundStyle: {
 | |
|                     color: 'rgba(24, 179, 245, 0.1)'
 | |
|                 },
 | |
|             }
 | |
|         ]
 | |
|     };
 | |
| 
 | |
|     chartInstance.setOption(option);
 | |
| };
 | |
| 
 | |
| // 响应窗口大小变化
 | |
| const handleResize = () => {
 | |
|     if (chartInstance) {
 | |
|         chartInstance.resize();
 | |
|     }
 | |
| };
 | |
| 
 | |
| // 生命周期钩子
 | |
| onMounted(() => {
 | |
|     initChart();
 | |
|     window.addEventListener('resize', handleResize);
 | |
| 
 | |
|     // 清理函数
 | |
|     return () => {
 | |
|         window.removeEventListener('resize', handleResize);
 | |
|         if (chartInstance) {
 | |
|             chartInstance.dispose();
 | |
|             chartInstance = null;
 | |
|         }
 | |
|     };
 | |
| });
 | |
| 
 | |
| </script>
 | |
| 
 | |
| <style scoped>
 | |
| .chart-container {
 | |
|     background-color: #fff;
 | |
|     border-radius: 8px;
 | |
|     overflow: hidden;
 | |
|     height: 400px;
 | |
|     width: 100%;
 | |
|     padding: 10px;
 | |
|     box-sizing: border-box;
 | |
| }
 | |
| 
 | |
| .chart-header {
 | |
|     display: flex;
 | |
|     justify-content: space-between;
 | |
|     align-items: center;
 | |
|     padding: 15px 20px;
 | |
|     border-bottom: 1px solid #f0f0f0;
 | |
| }
 | |
| 
 | |
| .chart-header h2 {
 | |
|     font-size: 16px;
 | |
|     font-weight: 600;
 | |
|     color: #333;
 | |
|     margin: 0;
 | |
| }
 | |
| 
 | |
| .chart-actions button {
 | |
|     background: none;
 | |
|     border: 1px solid #e0e0e0;
 | |
|     padding: 5px 12px;
 | |
|     border-radius: 4px;
 | |
|     margin-left: 8px;
 | |
|     cursor: pointer;
 | |
|     font-size: 12px;
 | |
|     transition: all 0.2s ease;
 | |
| }
 | |
| 
 | |
| .chart-actions button.active {
 | |
|     background-color: #1890ff;
 | |
|     color: white;
 | |
|     border-color: #1890ff;
 | |
| }
 | |
| 
 | |
| .chart-content {
 | |
|     width: 100%;
 | |
|     height: calc(100% - 70px);
 | |
|     padding: 10px;
 | |
|     box-sizing: border-box;
 | |
| }
 | |
| 
 | |
| @media (max-width: 768px) {
 | |
|     .chart-container {
 | |
|         height: 350px;
 | |
|     }
 | |
| }
 | |
| 
 | |
| @media (max-width: 480px) {
 | |
|     .chart-container {
 | |
|         height: 300px;
 | |
|     }
 | |
| 
 | |
|     .chart-header {
 | |
|         flex-direction: column;
 | |
|         align-items: flex-start;
 | |
|         gap: 10px;
 | |
|     }
 | |
| 
 | |
|     .chart-actions {
 | |
|         width: 100%;
 | |
|         display: flex;
 | |
|         justify-content: space-between;
 | |
|     }
 | |
| 
 | |
|     .chart-actions button {
 | |
|         margin: 0;
 | |
|         flex: 1;
 | |
|         margin-right: 5px;
 | |
|     }
 | |
| 
 | |
|     .chart-actions button:last-child {
 | |
|         margin-right: 0;
 | |
|     }
 | |
| }
 | |
| </style> | 
