1 line
39 KiB
JSON
1 line
39 KiB
JSON
{"remainingRequest":"/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/node_modules/vue-loader/lib/index.js??vue-loader-options!/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/src/views/user/Console.vue?vue&type=script&lang=js","dependencies":[{"path":"/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/src/views/user/Console.vue","mtime":1747403482892},{"path":"/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/node_modules/cache-loader/dist/cjs.js","mtime":1743264595665},{"path":"/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/node_modules/babel-loader/lib/index.js","mtime":1743264596348},{"path":"/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/node_modules/cache-loader/dist/cjs.js","mtime":1743264595665},{"path":"/Users/shuguang/Desktop/毕设/CodeMaster/CodeMaster/node_modules/vue-loader/lib/index.js","mtime":1743264596512}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:
import * as echarts from "echarts";

export default {
  name: "Console",
  data() {
    return {
      loading: false,
      // 用户统计数据
      totaluser: 0,
      totalteacher: 0,
      totalstudent: 0,
      totalmatch: 0,

      // 获奖数据
      firstlevel: 0,
      secondlevel: 0,
      thirdlevel: 0,

      // 报名趋势数据
      registrationTrendData: [],

      // 热门比赛列表
      hotCompetitions: [],

      // 其他统计数据
      totalinfo: 0,
      totalsuccess: 0,
      totalday: 0,
      totaldays: 0,

      // 图表实例
      registrationTrendChart: null,
      awardChart: null,

      // 是否使用模拟数据
      useMockData: false,
    };
  },
  mounted() {
    this.fetchAllData();
  },
  beforeDestroy() {
    // 销毁图表实例，避免内存泄漏
    if (this.registrationTrendChart) {
      this.registrationTrendChart.dispose();
    }
    if (this.awardChart) {
      this.awardChart.dispose();
    }
  },
  methods: {
    // 获取所有数据
    async fetchAllData() {
      this.loading = true;
      try {
        // 优先获取基础统计数据
        await this.fetchStatistics();
        // 获取其他数据
        await Promise.all([
          this.fetchAwardStats(),
          this.fetchRegistrationTrend(),
          this.fetchHotCompetitions(),
        ]);

        this.$nextTick(() => {
          this.initRegistrationTrendChart();
          this.initAwardChart();
        });
      } catch (error) {
        console.error("获取数据失败", error);
        this.$message.error("获取数据失败");
      } finally {
        this.loading = false;
      }
    },

    // 获取统计数据
    async fetchStatistics() {
      try {
        // this.$api.getWinCountByLevel
        const response = await this.$api.getCountTotal();
        console.log("response", response);
        if (response.code === 200) {
          console.log("response.data", response.data);
          const stats = response.data;
          // 直接使用接口返回的数据
          this.totaluser = stats.totalUsers;
          this.totalteacher = stats.teacherCount;
          this.totalstudent = stats.studentCount;
          this.totalmatch = stats.competitionCount;
        }
      } catch (error) {
        console.error("获取统计数据失败:", error);
        this.$message.error("获取统计数据失败");
        // 发生错误时重置数据为0
        this.totaluser = 0;
        this.totalteacher = 0;
        this.totalstudent = 0;
        this.totalmatch = 0;
      }
    },

    // 加载模拟数据
    loadMockData() {
      // 模拟用户统计数据
      this.totaluser = 1256;
      this.totalteacher = 128;
      this.totalstudent = 1128;
      this.totalmatch = 45;

      // 模拟获奖数据
      this.firstlevel = 56;
      this.secondlevel = 112;
      this.thirdlevel = 235;

      // 模拟其他统计数据
      this.totalinfo = 89;
      this.totalsuccess = 876;
      this.totalday = 32;
      this.totaldays = 5;

      // 模拟报名趋势数据
      this.generateMockTrendData();

      // 模拟热门比赛数据
      this.generateMockHotCompetitions();
    },

    // 生成模拟趋势数据
    generateMockTrendData() {
      const dates = [];
      const counts = [];
      const today = new Date();

      for (let i = 29; i >= 0; i--) {
        const date = new Date(today);
        date.setDate(date.getDate() - i);
        const dateStr = this.formatDate(date);
        dates.push(dateStr);

        // 生成随机报名数量，呈现波动上升趋势
        const baseCount = 5 + Math.floor((30 - i) / 3);
        const randomFactor = Math.floor(Math.random() * 10) - 3;
        const count = Math.max(0, baseCount + randomFactor);

        counts.push(count);
      }

      this.registrationTrendData = {
        dates,
        counts,
      };
    },

    // 生成模拟热门比赛数据
    generateMockHotCompetitions() {
      const competitionNames = [
        "全国大学生数学建模竞赛",
        "互联网+创新创业大赛",
        "ACM程序设计大赛",
        "挑战杯创新创业大赛",
        "全国大学生机器人大赛",
      ];

      this.hotCompetitions = competitionNames.map((name, index) => {
        // 生成随机报名人数，排名越前报名人数越多
        const baseCount = 200 - index * 30;
        const randomFactor = Math.floor(Math.random() * 20);

        return {
          competitionId: `comp-${index + 1}`,
          competitionName: name,
          registrationCount: baseCount + randomFactor,
          competitionStatus: Math.random() > 0.3 ? 0 : 1,
        };
      });
    },

    // 获取用户统计数据
    async fetchUserStats() {
      // 获取用户总数
      await this.$api.getCount().then((res) => {
        if (res.code == 200) {
          this.totaluser = res.data;
        }
      });

      // 获取教师总人数
      await this.$api.getTeacherCount().then((res) => {
        if (res.code == 200) {
          this.totalteacher = res.data;
        }
      });

      // 获取学生总人数
      await this.$api.getStudentCount().then((res) => {
        if (res.code == 200) {
          this.totalstudent = res.data;
        }
      });
    },

    // 获取比赛统计数据
    async fetchCompetitionStats() {
      // 获取所有比赛数量
      await this.$api.AllCompetition().then((res) => {
        if (res.code == 200) {
          this.totalmatch = res.data.length;

          // 处理热门比赛数据
          this.processCompetitionData(res.data);
        }
      });

      // 获取所有资料
      await this.$api.Allinformation().then((res) => {
        if (res.code == 200) {
          this.totalinfo = res.data.length;
        }
      });

      // 获取七天内比赛的数量
      const currentDate = new Date();
      const sevenDaysAgo = new Date(currentDate);
      sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7);

      await this.$api.AllCompetition().then((res) => {
        if (res.code == 200) {
          this.totaldays = res.data.filter((item) => {
            const registrationEndTime = new Date(item.registrationEndTime);
            return (
              registrationEndTime >= sevenDaysAgo &&
              registrationEndTime <= currentDate
            );
          }).length;
        }
      });
    },

    // 获取获奖统计数据
    async fetchAwardStats() {
      // 一等奖人数
      await this.$api.getWinCountByLevel({ level: "一等奖" }).then((res) => {
        if (res.code == 200) {
          this.firstlevel = res.data;
        }
      });

      // 二等奖人数
      await this.$api.getWinCountByLevel({ level: "二等奖" }).then((res) => {
        if (res.code == 200) {
          this.secondlevel = res.data;
        }
      });

      // 三等奖
      await this.$api.getWinCountByLevel({ level: "三等奖" }).then((res) => {
        if (res.code == 200) {
          this.thirdlevel = res.data;
        }
      });
    },

    // 获取报名趋势数据
    async fetchRegistrationTrend() {
      // 获取报名记录
      await this.$api.RegistrationAll().then((res) => {
        if (res.code == 200) {
          // 处理报名趋势数据
          this.processRegistrationTrendData(res.data);

          // 获取成功报名数量
          this.totalsuccess = res.data.filter(
            (item) => item.registrationStatus == "成功报名"
          ).length;

          // 获取七天内成功报名的数量
          const currentDate = new Date();
          const sevenDaysAgo = new Date(currentDate);
          sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7);

          this.totalday = res.data.filter((item) => {
            const registrationTime = new Date(item.registrationTime);
            return (
              registrationTime >= sevenDaysAgo &&
              registrationTime <= currentDate
            );
          }).length;
        }
      });
    },

    // 获取热门比赛数据
    async fetchHotCompetitions() {
      // 获取报名记录和比赛数据
      const [registrationRes, competitionRes] = await Promise.all([
        this.$api.RegistrationAll(),
        this.$api.AllCompetition(),
      ]);

      if (registrationRes.code == 200 && competitionRes.code == 200) {
        const registrations = registrationRes.data;
        const competitions = competitionRes.data;

        // 计算每个比赛的报名人数
        const competitionCounts = {};
        registrations.forEach((reg) => {
          if (!competitionCounts[reg.competitionId]) {
            competitionCounts[reg.competitionId] = 0;
          }
          competitionCounts[reg.competitionId]++;
        });

        // 将比赛数据与报名人数合并
        const competitionsWithCount = competitions.map((comp) => ({
          ...comp,
          registrationCount: competitionCounts[comp.competitionId] || 0,
        }));

        // 按报名人数排序，取前5个
        this.hotCompetitions = competitionsWithCount
          .sort((a, b) => b.registrationCount - a.registrationCount)
          .slice(0, 5);
      }
    },

    // 处理报名趋势数据
    processRegistrationTrendData(registrations) {
      // 获取最近30天的日期
      const dates = [];
      const counts = [];
      const today = new Date();

      for (let i = 29; i >= 0; i--) {
        const date = new Date(today);
        date.setDate(date.getDate() - i);
        const dateStr = this.formatDate(date);
        dates.push(dateStr);

        // 计算当天的报名数量
        const count = registrations.filter((reg) => {
          const regDate = new Date(reg.registrationTime);
          return this.formatDate(regDate) === dateStr;
        }).length;

        counts.push(count);
      }

      this.registrationTrendData = {
        dates,
        counts,
      };
    },

    // 处理比赛数据
    processCompetitionData(competitions) {
      // 在这里可以添加更多的数据处理逻辑
    },

    // 初始化报名趋势图表
    initRegistrationTrendChart() {
      const chartDom = this.$refs.registrationTrend;
      this.registrationTrendChart = echarts.init(chartDom);

      const option = {
        tooltip: {
          trigger: "axis",
          axisPointer: {
            type: "shadow",
          },
        },
        grid: {
          left: "3%",
          right: "4%",
          bottom: "3%",
          containLabel: true,
        },
        xAxis: {
          type: "category",
          data: this.registrationTrendData.dates,
          axisLabel: {
            rotate: 45,
          },
        },
        yAxis: {
          type: "value",
          name: "报名人数",
        },
        series: [
          {
            name: "报名人数",
            type: "line",
            data: this.registrationTrendData.counts,
            smooth: true,
            areaStyle: {
              opacity: 0.3,
            },
            itemStyle: {
              color: "#1890ff",
            },
          },
        ],
      };

      this.registrationTrendChart.setOption(option);

      // 响应窗口大小变化
      window.addEventListener("resize", () => {
        this.registrationTrendChart.resize();
      });
    },

    // 初始化获奖情况图表
    initAwardChart() {
      const chartDom = this.$refs.awardChart;
      this.awardChart = echarts.init(chartDom);

      const option = {
        tooltip: {
          trigger: "item",
          formatter: "{a} <br/>{b}: {c} ({d}%)",
        },
        legend: {
          orient: "horizontal",
          bottom: "bottom",
          data: ["一等奖", "二等奖", "三等奖"],
        },
        series: [
          {
            name: "获奖情况",
            type: "pie",
            radius: ["40%", "70%"],
            avoidLabelOverlap: false,
            itemStyle: {
              borderRadius: 10,
              borderColor: "#fff",
              borderWidth: 2,
            },
            label: {
              show: false,
              position: "center",
            },
            emphasis: {
              label: {
                show: true,
                fontSize: "18",
                fontWeight: "bold",
              },
            },
            labelLine: {
              show: false,
            },
            data: [
              {
                value: this.firstlevel,
                name: "一等奖",
                itemStyle: { color: "#ff4d4f" },
              },
              {
                value: this.secondlevel,
                name: "二等奖",
                itemStyle: { color: "#faad14" },
              },
              {
                value: this.thirdlevel,
                name: "三等奖",
                itemStyle: { color: "#52c41a" },
              },
            ],
          },
        ],
      };

      this.awardChart.setOption(option);

      // 响应窗口大小变化
      window.addEventListener("resize", () => {
        this.awardChart.resize();
      });
    },

    // 格式化日期
    formatDate(date) {
      const year = date.getFullYear();
      const month = String(date.getMonth() + 1).padStart(2, "0");
      const day = String(date.getDate()).padStart(2, "0");
      return `${year}-${month}-${day}`;
    },

    // 获取比赛状态文本
    getCompetitionStatus(status) {
      const statusMap = {
        0: "进行中",
        1: "已终止",
      };
      return statusMap[status] || "未知状态";
    },
  },
};
"},{"version":3,"sources":["Console.vue"],"names":[],"mappings":";AAkHA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"Console.vue","sourceRoot":"src/views/user","sourcesContent":["<template>\n <div class=\"dashboard-container\">\n <!-- 数据卡片区域 -->\n <a-row :gutter=\"24\">\n <a-col :span=\"6\">\n <a-card class=\"data-card\">\n <a-statistic\n title=\"用户总数\"\n :value=\"totaluser\"\n :precision=\"0\"\n style=\"text-align: center\"\n >\n <template #prefix>\n <a-icon type=\"user\" />\n </template>\n </a-statistic>\n </a-card>\n </a-col>\n <a-col :span=\"6\">\n <a-card class=\"data-card\">\n <a-statistic\n title=\"教师人数\"\n :value=\"totalteacher\"\n :precision=\"0\"\n style=\"text-align: center\"\n >\n <template #prefix>\n <a-icon type=\"team\" />\n </template>\n </a-statistic>\n </a-card>\n </a-col>\n <a-col :span=\"6\">\n <a-card class=\"data-card\">\n <a-statistic\n title=\"学生人数\"\n :value=\"totalstudent\"\n :precision=\"0\"\n style=\"text-align: center\"\n >\n <template #prefix>\n <a-icon type=\"solution\" />\n </template>\n </a-statistic>\n </a-card>\n </a-col>\n <a-col :span=\"6\">\n <a-card class=\"data-card\">\n <a-statistic\n title=\"比赛总数\"\n :value=\"totalmatch\"\n :precision=\"0\"\n style=\"text-align: center\"\n >\n <template #prefix>\n <a-icon type=\"trophy\" />\n </template>\n </a-statistic>\n </a-card>\n </a-col>\n </a-row>\n\n <!-- 趋势图和热门比赛区域 -->\n <a-row :gutter=\"24\" style=\"margin-top: 24px\">\n <!-- 报名趋势图 -->\n <a-col :span=\"16\">\n <a-card title=\"学生报名比赛趋势\" :bordered=\"false\">\n <div ref=\"registrationTrend\" style=\"height: 400px\"></div>\n </a-card>\n </a-col>\n\n <!-- 热门比赛列表 -->\n <a-col :span=\"8\">\n <a-card title=\"热门比赛\" :bordered=\"false\">\n <a-list\n itemLayout=\"horizontal\"\n :dataSource=\"hotCompetitions\"\n :loading=\"loading\"\n >\n <a-list-item slot=\"renderItem\" slot-scope=\"item\">\n <a-list-item-meta>\n <template slot=\"title\">\n <a href=\"javascript:;\">{{ item.competitionName }}</a>\n </template>\n <template slot=\"description\">\n <span>报名人数: {{ item.registrationCount }}</span>\n <span style=\"margin-left: 16px\">\n 状态: {{ getCompetitionStatus(item.competitionStatus) }}\n </span>\n </template>\n <a-avatar\n slot=\"avatar\"\n style=\"background-color: #1890ff\"\n icon=\"trophy\"\n />\n </a-list-item-meta>\n </a-list-item>\n </a-list>\n </a-card>\n </a-col>\n </a-row>\n\n <!-- 获奖情况统计 -->\n <a-row :gutter=\"24\" style=\"margin-top: 24px\">\n <a-col :span=\"24\">\n <a-card title=\"获奖情况统计\" :bordered=\"false\">\n <div ref=\"awardChart\" style=\"height: 300px\"></div>\n </a-card>\n </a-col>\n </a-row>\n </div>\n</template>\n\n<script>\nimport * as echarts from \"echarts\";\n\nexport default {\n name: \"Console\",\n data() {\n return {\n loading: false,\n // 用户统计数据\n totaluser: 0,\n totalteacher: 0,\n totalstudent: 0,\n totalmatch: 0,\n\n // 获奖数据\n firstlevel: 0,\n secondlevel: 0,\n thirdlevel: 0,\n\n // 报名趋势数据\n registrationTrendData: [],\n\n // 热门比赛列表\n hotCompetitions: [],\n\n // 其他统计数据\n totalinfo: 0,\n totalsuccess: 0,\n totalday: 0,\n totaldays: 0,\n\n // 图表实例\n registrationTrendChart: null,\n awardChart: null,\n\n // 是否使用模拟数据\n useMockData: false,\n };\n },\n mounted() {\n this.fetchAllData();\n },\n beforeDestroy() {\n // 销毁图表实例,避免内存泄漏\n if (this.registrationTrendChart) {\n this.registrationTrendChart.dispose();\n }\n if (this.awardChart) {\n this.awardChart.dispose();\n }\n },\n methods: {\n // 获取所有数据\n async fetchAllData() {\n this.loading = true;\n try {\n // 优先获取基础统计数据\n await this.fetchStatistics();\n // 获取其他数据\n await Promise.all([\n this.fetchAwardStats(),\n this.fetchRegistrationTrend(),\n this.fetchHotCompetitions(),\n ]);\n\n this.$nextTick(() => {\n this.initRegistrationTrendChart();\n this.initAwardChart();\n });\n } catch (error) {\n console.error(\"获取数据失败\", error);\n this.$message.error(\"获取数据失败\");\n } finally {\n this.loading = false;\n }\n },\n\n // 获取统计数据\n async fetchStatistics() {\n try {\n // this.$api.getWinCountByLevel\n const response = await this.$api.getCountTotal();\n console.log(\"response\", response);\n if (response.code === 200) {\n console.log(\"response.data\", response.data);\n const stats = response.data;\n // 直接使用接口返回的数据\n this.totaluser = stats.totalUsers;\n this.totalteacher = stats.teacherCount;\n this.totalstudent = stats.studentCount;\n this.totalmatch = stats.competitionCount;\n }\n } catch (error) {\n console.error(\"获取统计数据失败:\", error);\n this.$message.error(\"获取统计数据失败\");\n // 发生错误时重置数据为0\n this.totaluser = 0;\n this.totalteacher = 0;\n this.totalstudent = 0;\n this.totalmatch = 0;\n }\n },\n\n // 加载模拟数据\n loadMockData() {\n // 模拟用户统计数据\n this.totaluser = 1256;\n this.totalteacher = 128;\n this.totalstudent = 1128;\n this.totalmatch = 45;\n\n // 模拟获奖数据\n this.firstlevel = 56;\n this.secondlevel = 112;\n this.thirdlevel = 235;\n\n // 模拟其他统计数据\n this.totalinfo = 89;\n this.totalsuccess = 876;\n this.totalday = 32;\n this.totaldays = 5;\n\n // 模拟报名趋势数据\n this.generateMockTrendData();\n\n // 模拟热门比赛数据\n this.generateMockHotCompetitions();\n },\n\n // 生成模拟趋势数据\n generateMockTrendData() {\n const dates = [];\n const counts = [];\n const today = new Date();\n\n for (let i = 29; i >= 0; i--) {\n const date = new Date(today);\n date.setDate(date.getDate() - i);\n const dateStr = this.formatDate(date);\n dates.push(dateStr);\n\n // 生成随机报名数量,呈现波动上升趋势\n const baseCount = 5 + Math.floor((30 - i) / 3);\n const randomFactor = Math.floor(Math.random() * 10) - 3;\n const count = Math.max(0, baseCount + randomFactor);\n\n counts.push(count);\n }\n\n this.registrationTrendData = {\n dates,\n counts,\n };\n },\n\n // 生成模拟热门比赛数据\n generateMockHotCompetitions() {\n const competitionNames = [\n \"全国大学生数学建模竞赛\",\n \"互联网+创新创业大赛\",\n \"ACM程序设计大赛\",\n \"挑战杯创新创业大赛\",\n \"全国大学生机器人大赛\",\n ];\n\n this.hotCompetitions = competitionNames.map((name, index) => {\n // 生成随机报名人数,排名越前报名人数越多\n const baseCount = 200 - index * 30;\n const randomFactor = Math.floor(Math.random() * 20);\n\n return {\n competitionId: `comp-${index + 1}`,\n competitionName: name,\n registrationCount: baseCount + randomFactor,\n competitionStatus: Math.random() > 0.3 ? 0 : 1,\n };\n });\n },\n\n // 获取用户统计数据\n async fetchUserStats() {\n // 获取用户总数\n await this.$api.getCount().then((res) => {\n if (res.code == 200) {\n this.totaluser = res.data;\n }\n });\n\n // 获取教师总人数\n await this.$api.getTeacherCount().then((res) => {\n if (res.code == 200) {\n this.totalteacher = res.data;\n }\n });\n\n // 获取学生总人数\n await this.$api.getStudentCount().then((res) => {\n if (res.code == 200) {\n this.totalstudent = res.data;\n }\n });\n },\n\n // 获取比赛统计数据\n async fetchCompetitionStats() {\n // 获取所有比赛数量\n await this.$api.AllCompetition().then((res) => {\n if (res.code == 200) {\n this.totalmatch = res.data.length;\n\n // 处理热门比赛数据\n this.processCompetitionData(res.data);\n }\n });\n\n // 获取所有资料\n await this.$api.Allinformation().then((res) => {\n if (res.code == 200) {\n this.totalinfo = res.data.length;\n }\n });\n\n // 获取七天内比赛的数量\n const currentDate = new Date();\n const sevenDaysAgo = new Date(currentDate);\n sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7);\n\n await this.$api.AllCompetition().then((res) => {\n if (res.code == 200) {\n this.totaldays = res.data.filter((item) => {\n const registrationEndTime = new Date(item.registrationEndTime);\n return (\n registrationEndTime >= sevenDaysAgo &&\n registrationEndTime <= currentDate\n );\n }).length;\n }\n });\n },\n\n // 获取获奖统计数据\n async fetchAwardStats() {\n // 一等奖人数\n await this.$api.getWinCountByLevel({ level: \"一等奖\" }).then((res) => {\n if (res.code == 200) {\n this.firstlevel = res.data;\n }\n });\n\n // 二等奖人数\n await this.$api.getWinCountByLevel({ level: \"二等奖\" }).then((res) => {\n if (res.code == 200) {\n this.secondlevel = res.data;\n }\n });\n\n // 三等奖\n await this.$api.getWinCountByLevel({ level: \"三等奖\" }).then((res) => {\n if (res.code == 200) {\n this.thirdlevel = res.data;\n }\n });\n },\n\n // 获取报名趋势数据\n async fetchRegistrationTrend() {\n // 获取报名记录\n await this.$api.RegistrationAll().then((res) => {\n if (res.code == 200) {\n // 处理报名趋势数据\n this.processRegistrationTrendData(res.data);\n\n // 获取成功报名数量\n this.totalsuccess = res.data.filter(\n (item) => item.registrationStatus == \"成功报名\"\n ).length;\n\n // 获取七天内成功报名的数量\n const currentDate = new Date();\n const sevenDaysAgo = new Date(currentDate);\n sevenDaysAgo.setDate(sevenDaysAgo.getDate() - 7);\n\n this.totalday = res.data.filter((item) => {\n const registrationTime = new Date(item.registrationTime);\n return (\n registrationTime >= sevenDaysAgo &&\n registrationTime <= currentDate\n );\n }).length;\n }\n });\n },\n\n // 获取热门比赛数据\n async fetchHotCompetitions() {\n // 获取报名记录和比赛数据\n const [registrationRes, competitionRes] = await Promise.all([\n this.$api.RegistrationAll(),\n this.$api.AllCompetition(),\n ]);\n\n if (registrationRes.code == 200 && competitionRes.code == 200) {\n const registrations = registrationRes.data;\n const competitions = competitionRes.data;\n\n // 计算每个比赛的报名人数\n const competitionCounts = {};\n registrations.forEach((reg) => {\n if (!competitionCounts[reg.competitionId]) {\n competitionCounts[reg.competitionId] = 0;\n }\n competitionCounts[reg.competitionId]++;\n });\n\n // 将比赛数据与报名人数合并\n const competitionsWithCount = competitions.map((comp) => ({\n ...comp,\n registrationCount: competitionCounts[comp.competitionId] || 0,\n }));\n\n // 按报名人数排序,取前5个\n this.hotCompetitions = competitionsWithCount\n .sort((a, b) => b.registrationCount - a.registrationCount)\n .slice(0, 5);\n }\n },\n\n // 处理报名趋势数据\n processRegistrationTrendData(registrations) {\n // 获取最近30天的日期\n const dates = [];\n const counts = [];\n const today = new Date();\n\n for (let i = 29; i >= 0; i--) {\n const date = new Date(today);\n date.setDate(date.getDate() - i);\n const dateStr = this.formatDate(date);\n dates.push(dateStr);\n\n // 计算当天的报名数量\n const count = registrations.filter((reg) => {\n const regDate = new Date(reg.registrationTime);\n return this.formatDate(regDate) === dateStr;\n }).length;\n\n counts.push(count);\n }\n\n this.registrationTrendData = {\n dates,\n counts,\n };\n },\n\n // 处理比赛数据\n processCompetitionData(competitions) {\n // 在这里可以添加更多的数据处理逻辑\n },\n\n // 初始化报名趋势图表\n initRegistrationTrendChart() {\n const chartDom = this.$refs.registrationTrend;\n this.registrationTrendChart = echarts.init(chartDom);\n\n const option = {\n tooltip: {\n trigger: \"axis\",\n axisPointer: {\n type: \"shadow\",\n },\n },\n grid: {\n left: \"3%\",\n right: \"4%\",\n bottom: \"3%\",\n containLabel: true,\n },\n xAxis: {\n type: \"category\",\n data: this.registrationTrendData.dates,\n axisLabel: {\n rotate: 45,\n },\n },\n yAxis: {\n type: \"value\",\n name: \"报名人数\",\n },\n series: [\n {\n name: \"报名人数\",\n type: \"line\",\n data: this.registrationTrendData.counts,\n smooth: true,\n areaStyle: {\n opacity: 0.3,\n },\n itemStyle: {\n color: \"#1890ff\",\n },\n },\n ],\n };\n\n this.registrationTrendChart.setOption(option);\n\n // 响应窗口大小变化\n window.addEventListener(\"resize\", () => {\n this.registrationTrendChart.resize();\n });\n },\n\n // 初始化获奖情况图表\n initAwardChart() {\n const chartDom = this.$refs.awardChart;\n this.awardChart = echarts.init(chartDom);\n\n const option = {\n tooltip: {\n trigger: \"item\",\n formatter: \"{a} <br/>{b}: {c} ({d}%)\",\n },\n legend: {\n orient: \"horizontal\",\n bottom: \"bottom\",\n data: [\"一等奖\", \"二等奖\", \"三等奖\"],\n },\n series: [\n {\n name: \"获奖情况\",\n type: \"pie\",\n radius: [\"40%\", \"70%\"],\n avoidLabelOverlap: false,\n itemStyle: {\n borderRadius: 10,\n borderColor: \"#fff\",\n borderWidth: 2,\n },\n label: {\n show: false,\n position: \"center\",\n },\n emphasis: {\n label: {\n show: true,\n fontSize: \"18\",\n fontWeight: \"bold\",\n },\n },\n labelLine: {\n show: false,\n },\n data: [\n {\n value: this.firstlevel,\n name: \"一等奖\",\n itemStyle: { color: \"#ff4d4f\" },\n },\n {\n value: this.secondlevel,\n name: \"二等奖\",\n itemStyle: { color: \"#faad14\" },\n },\n {\n value: this.thirdlevel,\n name: \"三等奖\",\n itemStyle: { color: \"#52c41a\" },\n },\n ],\n },\n ],\n };\n\n this.awardChart.setOption(option);\n\n // 响应窗口大小变化\n window.addEventListener(\"resize\", () => {\n this.awardChart.resize();\n });\n },\n\n // 格式化日期\n formatDate(date) {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n return `${year}-${month}-${day}`;\n },\n\n // 获取比赛状态文本\n getCompetitionStatus(status) {\n const statusMap = {\n 0: \"进行中\",\n 1: \"已终止\",\n };\n return statusMap[status] || \"未知状态\";\n },\n },\n};\n</script>\n\n<style lang=\"less\" scoped>\n.dashboard-container {\n padding: 24px;\n background-color: #f0f2f5;\n min-height: 100%;\n}\n\n.data-card {\n margin-bottom: 24px;\n border-radius: 8px;\n overflow: hidden;\n box-shadow: 0 1px 2px rgba(0, 0, 0, 0.07);\n transition: all 0.3s;\n\n &:hover {\n box-shadow: 0 2px 8px rgba(0, 0, 0, 0.15);\n transform: translateY(-2px);\n }\n\n :deep(.ant-statistic-title) {\n font-size: 16px;\n color: rgba(0, 0, 0, 0.65);\n margin-bottom: 16px;\n }\n\n :deep(.ant-statistic-content) {\n font-size: 28px;\n font-weight: 600;\n color: rgba(0, 0, 0, 0.85);\n }\n\n :deep(.ant-statistic-content-prefix) {\n margin-right: 8px;\n font-size: 24px;\n color: #1890ff;\n }\n}\n\n.ant-card {\n border-radius: 8px;\n box-shadow: 0 1px 2px rgba(0, 0, 0, 0.07);\n\n :deep(.ant-card-head) {\n border-bottom: 1px solid #f0f0f0;\n padding: 0 24px;\n\n .ant-card-head-title {\n font-size: 16px;\n font-weight: 600;\n color: rgba(0, 0, 0, 0.85);\n }\n }\n}\n</style>\n"]}]} |