yyb
10 小时以前 d648788841a802a2f56db5bd657a408b9b229d65
src/views/index.vue
@@ -1,624 +1,1124 @@
<template>
  <div class="app-container home">
    <div style="display: flex;">
      <div>
        <div class="card-top-left">
          <div class="title">
            <span style="font-weight: bold">本月销售、采购情况计划</span>
          </div>
          <div class="card-group">
            <div class="info-card">
              <div class="info-message">
                <div class="info-number">{{ contractAmount }}</div>
                <div class="info-title">合同金额(元)</div>
              </div>
              <img src="@/assets/images/icon1.png" alt="" style="width: 63px;height: 63px">
            </div>
            <div class="info-card1">
              <div class="info-message">
                <div class="info-number">{{ invoiceAmount }}</div>
                <div class="info-title">开票金额(元)</div>
              </div>
              <img src="@/assets/images/icon2.png" alt="" style="width: 63px;height: 63px">
            </div>
            <div class="info-card2">
              <div class="info-message">
                <div class="info-number">{{ receiptAmount }}</div>
                <div class="info-title">回款金额(元)</div>
              </div>
              <img src="@/assets/images/icon%203.png" alt="" style="width: 63px;height: 63px">
            </div>
          </div>
        </div>
        <div class="card-top-left">
          <div class="title">
            <span style="font-weight: bold">本月应收、应付情况计划</span>
          </div>
          <div class="pie">
            <div class="card-group">
              <div class="pie-group">
                <div style="margin-right: 80px">
                  <Echarts ref="chart" :legend="pieLegend" :chartStyle="chartStyle" :series="materialPieSeries"
                    :tooltip="pieTooltip"></Echarts>
                </div>
                <div class="info-message2">
                  <div class="info-message1">
                    <div class="pie-title">本月回款金额</div>
                    <div class="pie-info"><span class="pie-number">{{ receiveAmount }}</span>元 <span
                        class="pie-number">{{ receiveAmountPercentage }}</span>%</div>
                  </div>
                  <div class="info-message1">
                    <div class="pie-title">应收款金额</div>
                    <div class="pie-info"><span class="pie-number">{{ contractAmountMonth }}</span>元</div>
                  </div>
                </div>
              </div>
            </div>
            <div class="card-group">
              <div class="pie-group">
                <div style="margin-right: 80px">
                  <Echarts ref="chart" :options="options" :legend="pieLegend" :chartStyle="chartStyle"
                    :series="materialPieSeries1" :tooltip="pieTooltip1"></Echarts>
                </div>
                <div class="info-message2">
                  <div class="info-message1">
                    <div class="pie-title1">本月付款金额</div>
                    <div class="pie-info"><span class="pie-number1">{{ paymentAmount }}</span>元 <span
                        class="pie-number1">{{ payableAmountPercentage }}</span>%</div>
                  </div>
                  <div class="info-message1">
                    <div class="pie-title1">应付款金额</div>
                    <div class="pie-info"><span class="pie-number1">{{ payableAmount }}</span>元</div>
                  </div>
                </div>
              </div>
            </div>
          </div>
        </div>
      </div>
      <div>
        <div class="card-top-right">
          <div class="title">
            <span style="font-weight: bold">客户合同金额TOP5统计</span>
          </div>
          <div>
            <Echarts ref="chart" :chartStyle="chartStyle1" :grid="grid" :legend="barLegend" :series="barSeries"
              :tooltip="tooltip" :xAxis="xAxis1" :yAxis="yAxis1" style="height: 42vh;"></Echarts>
          </div>
        </div>
      </div>
    </div>
    <div>
      <div>
        <div class="card-bottom">
          <div class="title">
            <span style="font-weight: bold">回款、开票近半年走势图</span>
          </div>
          <div>
            <Echarts ref="chart" :chartStyle="chartStyle1" :grid="grid" :legend="barLegend" :series="lineSeries"
              :tooltip="tooltipLine" :xAxis="xAxis2" :yAxis="yAxis2" style="height: 27vh;"></Echarts>
          </div>
        </div>
      </div>
    </div>
  </div>
</template>
<script setup name="Index">
const { proxy } = getCurrentInstance()
import * as echarts from 'echarts';
import Echarts from "@/components/Echarts/echarts.vue";
import {
  getAmountHalfYear,
  getAmountMouth,
  getContractAmount,
  getInvoiceAmount,
  getReceiptAmount,
  getTopFiveList, paymentMonthList
} from "@/api/viewIndex.js";
const pieLegend = reactive({
  show: false,
})
const contractAmount = ref(0)
const invoiceAmount = ref(0)
const receiptAmount = ref(0)
const receiveAmount = ref(0)
const contractAmountMonth = ref(0)
const receiveAmountPercentage = ref(0)
const paymentAmount = ref(0)
const payableAmount = ref(0)
const payableAmountPercentage = ref(0)
const options = reactive({
  graphic: {
    type: 'circle',
    left: 'center',
    top: 'middle',
    shape: {
      r: '70%' // 圆形半径与饼图外圈相同
    },
  }
})
const pieTooltip = reactive({
  trigger: 'item',
  formatter: function (params) {
    // 动态生成提示信息,基于数据项的 name 属性
    const description = params.name === '本月回款金额' ? '本月回款金额' : '应收款金额';
    return `${description} ${formatNumber(params.value)}元 ${params.percent}%`;
  },
  position: 'right'
})
const pieTooltip1 = reactive({
  trigger: 'item',
  formatter: function (params) {
    // 动态生成提示信息,基于数据项的 name 属性
    const description = params.name === '本月付款金额' ? '本月付款金额' : '应付款金额';
    return `${description} ${formatNumber(params.value)}元 ${params.percent}%`;
  },
  position: 'right'
})
// 数字格式化函数,添加逗号作为千位分隔符
function formatNumber(num) {
  return num.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ',');
}
const materialPieSeries = ref([
  {
    type: 'pie',
    radius: '92%',
    avoidLabelOverlap: false,
    itemStyle: {
      borderColor: '#fff',
      borderWidth: 2
    },
    label: {
      show: false
    },
    data: [
      { value: 0, name: '本月回款金额', itemStyle: { color: '#2D99FF' } },
      { value: 0, name: '应收款金额', itemStyle: { color: '#D4DDFF' } },
    ]
  }
])
const materialPieSeries1 = ref([
  {
    type: 'pie',
    radius: '92%',
    avoidLabelOverlap: false,
    itemStyle: {
      borderColor: '#fff',
      borderWidth: 2
    },
    label: {
      show: false
    },
    data: [
      { value: 0, name: '本月付款金额', itemStyle: { color: '#1EBFAC' } },
      { value: 0, name: '应付款金额', itemStyle: { color: '#D0EFE1' } },
    ]
  }
])
const chartStyle = {
  width: '150%',
  height: '120%' // 设置图表容器的高度
}
const chartStyle1 = {
  width: '100%',
  height: '100%' // 设置图表容器的高度
}
const grid = {
  left: '2%', // 增大左侧留白
  right: '10%',
  bottom: '15%',
  top: '10%',
  containLabel: true // 确保包含标签
}
const tooltip = {
  trigger: 'axis',
  axisPointer: {
    type: 'shadow'
  }
}
const tooltipLine = {
  trigger: 'axis',
}
const yAxis1 = ref([
  {
    type: 'value',
  }
])
const xAxis1 = ref([
  {
    type: 'category',
    data: []
  }
])
const yAxis2 = ref([
  {
    type: 'value',
  }
])
const xAxis2 = ref([
  {
    type: 'category',
    data: []
  }
])
const barLegend = reactive({})
const barSeries = ref([
  {
    type: 'bar',
    data: [],
    label: {
      show: true
    },
  },
])
const lineSeries = ref([
  {
    type: 'line',
    data: [],
    label: {
      show: true
    },
  },
])
// 合同金额
const getContractAmountNum = () => {
  getContractAmount().then((res) => {
    contractAmount.value = res.data
  })
}
// 开票金额
const getInvoiceAmountNum = () => {
  getInvoiceAmount().then((res) => {
    invoiceAmount.value = res.data
  })
}
// 回款金额
const getReceiptAmountNum = () => {
  getReceiptAmount().then((res) => {
    receiptAmount.value = res.data
  })
}
// 月回款金额饼图
const getAmountMouthNum = () => {
  getAmountMouth().then((res) => {
    receiveAmount.value = res.data.receiveAmount
    contractAmountMonth.value = res.data.contractAmount
    const percentage = (receiveAmount.value / contractAmountMonth.value) * 100;
    receiveAmountPercentage.value = percentage.toFixed(2)
    materialPieSeries.value[0].data[0].value = receiveAmount.value
    materialPieSeries.value[0].data[1].value = contractAmountMonth.value
  })
}
// 月付款金额饼图
const paymentMonthListNum = () => {
  paymentMonthList().then((res) => {
    paymentAmount.value = res.data.paymentAmount
    payableAmount.value = res.data.payableAmount
    const percentage = (paymentAmount.value / payableAmount.value) * 100;
    payableAmountPercentage.value = percentage.toFixed(2)
    materialPieSeries1.value[0].data[0].value = paymentAmount.value
    materialPieSeries1.value[0].data[1].value = payableAmount.value
  })
}
// 客户top5
const getTopFiveListNum = async () => {
  const res = await getTopFiveList()
  const customerName = []
  const totalAmount = []
  res.data.forEach(item => {
    customerName.push(item.customerName)
    totalAmount.push(item.totalAmount)
  })
  // 正确响应式赋值:创建新的 xAxis 和 series 对象
  xAxis1.value = [
    {
      type: 'category',
      data: customerName
    }
  ]
  barSeries.value = [
    {
      type: 'bar',
      data: totalAmount,
      itemStyle: {
        color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
          { offset: 0, color: '#F7D2FF' },
          { offset: 1, color: '#826AF9' }
        ])
      },
      emphasis: {
        itemStyle: {
          color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
            { offset: 1, color: '#826AF9' }
          ])
        }
      },
    }
  ]
}
// 线形图
const getAmountHalfYearNum = async () => {
  const res = await getAmountHalfYear()
  console.log(res)
  const monthName = []
  const receiptAmount = []
  const invoiceAmount = []
  res.data.forEach(item => {
    monthName.push(item.month)
    receiptAmount.push(item.receiptAmount)
    invoiceAmount.push(item.invoiceAmount)
  })
  // 正确响应式赋值:创建新的 xAxis 和 series 对象
  xAxis2.value = [
    {
      type: 'category',
      data: monthName
    }
  ]
  lineSeries.value = [
    {
      name: '开票',
      type: 'line',
      data: receiptAmount,
      smooth: true,
      stack: 'Total',
      areaStyle: {
        color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
          {
            offset: 0,
            color: 'rgba(131, 207, 255, 1)'
          },
          {
            offset: 1,
            color: 'rgba(186, 228, 255, 1)'
          }
        ])
      },
      // 设置小圆点的颜色
      itemStyle: {
        color: '#2D99FF', // 小圆点颜色设置为#2D99FF
        borderColor: '#2D99FF' // 如果需要的话,可以设置边框颜色
      },
      emphasis: {
        focus: 'series'
      },
      lineStyle: {
        width: 0
      },
      showSymbol: false,
    },
    {
      name: '回款',
      type: 'line',
      data: invoiceAmount,
      smooth: true,
      stack: 'Total',
      lineStyle: {
        width: 0
      },
      // 设置小圆点的颜色
      itemStyle: {
        color: '#83CFFF', // 小圆点颜色设置为#83CFFF
        borderColor: '#83CFFF' // 如果需要的话,可以设置边框颜色
      },
      showSymbol: false,
      areaStyle: {
        color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
          {
            offset: 0,
            color: 'rgba(54, 153, 255, 1)'
          },
          {
            offset: 1,
            color: 'rgba(89, 169, 254, 1)'
          }
        ])
      },
      emphasis: {
        focus: 'series'
      },
    }
  ]
}
getContractAmountNum()
getInvoiceAmountNum()
getReceiptAmountNum()
getTopFiveListNum()
getAmountMouthNum()
paymentMonthListNum()
getAmountHalfYearNum()
</script>
<style scoped>
.card-top-left {
  padding: 16px;
  background: #fff;
  height: 24vh;
  width: 56vw;
  margin-bottom: 20px;
}
.card-top-right {
  padding: 16px;
  background: #fff;
  height: 50.6vh;
  width: 28vw;
  margin-bottom: 20px;
  margin-left: 20px;
}
.card-bottom {
  padding: 16px;
  background: #fff;
  height: 34vh;
  width: 85.2vw;
  margin-bottom: 20px;
}
.title {
  position: relative;
  font-size: 18px;
  color: #333;
  font-weight: 400;
  padding-left: 10px;
  margin-bottom: 26px;
}
.title::before {
  position: absolute;
  left: 0;
  top: 4px;
  content: '';
  width: 4px;
  height: 18px;
  background-color: #002FA7;
  border-radius: 2px;
}
.card-group {
  display: flex;
}
.info-card {
  width: 300px;
  height: 126px;
  background-image: url("../assets/images/Rectangle 76@2x.png");
  background-size: 100% 100%;
  display: flex;
  justify-content: space-around;
  align-items: center;
}
.info-card1 {
  width: 300px;
  height: 126px;
  background-image: url("../assets/images/Rectangle 77@2x.png");
  background-size: 100% 100%;
  margin: 0 40px;
  display: flex;
  justify-content: space-around;
  align-items: center;
}
.info-card2 {
  width: 300px;
  height: 126px;
  background-image: url("../assets/images/Rectangle 77@2x(1).png");
  background-size: 100% 100%;
  display: flex;
  justify-content: space-around;
  align-items: center;
}
.info-message {
  display: flex;
  flex-direction: column;
  justify-content: center;
  align-items: center;
}
.info-message1 {
  font-weight: bold;
  display: flex;
  flex-direction: column;
  justify-content: center;
  align-items: flex-start;
}
.info-message2 {
  font-weight: bold;
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: space-around;
}
.info-number {
  font-weight: bold;
  font-size: 32px;
  color: #FFFFFF;
  margin-bottom: 10px;
}
.info-title {
  font-weight: bold;
  font-size: 18px;
  color: #FFFFFF;
}
.pie {
  display: flex;
  flex-direction: row;
  justify-content: space-around;
}
.pie-group {
  display: flex;
}
.pie-title {
  font-size: 14px;
  line-height: 24px;
  color: #2853FD;
  padding-left: 16px;
  position: relative;
}
.pie-title::before {
  content: '';
  width: 6px;
  /* 蓝点的宽度 */
  height: 6px;
  /* 蓝点的高度 */
  background-color: #2853FD;
  /* 蓝点的颜色 */
  border-radius: 50%;
  /* 将正方形变为圆形 */
  position: absolute;
  left: 0;
  /* 定位到左边 */
  top: 9px;
  /* 垂直居中对齐,根据行高调整 */
}
.pie-title1 {
  font-size: 14px;
  line-height: 24px;
  color: #1EBFAC;
  padding-left: 16px;
  position: relative;
}
.pie-title1::before {
  content: '';
  width: 6px;
  /* 蓝点的宽度 */
  height: 6px;
  /* 蓝点的高度 */
  background-color: #1EBFAC;
  /* 蓝点的颜色 */
  border-radius: 50%;
  /* 将正方形变为圆形 */
  position: absolute;
  left: 0;
  /* 定位到左边 */
  top: 9px;
  /* 垂直居中对齐,根据行高调整 */
}
.pie-info {
  padding-left: 16px;
  font-size: 14px;
  line-height: 24px;
}
.pie-number {
  color: #2853FD;
}
.pie-number1 {
  color: #1EBFAC;
}
</style>
<template>
  <div class="home-page">
    <div class="top-bar">
      <div class="user-box">
        <img :src="userStore.avatar" class="avatar" alt="" />
        <div>
          <div class="hello">{{ userStore.roleName || "系统管理员" }},你好</div>
          <div class="sub">登录时间:{{ userStore.currentLoginTime }}</div>
        </div>
      </div>
      <div class="top-actions">
        <span class="refresh-time">数据更新时间:{{ lastUpdatedAt || "-" }}</span>
        <el-button size="small" type="primary" plain @click="refreshDashboardData">刷新数据</el-button>
        <el-button size="small" plain @click="configDialogVisible = true">首页配置</el-button>
      </div>
    </div>
    <div class="content-grid">
      <div class="left-col">
        <section class="section-card">
          <div class="section-title-row">
            <div class="section-title">快捷操作</div>
            <el-button size="small" type="primary" link @click="openShortcutDialog">添加快捷入口</el-button>
          </div>
          <div class="quick-grid">
            <el-button
              v-for="item in shortcuts"
              :key="`${item.label}-${item.path}`"
              :type="item.invalid ? 'danger' : 'default'"
              @click="goTo(item.path)"
            >
              {{ item.label }}
            </el-button>
          </div>
        </section>
        <section class="section-card">
          <div class="section-title">重点待办</div>
          <div class="todo-row" v-for="todo in todos" :key="todo.title">
            <el-tag size="small" :type="todo.type">{{ todo.level }}</el-tag>
            <span>{{ todo.title }}</span>
          </div>
        </section>
        <section class="section-card">
          <div class="section-title">经营关注</div>
          <div class="focus-row" v-for="item in businessFocus" :key="item.name">
            <span class="focus-name">{{ item.name }}</span>
            <span class="focus-value">{{ item.value }}</span>
          </div>
        </section>
        <section class="section-card flex-fill-card">
          <div class="section-title-row">
            <div class="section-title">今日待处理</div>
            <el-radio-group v-model="pendingFilter" size="small">
              <el-radio-button label="all">全部</el-radio-button>
              <el-radio-button label="mine">我的</el-radio-button>
              <el-radio-button label="high">高优先</el-radio-button>
            </el-radio-group>
          </div>
          <div class="task-row" v-for="task in filteredPendingTasks" :key="task.id">
            <div class="task-left">
              <el-tag size="small" :type="task.type">{{ task.level }}</el-tag>
              <span class="task-title">{{ task.title }}</span>
            </div>
            <el-button link type="primary" @click="goTo(task.path)">去处理</el-button>
          </div>
          <el-empty v-if="filteredPendingTasks.length === 0" description="暂无待处理事项" :image-size="80" />
        </section>
      </div>
      <div class="right-col">
        <section class="section-card" v-if="isSectionVisible('trendCards')">
          <div class="section-title">最近7天关键指标趋势</div>
          <div class="trend-cards">
            <div class="trend-card clickable" v-for="card in recentTrendCards" :key="card.key" @click="handleTrendCardClick(card)">
              <div class="trend-head">
                <span class="trend-label">{{ card.label }}</span>
                <span class="trend-rate" :class="trendClass(card.change)">
                  {{ card.change > 0 ? "+" : "" }}{{ card.change.toFixed(1) }}%
                </span>
              </div>
              <div class="trend-value">{{ card.latest }} {{ card.unit }}</div>
              <div class="sparkline">
                <span
                  v-for="(v, idx) in card.values"
                  :key="`${card.key}-${idx}`"
                  class="sparkline-bar"
                  :style="{ height: `${calcBarHeight(v, card.values)}%` }"
                />
              </div>
            </div>
          </div>
        </section>
        <section class="section-card" v-if="isSectionVisible('planTrend')">
          <div class="section-title-row">
            <div class="section-title">计划与生产趋势</div>
            <el-radio-group v-model="chartRangePlan" size="small" @change="loadPlanTrend">
              <el-radio-button :label="1">日</el-radio-button>
              <el-radio-button :label="2">周</el-radio-button>
              <el-radio-button :label="3">月</el-radio-button>
            </el-radio-group>
          </div>
          <Echarts
            :chartStyle="chartStyle"
            :legend="planLegend"
            :grid="grid"
            :tooltip="lineTooltip"
            :xAxis="planXAxis"
            :yAxis="valueYAxis"
            :series="planSeries"
            style="height: 300px"
          />
        </section>
        <div class="row-two" v-if="isSectionVisible('qualityChart') || isSectionVisible('costChart')">
          <section class="section-card" v-if="isSectionVisible('qualityChart')">
            <div class="section-title-row">
              <div class="section-title">质检异常分布</div>
              <el-radio-group v-model="chartRangeQuality" size="small" @change="loadQualityData">
                <el-radio-button :label="1">周</el-radio-button>
                <el-radio-button :label="2">月</el-radio-button>
                <el-radio-button :label="3">季度</el-radio-button>
              </el-radio-group>
            </div>
            <Echarts
              :chartStyle="chartStyle"
              :grid="grid"
              :tooltip="barTooltip"
              :xAxis="qualityXAxis"
              :yAxis="valueYAxis"
              :series="qualitySeries"
              style="height: 260px"
            />
          </section>
          <section class="section-card" v-if="isSectionVisible('costChart')">
            <div class="section-title">能耗与成本结构</div>
            <Echarts
              :chartStyle="chartStyle"
              :legend="costLegend"
              :tooltip="pieTooltip"
              :series="costSeries"
              style="height: 260px"
            />
          </section>
        </div>
        <section class="section-card" v-if="isSectionVisible('warningCenter')">
          <div class="section-title">异常预警中心</div>
          <div class="warning-row" v-for="item in warningList" :key="item.id">
            <div class="warning-left">
              <el-tag size="small" :type="item.levelType">{{ item.levelText }}</el-tag>
              <span class="warning-title">{{ item.title }}</span>
            </div>
            <el-button link type="danger" @click="goTo(item.path)">处理</el-button>
          </div>
          <el-empty v-if="warningList.length === 0" description="暂无异常预警" :image-size="80" />
        </section>
        <section class="section-card mini-table-wrap" v-if="isSectionVisible('planTable')">
          <div class="section-title">生产计划执行明细</div>
          <el-table :data="planTable" size="small" stripe>
            <el-table-column prop="planNo" label="计划单号" min-width="150" />
            <el-table-column prop="product" label="产品" min-width="120" />
            <el-table-column prop="qty" label="计划量" min-width="90" />
            <el-table-column prop="issued" label="已下发" min-width="90" />
            <el-table-column prop="status" label="状态" min-width="100" />
            <el-table-column label="操作" min-width="120">
              <template #default="{ row }">
                <el-button link type="primary" @click="goTo(routePathMap.plan)">查看</el-button>
                <el-button
                  v-if="row.status !== '已完成'"
                  link
                  type="success"
                  @click="goTo(routePathMap.dispatch)"
                >
                  下发
                </el-button>
              </template>
            </el-table-column>
          </el-table>
        </section>
      </div>
    </div>
    <el-dialog v-model="shortcutDialogVisible" title="添加快捷入口(最多6个)" width="680px">
      <div class="shortcut-form-row">
        <el-tree-select
          v-model="selectedPagePath"
          placeholder="请选择页面(目录不可选)"
          filterable
          clearable
          check-strictly
          node-key="value"
          :data="menuTreeOptions"
          :props="{ label: 'label', value: 'value', children: 'children', disabled: 'disabled' }"
          style="grid-column: span 2"
        />
        <el-button type="success" @click="addShortcutBySelect">选择添加</el-button>
      </div>
      <el-table :data="shortcuts" size="small" border>
        <el-table-column prop="label" label="名称" min-width="220" />
        <el-table-column label="状态" min-width="80">
          <template #default="{ row }">
            <el-tag size="small" :type="row.invalid ? 'danger' : 'success'">{{ row.invalid ? "无效" : "有效" }}</el-tag>
          </template>
        </el-table-column>
        <el-table-column label="操作" min-width="90" align="center">
          <template #default="{ $index }">
            <el-button link type="danger" @click="removeShortcut($index)">删除</el-button>
          </template>
        </el-table-column>
      </el-table>
    </el-dialog>
    <el-dialog v-model="configDialogVisible" title="首页模块配置" width="520px">
      <el-checkbox-group v-model="enabledSectionKeys" class="config-check-group">
        <el-checkbox v-for="item in sectionConfigOptions" :key="item.key" :label="item.key">
          {{ item.label }}
        </el-checkbox>
      </el-checkbox-group>
      <template #footer>
        <el-button @click="configDialogVisible = false">取消</el-button>
        <el-button type="primary" @click="saveSectionConfig">保存</el-button>
      </template>
    </el-dialog>
  </div>
</template>
<script setup>
import { computed, onMounted, reactive, ref } from "vue";
import { useRouter } from "vue-router";
import { ElMessage } from "element-plus";
import Echarts from "@/components/Echarts/echarts.vue";
import useUserStore from "@/store/modules/user.js";
import usePermissionStore from "@/store/modules/permission";
import {
  expenseCompositionAnalysis,
  getProgressStatistics,
  homeTodos,
  orderCount,
  processDataProductionStatistics,
  qualityInspectionStatistics,
  nonComplianceWarning,
} from "@/api/viewIndex.js";
const router = useRouter();
const userStore = useUserStore();
const permissionStore = usePermissionStore();
const SHORTCUT_STORAGE_KEY = "home-shortcuts-v1";
const defaultShortcuts = [
  { label: "主生产计划", path: "/productionManagement/productionPlan" },
  { label: "生产订单", path: "/productionManagement/productionOrder" },
  { label: "生产报工", path: "/productionManagement/productionReporting" },
  { label: "过程检", path: "/qualityManagement/processInspection" },
  { label: "生产能耗", path: "/energyManagement/productionEnergyConsumption" },
  { label: "生产成本", path: "/costAccounting/productionCostAccounting" },
  { label: "标准vs实际", path: "/costAccounting/stdVsActCostAnalysis" },
  { label: "决策分析", path: "/reportAnalysis/dataDashboard" },
];
const isRouteValid = (path) => {
  if (!path || !path.startsWith("/")) return false;
  const resolved = router.resolve(path);
  return resolved.matched && resolved.matched.length > 0;
};
const withValidFlag = (list) =>
  list.map((item) => ({
    ...item,
    invalid: !isRouteValid(item.path),
  }));
const pageOptions = router
  .getRoutes()
  .filter((route) => {
    const hasTitle = Boolean(route.meta?.title);
    const validPath = route.path && route.path.startsWith("/") && !route.path.includes(":");
    const isVisibleMenu = !route.meta?.hidden && route.path !== "/index";
    const notSpecial =
      !route.path.includes("redirect") &&
      route.path !== "/login" &&
      route.path !== "/register" &&
      route.path !== "/401" &&
      !route.path.includes(":pathMatch");
    return hasTitle && validPath && isVisibleMenu && notSpecial;
  })
  .map((route) => ({
    title: route.meta.title,
    path: route.path,
  }))
  .sort((a, b) => a.path.localeCompare(b.path));
const normalizePath = (path) => String(path || "").replace(/\/+/g, "/");
const joinPath = (parentPath, childPath) => {
  if (!childPath) return normalizePath(parentPath || "/");
  if (String(childPath).startsWith("/")) return normalizePath(childPath);
  return normalizePath(`${parentPath || ""}/${childPath}`);
};
const buildMenuTreeOptions = (routes = [], parentPath = "") => {
  const result = [];
  routes.forEach((route) => {
    if (!route || route.hidden) return;
    const fullPath = joinPath(parentPath, route.path);
    const children = buildMenuTreeOptions(route.children || [], fullPath);
    const title = route.meta?.title;
    if (!title && children.length > 0) {
      result.push(...children);
      return;
    }
    if (!title) return;
    result.push({
      label: title,
      value: fullPath,
      disabled: children.length > 0,
      children,
    });
  });
  return result;
};
const menuTreeOptions = computed(() => buildMenuTreeOptions(permissionStore.sidebarRouters || []));
const selectableMenuMap = computed(() => {
  const map = new Map();
  const walk = (list = []) => {
    list.forEach((item) => {
      if (!item.disabled) map.set(item.value, item.label);
      if (item.children?.length) walk(item.children);
    });
  };
  walk(menuTreeOptions.value);
  return map;
});
const keywordMap = {
  "主生产计划": ["生产计划", "productionPlan"],
  "生产订单": ["生产订单", "productionOrder"],
  "生产报工": ["报工", "productionReporting"],
  "过程检": ["过程检", "processInspection"],
  "生产能耗": ["生产能耗", "productionEnergyConsumption"],
  "生产成本": ["生产成本", "productionCostAccounting"],
  "标准vs实际": ["标准", "实际", "stdVsActCostAnalysis"],
  "决策分析": ["决策", "看板", "dataDashboard"],
};
const findRouteByKeywords = (keywords = []) => {
  const lowerKeywords = keywords.map((k) => String(k).toLowerCase());
  return pageOptions.find((item) => {
    const title = String(item.title || "").toLowerCase();
    const path = String(item.path || "").toLowerCase();
    return lowerKeywords.some((k) => title.includes(k) || path.includes(k));
  });
};
const getPathByKeywords = (keywords = []) => findRouteByKeywords(keywords)?.path || "";
const getRecommendedShortcuts = () => {
  const list = defaultShortcuts
    .map((item) => {
      const matched = findRouteByKeywords(keywordMap[item.label] || [item.label]);
      return matched ? { label: item.label, path: matched.path } : null;
    })
    .filter(Boolean);
  return list.length > 0 ? list : defaultShortcuts;
};
const tryRepairSavedShortcut = (item) => {
  const matched = findRouteByKeywords(keywordMap[item.label] || [item.label]);
  if (matched) return { label: item.label, path: matched.path };
  return item;
};
const getSavedShortcuts = () => {
  const recommended = getRecommendedShortcuts();
  try {
    const saved = localStorage.getItem(SHORTCUT_STORAGE_KEY);
    if (!saved) return recommended;
    const parsed = JSON.parse(saved);
    if (!Array.isArray(parsed) || parsed.length === 0) return recommended;
    return parsed.map((item) => tryRepairSavedShortcut(item));
  } catch (error) {
    return recommended;
  }
};
const shortcuts = reactive(withValidFlag(getSavedShortcuts().slice(0, 6)));
const shortcutDialogVisible = ref(false);
const configDialogVisible = ref(false);
const selectedPagePath = ref("");
const lastUpdatedAt = ref("");
const pendingFilter = ref("all");
const chartRangePlan = ref(3);
const chartRangeQuality = ref(2);
const routePathMap = {
  plan: getPathByKeywords(["生产计划", "productionPlan"]),
  order: getPathByKeywords(["生产订单", "productionOrder"]),
  processInspection: getPathByKeywords(["过程检", "processInspection"]),
  meter: getPathByKeywords(["抄表", "meterCollection", "能耗"]),
  dispatch: getPathByKeywords(["生产调度", "productionDispatching"]),
};
const persistShortcuts = () => {
  localStorage.setItem(
    SHORTCUT_STORAGE_KEY,
    JSON.stringify(shortcuts.slice(0, 6).map(({ label, path }) => ({ label, path })))
  );
};
const todos = reactive([]);
const businessFocus = reactive([
  { name: "生产订单总数", value: "-" },
  { name: "已完成订单数", value: "-" },
  { name: "未完成订单数", value: "-" },
  { name: "部分完成订单数", value: "-" },
  { name: "质检总数", value: "-" },
  { name: "过程检总数", value: "-" },
]);
const pendingTasks = reactive([]);
const warningList = reactive([]);
const SECTION_CONFIG_KEY = "home-sections-v1";
const sectionConfigOptions = [
  { key: "trendCards", label: "最近7天趋势卡" },
  { key: "planTrend", label: "计划与生产趋势图" },
  { key: "qualityChart", label: "质检异常分布图" },
  { key: "costChart", label: "能耗与成本结构图" },
  { key: "warningCenter", label: "异常预警中心" },
  { key: "planTable", label: "生产计划执行明细表" },
];
const enabledSectionKeys = ref(sectionConfigOptions.map((i) => i.key));
const chartStyle = { width: "100%", height: "100%" };
const grid = { left: "3%", right: "4%", bottom: "3%", containLabel: true };
const lineTooltip = { trigger: "axis" };
const barTooltip = { trigger: "axis", axisPointer: { type: "shadow" } };
const pieTooltip = { trigger: "item" };
const valueYAxis = [{ type: "value" }];
const planXAxis = [{ type: "category", data: [] }];
const qualityXAxis = [{ type: "category", data: [] }];
const planLegend = { show: true, data: ["计划量", "下发量", "完成量"] };
const costLegend = {
  show: true,
  orient: "vertical",
  right: 10,
  top: "center",
  data: ["能耗成本", "生产成本", "质量损失成本", "其他成本"],
};
const planSeries = reactive([
  { name: "计划量", type: "line", smooth: true, data: [] },
  { name: "下发量", type: "line", smooth: true, data: [] },
  { name: "完成量", type: "line", smooth: true, data: [] },
]);
const qualitySeries = reactive([
  {
    name: "异常数",
    type: "bar",
    barWidth: 26,
    itemStyle: { color: "#e67e22", borderRadius: [6, 6, 0, 0] },
    data: [],
  },
]);
const costSeries = reactive([
  {
    type: "pie",
    radius: ["45%", "68%"],
    center: ["35%", "50%"],
    label: { formatter: "{b}: {d}%" },
    data: [],
  },
]);
const planTable = reactive([]);
const recentTrendCards = reactive([
  { key: "planIssued", label: "计划下发量", unit: "单", values: [0, 0, 0, 0, 0, 0, 0], latest: 0, change: 0 },
  { key: "qualityRaw", label: "来料检数量", unit: "条", values: [0, 0, 0, 0, 0, 0, 0], latest: 0, change: 0 },
  { key: "qualityProcess", label: "过程检数量", unit: "条", values: [0, 0, 0, 0, 0, 0, 0], latest: 0, change: 0 },
  { key: "qualityFactory", label: "成品检数量", unit: "条", values: [0, 0, 0, 0, 0, 0, 0], latest: 0, change: 0 },
]);
const toNumber = (value) => {
  const num = Number(value);
  return Number.isFinite(num) ? num : 0;
};
const pickFirstNumber = (obj, keys = []) => {
  for (const key of keys) {
    if (obj && obj[key] !== undefined && obj[key] !== null) return toNumber(obj[key]);
  }
  return 0;
};
const updateArray = (target, list) => {
  target.splice(0, target.length, ...list);
};
const toFixedOne = (num) => Number(num || 0).toFixed(1);
const normalizeSeven = (list = []) => {
  const nums = list.map((i) => toNumber(i));
  if (nums.length >= 7) return nums.slice(-7);
  return [...Array(7 - nums.length).fill(0), ...nums];
};
const calcTrend = (list = []) => {
  if (!Array.isArray(list) || list.length === 0) return { latest: 0, change: 0 };
  const first = toNumber(list[0]);
  const latest = toNumber(list[list.length - 1]);
  if (first === 0) return { latest, change: latest > 0 ? 100 : 0 };
  return { latest, change: ((latest - first) / first) * 100 };
};
const setTrendCard = (key, values) => {
  const target = recentTrendCards.find((i) => i.key === key);
  if (!target) return;
  const series = normalizeSeven(values);
  const { latest, change } = calcTrend(series);
  target.values = series;
  target.latest = latest;
  target.change = Number(toFixedOne(change));
};
const trendClass = (change) => (change > 0 ? "up" : change < 0 ? "down" : "flat");
const calcBarHeight = (value, list) => {
  const max = Math.max(...list, 1);
  return Math.max(18, Math.round((toNumber(value) / max) * 100));
};
const filteredPendingTasks = computed(() => {
  if (pendingFilter.value === "high") return pendingTasks.filter((i) => i.level === "高");
  if (pendingFilter.value === "mine") {
    const currentUserName = String(userStore?.name || "").toLowerCase();
    const currentUserId = String(userStore?.userId || "");
    return pendingTasks.filter((i) => {
      const ownerName = String(i.ownerName || "").toLowerCase();
      const ownerId = String(i.ownerId || "");
      return (currentUserName && ownerName && ownerName.includes(currentUserName)) || (currentUserId && ownerId === currentUserId);
    });
  }
  return pendingTasks;
});
const isSectionVisible = (key) => enabledSectionKeys.value.includes(key);
const goTo = (path) => {
  if (!isRouteValid(path)) {
    ElMessage.warning("当前菜单未配置该页面或无访问权限");
    return;
  }
  router.push(path);
};
const handleTrendCardClick = (card) => {
  const mapping = {
    planIssued: routePathMap.plan || routePathMap.order,
    qualityRaw: routePathMap.processInspection,
    qualityProcess: routePathMap.processInspection,
    qualityFactory: routePathMap.processInspection,
  };
  const target = mapping[card.key];
  if (!target) {
    ElMessage.warning("未配置可跳转页面");
    return;
  }
  const query =
    card.key === "planIssued"
      ? { dateType: String(chartRangePlan.value), source: "homeTrend" }
      : { dateType: String(chartRangeQuality.value), source: "homeTrend" };
  router.push({ path: target, query });
};
const openShortcutDialog = () => {
  shortcutDialogVisible.value = true;
};
const addShortcutBySelect = () => {
  if (shortcuts.length >= 6) {
    ElMessage.warning("快捷入口最多只能添加6个");
    return;
  }
  if (!selectedPagePath.value) {
    ElMessage.warning("请先选择页面");
    return;
  }
  if (shortcuts.some((item) => item.path === selectedPagePath.value)) {
    ElMessage.warning("该快捷入口已存在");
    return;
  }
  const label = selectableMenuMap.value.get(selectedPagePath.value);
  if (!label) {
    ElMessage.warning("请选择可添加的页面,目录节点不可选");
    return;
  }
  shortcuts.push({
    label,
    path: selectedPagePath.value,
    invalid: !isRouteValid(selectedPagePath.value),
  });
  persistShortcuts();
  selectedPagePath.value = "";
};
const removeShortcut = (index) => {
  shortcuts.splice(index, 1);
  persistShortcuts();
  ElMessage.success("已删除快捷入口");
};
const loadHomeTodos = async () => {
  try {
    const res = await homeTodos();
    const list = Array.isArray(res?.data) ? res.data : [];
    const mapped = list.slice(0, 4).map((item, idx) => {
      const text = item?.approveReason || item?.approveTypeName || `待处理事项 ${idx + 1}`;
      const levelType = idx === 0 ? "danger" : idx <= 2 ? "warning" : "success";
      const level = idx === 0 ? "高" : idx <= 2 ? "中" : "低";
      return { level, title: text, type: levelType };
    });
    updateArray(todos, mapped);
    const pendingMapped = list.slice(0, 4).map((item, idx) => {
      const title = item?.approveReason || item?.approveTypeName || `待处理事项 ${idx + 1}`;
      const path = inferTodoPath(item);
      return {
        id: item?.id || `${idx}-${title}`,
        title,
        level: idx === 0 ? "高" : idx <= 2 ? "中" : "低",
        type: idx === 0 ? "danger" : idx <= 2 ? "warning" : "success",
        path,
        ownerId: item?.approveUserId || item?.userId || "",
        ownerName: item?.approveUserName || item?.userName || "",
      };
    });
    updateArray(pendingTasks, pendingMapped);
  } catch (error) {
    console.error("homeTodos接口获取失败:", error);
  }
};
const loadOrderAndProgress = async () => {
  try {
    const [orderRes, progressRes] = await Promise.allSettled([orderCount(), getProgressStatistics()]);
    if (orderRes.status === "fulfilled") {
      const items = Array.isArray(orderRes.value?.data) ? orderRes.value.data : [];
      const byName = Object.fromEntries(
        items.map((i) => [String(i?.name || "").replace(/\s/g, ""), i?.value])
      );
      businessFocus[0].value = `${pickFirstNumber(byName, ["生产订单数", "生产订单总数", "总订单数"]) || 0} 单`;
      businessFocus[1].value = `${pickFirstNumber(byName, ["已完成订单数"]) || 0} 单`;
      businessFocus[2].value = `${pickFirstNumber(byName, ["待生产订单数", "未完成订单数"]) || 0} 单`;
      businessFocus[3].value = `${pickFirstNumber(byName, ["部分完成订单数"]) || 0} 单`;
    }
    if (progressRes.status === "fulfilled") {
      const p = progressRes.value?.data || {};
      const detail = Array.isArray(p.completedOrderDetails) ? p.completedOrderDetails : [];
      const rows = detail.slice(0, 6).map((item, index) => {
        const qty = pickFirstNumber(item, ["quantity", "planQuantity"]);
        const done = pickFirstNumber(item, ["completeQuantity", "completedQuantity"]);
        return {
          planNo: item.npsNo || item.productionPlanNo || `NO-${index + 1}`,
          product: item.productCategory || item.productName || "-",
          qty,
          issued: done,
          status: qty > 0 && done >= qty ? "已完成" : done > 0 ? "执行中" : "待下发",
        };
      });
      updateArray(planTable, rows);
      setTrendCard(
        "planIssued",
        detail.slice(-7).map((i) => pickFirstNumber(i, ["completeQuantity", "completedQuantity", "issueNum"]))
      );
    }
  } catch (error) {
    console.error("orderCount/getProgressStatistics接口获取失败:", error);
  }
};
const inferTodoPath = (todo) => {
  const text = `${todo?.approveTypeName || ""}${todo?.approveReason || ""}`.toLowerCase();
  if (text.includes("计划")) return routePathMap.plan || routePathMap.order;
  if (text.includes("订单")) return routePathMap.order || routePathMap.plan;
  if (text.includes("过程检") || text.includes("质检")) return routePathMap.processInspection || routePathMap.plan;
  if (text.includes("能耗") || text.includes("抄表")) return routePathMap.meter || routePathMap.plan;
  return routePathMap.plan || routePathMap.order || "";
};
const loadPlanTrend = async () => {
  try {
    const res = await processDataProductionStatistics({ type: chartRangePlan.value });
    const list = Array.isArray(res?.data) ? res.data : [];
    planXAxis[0].data = list.map((i, index) => i.processName || `工序${index + 1}`);
    planSeries[0].data = list.map((i) => pickFirstNumber(i, ["totalInput", "input", "planNum"]));
    planSeries[1].data = list.map((i) => pickFirstNumber(i, ["totalOutput", "output", "issueNum"]));
    planSeries[2].data = list.map((i) => pickFirstNumber(i, ["totalScrap", "scrap", "completeNum"]));
  } catch (error) {
    console.error("processDataProductionStatistics接口获取失败:", error);
  }
};
const loadQualityData = async () => {
  try {
    const res = await qualityInspectionStatistics({ type: chartRangeQuality.value });
    const data = res?.data || {};
    const items = Array.isArray(data.item) ? data.item : [];
    if (items.length > 0) {
      qualityXAxis[0].data = items.map((i) => i.date || i.name || "-");
      qualitySeries[0].data = items.map((i) =>
        pickFirstNumber(i, ["supplierNum", "processNum", "factoryNum", "totalNum"])
      );
      setTrendCard("qualityRaw", items.map((i) => pickFirstNumber(i, ["supplierNum"])));
      setTrendCard("qualityProcess", items.map((i) => pickFirstNumber(i, ["processNum"])));
      setTrendCard("qualityFactory", items.map((i) => pickFirstNumber(i, ["factoryNum"])));
    } else {
      qualityXAxis[0].data = ["来料检", "过程检", "成品检"];
      qualitySeries[0].data = [
        pickFirstNumber(data, ["supplierNum"]),
        pickFirstNumber(data, ["processNum"]),
        pickFirstNumber(data, ["factoryNum"]),
      ];
      setTrendCard("qualityRaw", [pickFirstNumber(data, ["supplierNum"])]);
      setTrendCard("qualityProcess", [pickFirstNumber(data, ["processNum"])]);
      setTrendCard("qualityFactory", [pickFirstNumber(data, ["factoryNum"])]);
    }
    businessFocus[4].value = `${pickFirstNumber(data, ["supplierNum", "totalNum"])} 条`;
    businessFocus[5].value = `${pickFirstNumber(data, ["processNum"])} 条`;
  } catch (error) {
    console.error("qualityInspectionStatistics接口获取失败:", error);
  }
};
const loadWarningCenter = async () => {
  try {
    const res = await nonComplianceWarning();
    const list = Array.isArray(res?.data) ? res.data : [];
    const mapped = list.slice(0, 6).map((item, idx) => {
      const levelNum = toNumber(item.level ?? item.warningLevel ?? 2);
      const levelType = levelNum >= 3 ? "danger" : levelNum === 2 ? "warning" : "info";
      const levelText = levelNum >= 3 ? "高" : levelNum === 2 ? "中" : "低";
      const title = item.name || item.title || item.paramName || `异常预警 ${idx + 1}`;
      const text = `${title}${item.processName || ""}${item.orderNo || ""}`.toLowerCase();
      const path = text.includes("质检")
        ? routePathMap.processInspection
        : text.includes("订单")
          ? routePathMap.order
          : routePathMap.processInspection || routePathMap.order || routePathMap.plan;
      return { id: item.id || `${idx}-${title}`, levelType, levelText, title, path };
    });
    updateArray(warningList, mapped);
  } catch (error) {
    console.error("nonComplianceWarning接口获取失败:", error);
    updateArray(warningList, []);
  }
};
const initSectionConfig = () => {
  try {
    const raw = localStorage.getItem(SECTION_CONFIG_KEY);
    if (!raw) return;
    const parsed = JSON.parse(raw);
    if (Array.isArray(parsed) && parsed.length > 0) {
      enabledSectionKeys.value = parsed.filter((k) => sectionConfigOptions.some((i) => i.key === k));
    }
  } catch (error) {
    console.error("读取首页配置失败:", error);
  }
};
const saveSectionConfig = () => {
  if (enabledSectionKeys.value.length === 0) {
    ElMessage.warning("至少保留一个模块");
    return;
  }
  localStorage.setItem(SECTION_CONFIG_KEY, JSON.stringify(enabledSectionKeys.value));
  configDialogVisible.value = false;
  ElMessage.success("首页配置已保存");
};
const loadCostComposition = async () => {
  try {
    const res = await expenseCompositionAnalysis({ type: 1 });
    const list = Array.isArray(res?.data) ? res.data : [];
    const mapped = list.map((i) => ({
      name: i.name || "未命名",
      value: pickFirstNumber(i, ["value", "amount", "cost"]),
    }));
    costSeries[0].data = mapped;
  } catch (error) {
    console.error("expenseCompositionAnalysis接口获取失败:", error);
  }
};
const refreshDashboardData = () => {
  loadHomeTodos();
  loadOrderAndProgress();
  loadPlanTrend();
  loadQualityData();
  loadCostComposition();
  loadWarningCenter();
  lastUpdatedAt.value = new Date().toLocaleString();
};
onMounted(() => {
  initSectionConfig();
  refreshDashboardData();
});
</script>
<style scoped>
.home-page {
  min-height: 100vh;
  background: #f5f7fb;
  padding: 20px;
}
.top-bar {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 16px;
  background: #fff;
  border-radius: 12px;
  padding: 16px;
  margin-bottom: 16px;
}
.top-actions {
  display: flex;
  align-items: center;
  gap: 10px;
}
.refresh-time {
  font-size: 12px;
  color: #7b8794;
}
.user-box {
  display: flex;
  align-items: center;
  gap: 12px;
}
.avatar {
  width: 54px;
  height: 54px;
  border-radius: 50%;
  object-fit: cover;
}
.hello {
  font-size: 18px;
  font-weight: 700;
  color: #1f2d3d;
}
.sub {
  margin-top: 4px;
  color: #6b7785;
  font-size: 13px;
}
.content-grid {
  display: grid;
  grid-template-columns: 320px 1fr;
  gap: 16px;
  align-items: stretch;
}
.left-col,
.right-col {
  display: flex;
  flex-direction: column;
}
.section-card {
  background: #fff;
  border-radius: 12px;
  padding: 16px;
  margin-bottom: 16px;
  box-shadow: 0 2px 10px rgba(20, 35, 90, 0.06);
}
.flex-fill-card {
  flex: 1;
}
.section-title {
  position: relative;
  padding-left: 10px;
  margin-bottom: 14px;
  font-size: 16px;
  font-weight: 700;
  color: #243447;
}
.section-title-row {
  display: flex;
  justify-content: space-between;
  align-items: center;
}
.section-title::before {
  content: "";
  position: absolute;
  left: 0;
  top: 4px;
  width: 4px;
  height: 16px;
  border-radius: 2px;
  background: #409eff;
}
.quick-grid {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 10px;
}
.quick-grid :deep(.el-button) {
  margin-left: 0;
}
.shortcut-form-row {
  display: grid;
  grid-template-columns: 1fr 1.5fr auto;
  gap: 10px;
  margin-bottom: 12px;
}
.todo-row {
  display: flex;
  align-items: center;
  gap: 10px;
  margin-bottom: 10px;
  font-size: 13px;
  color: #3b4a5b;
}
.focus-row {
  display: flex;
  justify-content: space-between;
  align-items: center;
  padding: 8px 0;
  border-bottom: 1px dashed #e8edf5;
}
.focus-row:last-child {
  border-bottom: none;
}
.focus-name {
  font-size: 13px;
  color: #516174;
}
.focus-value {
  font-weight: 700;
  color: #1f2d3d;
}
.task-row {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 10px;
  padding: 8px 0;
  border-bottom: 1px dashed #e8edf5;
}
.task-row:last-child {
  border-bottom: none;
}
.task-left {
  display: flex;
  align-items: center;
  gap: 10px;
}
.task-title {
  font-size: 13px;
  color: #3d4d5f;
}
.row-two {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 16px;
}
.trend-cards {
  display: grid;
  grid-template-columns: repeat(4, minmax(0, 1fr));
  gap: 12px;
}
.trend-card {
  border: 1px solid #e8edf5;
  border-radius: 10px;
  padding: 12px;
}
.trend-card.clickable {
  cursor: pointer;
  transition: all 0.2s ease;
}
.trend-card.clickable:hover {
  border-color: #8eb8ff;
  background: #f6f9ff;
}
.trend-head {
  display: flex;
  justify-content: space-between;
  align-items: center;
}
.trend-label {
  font-size: 13px;
  color: #5f6b7a;
}
.trend-rate {
  font-size: 12px;
  font-weight: 700;
}
.trend-rate.up {
  color: #67c23a;
}
.trend-rate.down {
  color: #f56c6c;
}
.trend-rate.flat {
  color: #909399;
}
.trend-value {
  margin-top: 6px;
  font-size: 20px;
  color: #1f2d3d;
  font-weight: 700;
}
.sparkline {
  margin-top: 10px;
  height: 48px;
  display: flex;
  align-items: flex-end;
  gap: 4px;
}
.sparkline-bar {
  flex: 1;
  min-height: 6px;
  border-radius: 3px 3px 0 0;
  background: linear-gradient(180deg, #82b1ff 0%, #409eff 100%);
}
.warning-row {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: 10px;
  padding: 8px 0;
  border-bottom: 1px dashed #e8edf5;
}
.warning-row:last-child {
  border-bottom: none;
}
.warning-left {
  display: flex;
  align-items: center;
  gap: 10px;
}
.warning-title {
  font-size: 13px;
  color: #3d4d5f;
}
.config-check-group {
  display: grid;
  grid-template-columns: repeat(2, minmax(0, 1fr));
  gap: 10px 16px;
}
.mini-table-wrap :deep(.el-table th) {
  background: #f8fbff;
}
@media (max-width: 1100px) {
  .content-grid {
    grid-template-columns: 1fr;
  }
  .row-two {
    grid-template-columns: 1fr;
  }
  .trend-cards {
    grid-template-columns: repeat(2, minmax(0, 1fr));
  }
}
</style>