增加了分析界面

pull/409/head
math-zhuxy 2025-07-03 14:44:18 +08:00
parent 6a7019ec1a
commit 4f89950fc1
6 changed files with 5198 additions and 1771 deletions

22
components.d.ts vendored
View File

@ -10,16 +10,10 @@ declare module '@vue/runtime-core' {
Countup: typeof import('./src/components/countup.vue')['default']
ElAvatar: typeof import('element-plus/es')['ElAvatar']
ElButton: typeof import('element-plus/es')['ElButton']
ElCalendar: typeof import('element-plus/es')['ElCalendar']
ElCard: typeof import('element-plus/es')['ElCard']
ElCarousel: typeof import('element-plus/es')['ElCarousel']
ElCarouselItem: typeof import('element-plus/es')['ElCarouselItem']
ElCascader: typeof import('element-plus/es')['ElCascader']
ElCheckbox: typeof import('element-plus/es')['ElCheckbox']
ElCheckboxGroup: typeof import('element-plus/es')['ElCheckboxGroup']
ElCol: typeof import('element-plus/es')['ElCol']
ElColorPicker: typeof import('element-plus/es')['ElColorPicker']
ElCountdown: typeof import('element-plus/es')['ElCountdown']
ElDatePicker: typeof import('element-plus/es')['ElDatePicker']
ElDescriptions: typeof import('element-plus/es')['ElDescriptions']
ElDescriptionsItem: typeof import('element-plus/es')['ElDescriptionsItem']
@ -31,7 +25,6 @@ declare module '@vue/runtime-core' {
ElForm: typeof import('element-plus/es')['ElForm']
ElFormItem: typeof import('element-plus/es')['ElFormItem']
ElIcon: typeof import('element-plus/es')['ElIcon']
ElImage: typeof import('element-plus/es')['ElImage']
ElInput: typeof import('element-plus/es')['ElInput']
ElInputNumber: typeof import('element-plus/es')['ElInputNumber']
ElLink: typeof import('element-plus/es')['ElLink']
@ -40,18 +33,8 @@ declare module '@vue/runtime-core' {
ElOption: typeof import('element-plus/es')['ElOption']
ElPagination: typeof import('element-plus/es')['ElPagination']
ElProgress: typeof import('element-plus/es')['ElProgress']
ElRadio: typeof import('element-plus/es')['ElRadio']
ElRadioButton: typeof import('element-plus/es')['ElRadioButton']
ElRadioGroup: typeof import('element-plus/es')['ElRadioGroup']
ElRate: typeof import('element-plus/es')['ElRate']
ElResult: typeof import('element-plus/es')['ElResult']
ElRow: typeof import('element-plus/es')['ElRow']
ElSelect: typeof import('element-plus/es')['ElSelect']
ElSlider: typeof import('element-plus/es')['ElSlider']
ElSpace: typeof import('element-plus/es')['ElSpace']
ElStatistic: typeof import('element-plus/es')['ElStatistic']
ElStep: typeof import('element-plus/es')['ElStep']
ElSteps: typeof import('element-plus/es')['ElSteps']
ElSubMenu: typeof import('element-plus/es')['ElSubMenu']
ElSwitch: typeof import('element-plus/es')['ElSwitch']
ElTable: typeof import('element-plus/es')['ElTable']
@ -61,13 +44,8 @@ declare module '@vue/runtime-core' {
ElTag: typeof import('element-plus/es')['ElTag']
ElTimeline: typeof import('element-plus/es')['ElTimeline']
ElTimelineItem: typeof import('element-plus/es')['ElTimelineItem']
ElTimePicker: typeof import('element-plus/es')['ElTimePicker']
ElTooltip: typeof import('element-plus/es')['ElTooltip']
ElTour: typeof import('element-plus/es')['ElTour']
ElTourStep: typeof import('element-plus/es')['ElTourStep']
ElTransfer: typeof import('element-plus/es')['ElTransfer']
ElUpload: typeof import('element-plus/es')['ElUpload']
ElWatermark: typeof import('element-plus/es')['ElWatermark']
Header: typeof import('./src/components/header.vue')['default']
RouterLink: typeof import('vue-router')['RouterLink']
RouterView: typeof import('vue-router')['RouterView']

2846
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

View File

@ -12,6 +12,7 @@
<Fold />
</el-icon>
</div>
<div id="analysis-page" @click="handleAnalysisPage"></div>
</div>
<div class="header-right">
<div class="header-user-con">
@ -94,6 +95,10 @@ const handleCommand = (command: string) => {
}
};
const handleAnalysisPage = () => {
router.push('/analysis');
};
const setFullScreen = () => {
if (document.fullscreenElement) {
document.exitFullscreen();
@ -176,6 +181,15 @@ const setFullScreen = () => {
font-size: 20px;
}
#analysis-page {
margin-left: 20px;
cursor: pointer;
}
#analysis-page:hover {
color: rgb(190, 190, 255);
}
.btn-bell-badge {
position: absolute;
right: 4px;

View File

@ -1,293 +1,362 @@
import { createRouter, createWebHashHistory, RouteRecordRaw } from 'vue-router';
import { usePermissStore } from '../store/permiss';
import Home from '../views/home.vue';
import NProgress from 'nprogress';
import 'nprogress/nprogress.css';
import { createRouter, createWebHashHistory, RouteRecordRaw } from "vue-router";
import { usePermissStore } from "../store/permiss";
import Home from "../views/home.vue";
import NProgress from "nprogress";
import "nprogress/nprogress.css";
import { pa } from "element-plus/es/locale";
const routes: RouteRecordRaw[] = [
{
path: '/',
redirect: '/dashboard',
},
{
path: '/',
name: 'Home',
component: Home,
children: [
{
path: '/dashboard',
name: 'dashboard',
meta: {
title: '系统首页',
noAuth: true,
},
component: () => import(/* webpackChunkName: "dashboard" */ '../views/dashboard.vue'),
},
{
path: '/system-user',
name: 'system-user',
meta: {
title: '用户管理',
permiss: '11',
},
component: () => import(/* webpackChunkName: "system-user" */ '../views/system/user.vue'),
},
{
path: '/system-role',
name: 'system-role',
meta: {
title: '角色管理',
permiss: '12',
},
component: () => import(/* webpackChunkName: "system-role" */ '../views/system/role.vue'),
},
{
path: '/system-menu',
name: 'system-menu',
meta: {
title: '菜单管理',
permiss: '13',
},
component: () => import(/* webpackChunkName: "system-menu" */ '../views/system/menu.vue'),
},
{
path: '/table',
name: 'basetable',
meta: {
title: '基础表格',
permiss: '31',
},
component: () => import(/* webpackChunkName: "table" */ '../views/table/basetable.vue'),
},
{
path: '/table-editor',
name: 'table-editor',
meta: {
title: '可编辑表格',
permiss: '32',
},
component: () => import(/* webpackChunkName: "table-editor" */ '../views/table/table-editor.vue'),
},
{
path: '/schart',
name: 'schart',
meta: {
title: 'schart图表',
permiss: '41',
},
component: () => import(/* webpackChunkName: "schart" */ '../views/chart/schart.vue'),
},
{
path: '/echarts',
name: 'echarts',
meta: {
title: 'echarts图表',
permiss: '42',
},
component: () => import(/* webpackChunkName: "echarts" */ '../views/chart/echarts.vue'),
},
{
path: "/",
redirect: "/dashboard",
},
{
path: "/",
name: "Home",
component: Home,
children: [
{
path: "/dashboard",
name: "dashboard",
meta: {
title: "系统首页",
noAuth: true,
},
component: () =>
import(/* webpackChunkName: "dashboard" */ "../views/dashboard.vue"),
},
{
path: "/system-user",
name: "system-user",
meta: {
title: "用户管理",
permiss: "11",
},
component: () =>
import(
/* webpackChunkName: "system-user" */ "../views/system/user.vue"
),
},
{
path: "/system-role",
name: "system-role",
meta: {
title: "角色管理",
permiss: "12",
},
component: () =>
import(
/* webpackChunkName: "system-role" */ "../views/system/role.vue"
),
},
{
path: "/system-menu",
name: "system-menu",
meta: {
title: "菜单管理",
permiss: "13",
},
component: () =>
import(
/* webpackChunkName: "system-menu" */ "../views/system/menu.vue"
),
},
{
path: "/table",
name: "basetable",
meta: {
title: "基础表格",
permiss: "31",
},
component: () =>
import(
/* webpackChunkName: "table" */ "../views/table/basetable.vue"
),
},
{
path: "/table-editor",
name: "table-editor",
meta: {
title: "可编辑表格",
permiss: "32",
},
component: () =>
import(
/* webpackChunkName: "table-editor" */ "../views/table/table-editor.vue"
),
},
{
path: "/schart",
name: "schart",
meta: {
title: "schart图表",
permiss: "41",
},
component: () =>
import(/* webpackChunkName: "schart" */ "../views/chart/schart.vue"),
},
{
path: "/echarts",
name: "echarts",
meta: {
title: "echarts图表",
permiss: "42",
},
component: () =>
import(
/* webpackChunkName: "echarts" */ "../views/chart/echarts.vue"
),
},
{
path: '/icon',
name: 'icon',
meta: {
title: '图标',
permiss: '5',
},
component: () => import(/* webpackChunkName: "icon" */ '../views/pages/icon.vue'),
},
{
path: '/ucenter',
name: 'ucenter',
meta: {
title: '个人中心',
},
component: () => import(/* webpackChunkName: "ucenter" */ '../views/pages/ucenter.vue'),
},
{
path: '/editor',
name: 'editor',
meta: {
title: '富文本编辑器',
permiss: '291',
},
component: () => import(/* webpackChunkName: "editor" */ '../views/pages/editor.vue'),
},
{
path: '/markdown',
name: 'markdown',
meta: {
title: 'markdown编辑器',
permiss: '292',
},
component: () => import(/* webpackChunkName: "markdown" */ '../views/pages/markdown.vue'),
},
{
path: '/export',
name: 'export',
meta: {
title: '导出Excel',
permiss: '34',
},
component: () => import(/* webpackChunkName: "export" */ '../views/table/export.vue'),
},
{
path: '/import',
name: 'import',
meta: {
title: '导入Excel',
permiss: '33',
},
component: () => import(/* webpackChunkName: "import" */ '../views/table/import.vue'),
},
{
path: '/theme',
name: 'theme',
meta: {
title: '主题设置',
permiss: '7',
},
component: () => import(/* webpackChunkName: "theme" */ '../views/pages/theme.vue'),
},
{
path: '/calendar',
name: 'calendar',
meta: {
title: '日历',
permiss: '24',
},
component: () => import(/* webpackChunkName: "calendar" */ '../views/element/calendar.vue'),
},
{
path: '/watermark',
name: 'watermark',
meta: {
title: '水印',
permiss: '25',
},
component: () => import(/* webpackChunkName: "watermark" */ '../views/element/watermark.vue'),
},
{
path: '/carousel',
name: 'carousel',
meta: {
title: '走马灯',
permiss: '23',
},
component: () => import(/* webpackChunkName: "carousel" */ '../views/element/carousel.vue'),
},
{
path: '/tour',
name: 'tour',
meta: {
title: '分步引导',
permiss: '26',
},
component: () => import(/* webpackChunkName: "tour" */ '../views/element/tour.vue'),
},
{
path: '/steps',
name: 'steps',
meta: {
title: '步骤条',
permiss: '27',
},
component: () => import(/* webpackChunkName: "steps" */ '../views/element/steps.vue'),
},
{
path: '/form',
name: 'forms',
meta: {
title: '表单',
permiss: '21',
},
component: () => import(/* webpackChunkName: "form" */ '../views/element/form.vue'),
},
{
path: '/upload',
name: 'upload',
meta: {
title: '上传',
permiss: '22',
},
component: () => import(/* webpackChunkName: "upload" */ '../views/element/upload.vue'),
},
{
path: '/statistic',
name: 'statistic',
meta: {
title: '统计',
permiss: '28',
},
component: () => import(/* webpackChunkName: "statistic" */ '../views/element/statistic.vue'),
},
],
},
{
path: '/login',
{
path: "/icon",
name: "icon",
meta: {
title: '登录',
noAuth: true,
title: "图标",
permiss: "5",
},
component: () => import(/* webpackChunkName: "login" */ '../views/pages/login.vue'),
},
{
path: '/register',
component: () =>
import(/* webpackChunkName: "icon" */ "../views/pages/icon.vue"),
},
{
path: "/ucenter",
name: "ucenter",
meta: {
title: '注册',
noAuth: true,
title: "个人中心",
},
component: () => import(/* webpackChunkName: "register" */ '../views/pages/register.vue'),
},
{
path: '/reset-pwd',
component: () =>
import(
/* webpackChunkName: "ucenter" */ "../views/pages/ucenter.vue"
),
},
{
path: "/editor",
name: "editor",
meta: {
title: '重置密码',
noAuth: true,
title: "富文本编辑器",
permiss: "291",
},
component: () => import(/* webpackChunkName: "reset-pwd" */ '../views/pages/reset-pwd.vue'),
},
{
path: '/403',
component: () =>
import(/* webpackChunkName: "editor" */ "../views/pages/editor.vue"),
},
{
path: "/markdown",
name: "markdown",
meta: {
title: '没有权限',
noAuth: true,
title: "markdown编辑器",
permiss: "292",
},
component: () => import(/* webpackChunkName: "403" */ '../views/pages/403.vue'),
},
{
path: '/404',
component: () =>
import(
/* webpackChunkName: "markdown" */ "../views/pages/markdown.vue"
),
},
{
path: "/export",
name: "export",
meta: {
title: '找不到页面',
noAuth: true,
title: "导出Excel",
permiss: "34",
},
component: () => import(/* webpackChunkName: "404" */ '../views/pages/404.vue'),
component: () =>
import(/* webpackChunkName: "export" */ "../views/table/export.vue"),
},
{
path: "/import",
name: "import",
meta: {
title: "导入Excel",
permiss: "33",
},
component: () =>
import(/* webpackChunkName: "import" */ "../views/table/import.vue"),
},
{
path: "/theme",
name: "theme",
meta: {
title: "主题设置",
permiss: "7",
},
component: () =>
import(/* webpackChunkName: "theme" */ "../views/pages/theme.vue"),
},
{
path: "/calendar",
name: "calendar",
meta: {
title: "日历",
permiss: "24",
},
component: () =>
import(
/* webpackChunkName: "calendar" */ "../views/element/calendar.vue"
),
},
{
path: "/watermark",
name: "watermark",
meta: {
title: "水印",
permiss: "25",
},
component: () =>
import(
/* webpackChunkName: "watermark" */ "../views/element/watermark.vue"
),
},
{
path: "/carousel",
name: "carousel",
meta: {
title: "走马灯",
permiss: "23",
},
component: () =>
import(
/* webpackChunkName: "carousel" */ "../views/element/carousel.vue"
),
},
{
path: "/tour",
name: "tour",
meta: {
title: "分步引导",
permiss: "26",
},
component: () =>
import(/* webpackChunkName: "tour" */ "../views/element/tour.vue"),
},
{
path: "/steps",
name: "steps",
meta: {
title: "步骤条",
permiss: "27",
},
component: () =>
import(/* webpackChunkName: "steps" */ "../views/element/steps.vue"),
},
{
path: "/form",
name: "forms",
meta: {
title: "表单",
permiss: "21",
},
component: () =>
import(/* webpackChunkName: "form" */ "../views/element/form.vue"),
},
{
path: "/upload",
name: "upload",
meta: {
title: "上传",
permiss: "22",
},
component: () =>
import(
/* webpackChunkName: "upload" */ "../views/element/upload.vue"
),
},
{
path: "/statistic",
name: "statistic",
meta: {
title: "统计",
permiss: "28",
},
component: () =>
import(
/* webpackChunkName: "statistic" */ "../views/element/statistic.vue"
),
},
{
path: "/analysis",
name: "analysis",
meta: {
title: "订单分析",
permiss: "1",
},
component: () => import(/* webpackChunkName: "theme" */ "../views/pages/Analysis.vue"),
},
],
},
{
path: "/login",
meta: {
title: "登录",
noAuth: true,
},
{ path: '/:path(.*)', redirect: '/404' },
component: () =>
import(/* webpackChunkName: "login" */ "../views/pages/login.vue"),
},
{
path: "/register",
meta: {
title: "注册",
noAuth: true,
},
component: () =>
import(/* webpackChunkName: "register" */ "../views/pages/register.vue"),
},
{
path: "/reset-pwd",
meta: {
title: "重置密码",
noAuth: true,
},
component: () =>
import(
/* webpackChunkName: "reset-pwd" */ "../views/pages/reset-pwd.vue"
),
},
{
path: "/403",
meta: {
title: "没有权限",
noAuth: true,
},
component: () =>
import(/* webpackChunkName: "403" */ "../views/pages/403.vue"),
},
{
path: "/404",
meta: {
title: "找不到页面",
noAuth: true,
},
component: () =>
import(/* webpackChunkName: "404" */ "../views/pages/404.vue"),
},
{ path: "/:path(.*)", redirect: "/404" },
];
const router = createRouter({
history: createWebHashHistory(),
routes,
history: createWebHashHistory(),
routes,
});
router.beforeEach((to, from, next) => {
NProgress.start();
const role = localStorage.getItem('vuems_name');
const permiss = usePermissStore();
NProgress.start();
const role = localStorage.getItem("vuems_name");
const permiss = usePermissStore();
if (!role && to.meta.noAuth !== true) {
next('/login');
} else if (typeof to.meta.permiss == 'string' && !permiss.key.includes(to.meta.permiss)) {
// 如果没有权限则进入403
next('/403');
} else {
next();
}
if (!role && to.meta.noAuth !== true) {
next("/login");
} else if (
typeof to.meta.permiss == "string" &&
!permiss.key.includes(to.meta.permiss)
) {
// 如果没有权限则进入403
next("/403");
} else {
next();
}
});
router.afterEach(() => {
NProgress.done();
NProgress.done();
});
export default router;

View File

@ -0,0 +1,515 @@
<template>
<div class="analysis-container">
<!-- 顶部工具栏 -->
<div class="toolbar">
<h2 class="title">数据分析</h2>
<div style="display: flex; gap: 20px;">
<button class="edit-btn" @click="showEditDialog = true">
编辑数据
</button>
<button class="edit-btn" @click="() => chartData.push(Math.floor(Math.random() * 100))">
插入数据
</button>
<button class="edit-btn" @click="() => chartData.pop()">
删除数据
</button>
</div>
</div>
<!-- 统计信息卡片 -->
<div class="stats-container">
<div class="stat-card">
<div class="stat-label">平均值</div>
<div class="stat-value">{{ statistics.mean }}</div>
</div>
<div class="stat-card">
<div class="stat-label">最大值</div>
<div class="stat-value">{{ statistics.max }}</div>
</div>
<div class="stat-card">
<div class="stat-label">最小值</div>
<div class="stat-value">{{ statistics.min }}</div>
</div>
<div class="stat-card">
<div class="stat-label">总数</div>
<div class="stat-value">{{ statistics.sum }}</div>
</div>
<div class="stat-card">
<div class="stat-label">标准差</div>
<div class="stat-value">{{ statistics.stdDev }}</div>
</div>
</div>
<!-- 图表容器 -->
<div class="chart-container">
<h2>当前数据</h2>
<v-chart class="map-chart" :option="mapOptions" />
</div>
<br><br><br>
<div class="predict-chart-container">
<h2>未来趋势</h2>
<v-chart class="map-chart" :option="predictOptions" />
</div>
<!-- 编辑数据对话框 -->
<div v-if="showEditDialog" class="dialog-overlay" @click="closeDialog">
<div class="dialog" @click.stop>
<div class="dialog-header">
<h3>编辑数据</h3>
<button class="close-btn" @click="closeDialog">×</button>
</div>
<div class="dialog-content">
<div class="input-group" v-for="(value, index) in tempChartData" :key="index">
<label>数据点 {{ index + 1 }}:</label>
<input type="number" v-model.number="tempChartData[index]" class="data-input" />
</div>
</div>
<div class="dialog-footer">
<button class="cancel-btn" @click="closeDialog"></button>
<button class="save-btn" @click="saveData"></button>
</div>
</div>
</div>
</div>
</template>
<script setup>
import { ref, computed, watch } from 'vue';
import VChart from 'vue-echarts';
import { use } from 'echarts/core';
import { CanvasRenderer } from 'echarts/renderers';
import { LineChart } from 'echarts/charts';
import {
TitleComponent,
TooltipComponent,
LegendComponent,
GridComponent
} from 'echarts/components';
import * as echarts from 'echarts/core';
//
use([
CanvasRenderer,
LineChart,
TitleComponent,
TooltipComponent,
LegendComponent,
GridComponent
]);
const { graphic } = echarts;
//
const chartData = ref([120, 132, 301, 134, 90, 230, 210, 100, 400]);
const showEditDialog = ref(false);
const tempChartData = ref([...chartData.value]);
//
const statistics = computed(() => {
const data = chartData.value;
const sum = data.reduce((acc, val) => acc + val, 0);
const mean = Math.round((sum / data.length) * 100) / 100;
const max = Math.max(...data);
const min = Math.min(...data);
//
const variance = Math.round((data.reduce((acc, val) => acc + Math.pow(val - mean, 2), 0) / data.length) * 100) / 100;
//
const stdDev = Math.round(Math.sqrt(variance) * 100) / 100;
return {
mean,
max,
min,
sum,
stdDev
};
});
//
const mapOptions = computed(() => ({
xAxis: {
type: 'category',
boundaryGap: false,
data: chartData.value.map((_, index) => `${index + 1}`),
},
yAxis: {
type: 'value',
},
grid: {
top: '5%',
left: '3%',
right: '3%',
bottom: '5%',
containLabel: true,
},
color: ['#1976d2'],
tooltip: {
trigger: 'axis',
backgroundColor: 'rgba(0,0,0,0.8)',
borderColor: '#1976d2',
textStyle: {
color: '#fff'
}
},
series: [
{
type: 'line',
areaStyle: {
color: new graphic.LinearGradient(0, 0, 0, 1, [
{
offset: 0,
color: 'rgba(25, 118, 210, 0.8)',
},
{
offset: 1,
color: 'rgba(25, 118, 210, 0)',
},
]),
},
lineStyle: {
width: 4,
},
symbolSize: 8,
data: chartData.value,
}
],
}));
// const predictData = ref([120, 132, 301, 134, 500, 230, 210]);
const predictData = computed(() => {
const data = chartData.value;
if (data.length < 2) return [];
const predictions = [];
const windowSize = Math.min(3, data.length); // 使3
for (let i = 0; i < 5; i++) {
let predictedValue;
if (i % 2 === 0) {
// 使线
const recentData = [...data, ...predictions];
const len = recentData.length;
const x1 = len - windowSize, y1 = recentData[len - windowSize];
const x2 = len - 1, y2 = recentData[len - 1];
//
const slope = (y2 - y1) / (x2 - x1);
predictedValue = y2 + slope;
} else {
const lastValues = [...data, ...predictions];
const sum = lastValues.slice(-windowSize).reduce((acc, val) => acc + val, 0);
predictedValue = sum / Math.min(windowSize, lastValues.length);
}
// 使
const noise = (Math.random() - 0.5) * 20;
predictedValue = Math.max(0, Math.round(predictedValue + noise));
predictions.push(predictedValue);
}
return predictions;
});
const predictOptions = computed(() => ({
xAxis: {
type: 'category',
boundaryGap: false,
data: predictData.value.map((_, index) => `${index + chartData.value.length + 1}`),
},
yAxis: {
type: 'value',
},
grid: {
top: '5%',
left: '3%',
right: '3%',
bottom: '5%',
containLabel: true,
},
color: ['#dccc13'],
tooltip: {
trigger: 'axis',
backgroundColor: 'rgba(0,0,0,0.8)',
borderColor: '#dccc13',
textStyle: {
color: '#fff'
}
},
series: [
{
type: 'line',
areaStyle: {
color: new graphic.LinearGradient(0, 0, 0, 1, [
{
offset: 0,
color: 'rgba(255, 246, 143, 0.8)',
},
{
offset: 1,
color: 'rgba(255, 246, 143, 0)',
},
]),
},
smooth: true,
lineStyle: {
width: 4,
},
symbolSize: 8,
data: predictData.value,
}
],
}));
//
const closeDialog = () => {
showEditDialog.value = false;
tempChartData.value = [...chartData.value]; //
};
const saveData = () => {
chartData.value = [...tempChartData.value];
showEditDialog.value = false;
};
//
watch(showEditDialog, (newVal) => {
if (newVal) {
tempChartData.value = [...chartData.value];
}
});
</script>
<style scoped>
.analysis-container {
padding: 20px;
background: linear-gradient(135deg, #e3f2fd 0%, #f1f8e9 100%);
min-height: 100vh;
}
.toolbar {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 20px;
padding: 0 10px;
}
.title {
color: #1565c0;
margin: 0;
font-weight: 600;
}
.edit-btn {
background: linear-gradient(45deg, #1976d2, #42a5f5);
color: white;
border: none;
padding: 10px 20px;
border-radius: 25px;
cursor: pointer;
font-weight: 500;
transition: all 0.3s ease;
box-shadow: 0 2px 8px rgba(25, 118, 210, 0.3);
}
.edit-btn:hover {
transform: translateY(-2px);
box-shadow: 0 4px 15px rgba(25, 118, 210, 0.4);
background: linear-gradient(45deg, #1565c0, #2196f3);
}
.stats-container {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px, 1fr));
gap: 15px;
margin-bottom: 25px;
}
.stat-card {
background: white;
padding: 20px;
border-radius: 12px;
box-shadow: 0 2px 10px rgba(25, 118, 210, 0.1);
text-align: center;
transition: transform 0.3s ease;
border-left: 4px solid #1976d2;
}
.stat-card:hover {
transform: translateY(-3px);
box-shadow: 0 4px 20px rgba(25, 118, 210, 0.15);
}
.stat-label {
font-size: 14px;
color: #546e7a;
margin-bottom: 8px;
font-weight: 500;
}
.stat-value {
font-size: 24px;
font-weight: bold;
color: #1976d2;
}
.chart-container {
background: white;
padding: 20px;
border-radius: 12px;
box-shadow: 0 2px 10px rgba(25, 118, 210, 0.1);
border: 3px solid #1976d2;
}
.predict-chart-container {
background: white;
padding: 20px;
border-radius: 12px;
box-shadow: 0 2px 10px rgba(25, 118, 210, 0.1);
border: 3px solid #dccc13;
}
.map-chart {
width: 100%;
height: 400px;
}
.dialog-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(25, 118, 210, 0.1);
display: flex;
justify-content: center;
align-items: center;
z-index: 1000;
backdrop-filter: blur(2px);
}
.dialog {
background: white;
border-radius: 15px;
width: 90%;
max-width: 500px;
max-height: 80vh;
overflow: hidden;
box-shadow: 0 10px 30px rgba(25, 118, 210, 0.3);
border-top: 4px solid #1976d2;
}
.dialog-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 20px;
border-bottom: 1px solid #e3f2fd;
background: linear-gradient(135deg, #e3f2fd, #f8f9fa);
}
.dialog-header h3 {
margin: 0;
color: #1565c0;
}
.close-btn {
background: none;
border: none;
font-size: 24px;
cursor: pointer;
color: #546e7a;
width: 30px;
height: 30px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
transition: all 0.3s ease;
}
.close-btn:hover {
background: #e3f2fd;
color: #1976d2;
}
.dialog-content {
padding: 20px;
max-height: 400px;
overflow-y: auto;
}
.input-group {
display: flex;
align-items: center;
margin-bottom: 15px;
}
.input-group label {
flex: 0 0 120px;
color: #546e7a;
font-weight: 500;
}
.data-input {
flex: 1;
padding: 8px 12px;
border: 2px solid #e3f2fd;
border-radius: 8px;
font-size: 14px;
transition: border-color 0.3s ease;
}
.data-input:focus {
outline: none;
border-color: #1976d2;
box-shadow: 0 0 0 3px rgba(25, 118, 210, 0.1);
}
.dialog-footer {
display: flex;
justify-content: flex-end;
gap: 10px;
padding: 20px;
border-top: 1px solid #e3f2fd;
background: linear-gradient(135deg, #e3f2fd, #f8f9fa);
}
.cancel-btn,
.save-btn {
padding: 10px 20px;
border: none;
border-radius: 8px;
cursor: pointer;
font-weight: 500;
transition: all 0.3s ease;
}
.cancel-btn {
background: #eceff1;
color: #546e7a;
border: 1px solid #cfd8dc;
}
.cancel-btn:hover {
background: #cfd8dc;
color: #37474f;
}
.save-btn {
background: linear-gradient(45deg, #1976d2, #42a5f5);
color: white;
box-shadow: 0 2px 8px rgba(25, 118, 210, 0.3);
}
.save-btn:hover {
background: linear-gradient(45deg, #1565c0, #2196f3);
transform: translateY(-1px);
box-shadow: 0 4px 12px rgba(25, 118, 210, 0.4);
}
</style>

2971
yarn.lock

File diff suppressed because it is too large Load Diff