From 18dba31d39dcb701c16979ed3f607767dbdae80f Mon Sep 17 00:00:00 2001
From: huminmin <mac@MacBook-Pro.local>
Date: 星期四, 02 七月 2026 17:10:10 +0800
Subject: [PATCH] Merge branch 'dev_NEW_pro' of http://114.132.189.42:9002/r/product-inventory-management into dev_NEW_pro

---
 src/views/financialManagement/financialStatements/index.vue |  646 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 645 insertions(+), 1 deletions(-)

diff --git a/src/views/financialManagement/financialStatements/index.vue b/src/views/financialManagement/financialStatements/index.vue
index c272707..5dbce9b 100644
--- a/src/views/financialManagement/financialStatements/index.vue
+++ b/src/views/financialManagement/financialStatements/index.vue
@@ -1,4 +1,648 @@
 <template>
+  <div style="padding: 20px;">
+    <!-- 椤甸潰鏍囬鍜屾湀浠界瓫閫� -->
+    <div class="w-full md:w-auto flex items-center gap-3"
+         style="margin-bottom: 20px;">
+      <el-date-picker v-model="dateRange"
+                      type="monthrange"
+                      format="YYYY-MM"
+                      value-format="YYYY-MM"
+                      range-separator="鑷�"
+                      start-placeholder="寮�濮嬫湀浠�"
+                      end-placeholder="缁撴潫鏈堜唤"
+                      :disabled-date="disabledDate"
+                      @change="handleDateChange"
+                      class="w-full md:w-auto"
+                      style="margin-right: 30px;" />
+      <el-button type="primary"
+                 icon="Refresh"
+                 @click="resetDateRange"
+                 size="default">
+        閲嶇疆
+      </el-button>
+    </div>
+    <main class="container mx-auto px-4 pb-10">
+      <!-- 璐㈠姟鎸囨爣鍗$墖 -->
+      <div class="stats-cards">
+        <div class="stat-card stat-card-blue">
+          <div class="stat-icon"><img src="@/assets/icons/png/walletBlue@2x.png"
+                 alt="鎬昏惀鏀�" /></div>
+          <div class="stat-content">
+            <div class="stat-label">鎬昏惀鏀�</div>
+            <div class="stat-value">{{ formatMoney(pageInfo.totalIncome || 0) }}{{ Math.abs(pageInfo.totalIncome) < 10000 ? ' 鍏�' : '' }}</div>
+          </div>
+        </div>
+        <div class="stat-card stat-card-orange">
+          <div class="stat-icon"><img src="@/assets/icons/png/walletOrange@2x.png"
+                 alt="鎬绘敮鍑�" /></div>
+          <div class="stat-content">
+            <div class="stat-label">鎬绘敮鍑�</div>
+            <div class="stat-value">{{ formatMoney(pageInfo.totalExpense || 0) }}{{ Math.abs(pageInfo.totalExpense) < 10000 ? ' 鍏�' : '' }}</div>
+          </div>
+        </div>
+        <div class="stat-card stat-card-green">
+          <div class="stat-icon"><img src="@/assets/icons/png/walletGreen@2x.png"
+                 alt="搴旀敹璐︽" /></div>
+          <div class="stat-content">
+            <div class="stat-label">搴旀敹璐︽</div>
+            <div class="stat-value">{{ formatMoney(pageInfo.totalReceivable || 0) }}{{ Math.abs(pageInfo.totalReceivable) < 10000 ? ' 鍏�' : '' }}</div>
+          </div>
+        </div>
+        <div class="stat-card stat-card-red">
+          <div class="stat-icon"><img src="@/assets/icons/png/walletRed@2x.png"
+                 alt="搴斾粯璐︽" /></div>
+          <div class="stat-content">
+            <div class="stat-label">搴斾粯璐︽</div>
+            <div class="stat-value">{{ formatMoney(pageInfo.totalPayable || 0) }}{{ Math.abs(pageInfo.totalPayable) < 10000 ? ' 鍏�' : '' }}</div>
+          </div>
+        </div>
+        <div class="stat-card stat-card-yellow">
+          <div class="stat-icon"><img src="@/assets/icons/png/walletYellow@2x.png"
+                 alt="鍑�鍒╂鼎" /></div>
+          <div class="stat-content">
+            <div class="stat-label">鍑�鍒╂鼎</div>
+            <div class="stat-value">{{ formatMoney(pageInfo.netRevenue || 0) }}{{ Math.abs(pageInfo.netRevenue) < 10000 ? ' 鍏�' : '' }}</div>
+          </div>
+        </div>
+      </div>
+      <!-- 鍥捐〃鍖哄煙 -->
+      <div class="charts-row">
+        <!-- 1. 鏀舵敮鏋勬垚鍒嗘瀽 (鍙岀幆褰㈠浘 + 鍑�鍒╀腑蹇�) -->
+        <el-card class="chart-card">
+          <template #header>
+            <div class="card-header">
+              <span class="header-title">鏀舵敮鏋勬垚鍙婂噣鍒╁垎鏋�</span>
+              <el-tooltip content="宸︿晶涓烘敹鍏ユ瀯鎴愶紝鍙充晶涓烘敮鍑烘瀯鎴愶紝涓棿灞曠ず鐩堜簭鍑�棰�"
+                          placement="top">
+                <el-icon>
+                  <QuestionFilled />
+                </el-icon>
+              </el-tooltip>
+            </div>
+          </template>
+          <div class="financial-overview-container">
+            <!-- 鏀跺叆灞曠ず (宸︿晶) -->
+            <div style="width:60%">
+              <div class="overview-item income"
+                   style="margin-bottom: 20px;">
+                <div class="overview-box">
+                  <div class="icon-circle">
+                    <el-icon>
+                      <TrendCharts />
+                    </el-icon>
+                  </div>
+                  <div class="data-content">
+                    <div class="label">鏈湡鎬绘敹鍏�</div>
+                    <div class="value">{{ formatMoney(pageInfo.totalIncome) }}</div>
+                    <div class="unit">RMB{{ Math.abs(pageInfo.totalIncome) < 10000 ? ' / 鍏�' : '' }}</div>
+                  </div>
+                  <div class="bg-decoration">INCOME</div>
+                </div>
+              </div>
+              <div class="overview-item expense">
+                <div class="overview-box">
+                  <div class="icon-circle">
+                    <el-icon>
+                      <Sell />
+                    </el-icon>
+                  </div>
+                  <div class="data-content">
+                    <div class="label">鏈湡鎬绘敮鍑�</div>
+                    <div class="value">{{ formatMoney(pageInfo.totalExpense) }}</div>
+                    <div class="unit">RMB{{ Math.abs(pageInfo.totalExpense) < 10000 ? ' / 鍏�' : '' }}</div>
+                  </div>
+                  <div class="bg-decoration">EXPENSE</div>
+                </div>
+              </div>
+            </div>
+            <!-- 鍑�鍒╂鼎鏍稿績鎸囩ず (涓棿) -->
+            <div class="profit-indicator">
+              <div class="profit-gauge-wrapper">
+                <Echarts :chartStyle="chartStylePie"
+                         :series="profitGaugeSeries"
+                         :tooltip="gaugeTooltip"
+                         style="height: 200px; width: 100%; max-width: 200px;">
+                </Echarts>
+                <div class="profit-center-text">
+                  <div class="label">鍑�鍒╂鼎</div>
+                  <div class="value"
+                       :class="pageInfo.netRevenue >= 0 ? 'plus' : 'minus'">
+                    {{ pageInfo.netRevenue >= 0 ? '+' : '' }}{{ formatMoney(pageInfo.netRevenue) }}
+                  </div>
+                  <div class="rate">鍒╂鼎鐜�: {{ pageInfo.totalIncome > 0 ? ((pageInfo.netRevenue / pageInfo.totalIncome) * 100).toFixed(1) : 0 }}%</div>
+                </div>
+              </div>
+            </div>
+            <!-- 鏀嚭灞曠ず (鍙充晶) -->
+          </div>
+        </el-card>
+        <!-- 2. 搴旀敹/搴斾粯瀵瑰啿鍒嗘瀽 (鏌辩姸鍥�) -->
+        <el-card class="chart-card">
+          <template #header>
+            <div class="card-header">
+              <span class="header-title">搴旀敹/搴斾粯姒傝</span>
+              <el-tooltip content="瀵规瘮褰撳墠鍚勬湀浠界殑搴旀敹璐︽涓庡簲浠樿处娆�"
+                          placement="top">
+                <el-icon>
+                  <QuestionFilled />
+                </el-icon>
+              </el-tooltip>
+            </div>
+          </template>
+          <Echarts :chartStyle="chartStyle"
+                   :grid="barGrid"
+                   :legend="barLegend"
+                   :series="barSeries"
+                   :tooltip="barTooltip"
+                   :xAxis="barXAxis"
+                   :yAxis="barYAxis"
+                   style="height: 270px; width: 100%;">
+          </Echarts>
+        </el-card>
+      </div>
+      <!-- 3. 璐㈠姟缁煎悎瓒嬪娍鍒嗘瀽 (鎶樼嚎鍥�) -->
+      <el-card class="trend-chart-card">
+        <template #header>
+          <div class="card-header">
+            <span class="header-title">璐㈠姟缁╂晥缁煎悎瓒嬪娍</span>
+            <el-tooltip content="灞曠ず鏀跺叆銆佹敮鍑哄強鍑�鍒╂鼎鐨勬湀搴﹀彉鍖栬秼鍔�"
+                        placement="top">
+              <el-icon>
+                <QuestionFilled />
+              </el-icon>
+            </el-tooltip>
+          </div>
+        </template>
+        <Echarts :chartStyle="chartStyle"
+                 :grid="trendGrid"
+                 :legend="trendLegend"
+                 :series="trendSeries"
+                 :tooltip="trendTooltip"
+                 :xAxis="trendXAxis"
+                 :yAxis="trendYAxis"
+                 style="height: 350px; width: 100%;">
+        </Echarts>
+      </el-card>
+    </main>
+  </div>
 </template>
+
 <script setup>
-</script>
\ No newline at end of file
+  import {
+    ref,
+    computed,
+    onMounted,
+    reactive,
+    nextTick,
+    getCurrentInstance,
+  } from "vue";
+  import { QuestionFilled, TrendCharts, Sell } from "@element-plus/icons-vue";
+  import Echarts from "@/components/Echarts/echarts.vue";
+  import { accountStatementDetailsByMonth } from "@/api/financialManagement/financialStatements";
+  import dayjs from "dayjs";
+
+  const { proxy } = getCurrentInstance();
+  const dateRange = ref(null);
+  const pageInfo = reactive({
+    totalIncome: 0,
+    totalExpense: 0,
+    totalReceivable: 0,
+    totalPayable: 0,
+    netRevenue: 0,
+  });
+
+  const chartStyle = { width: "100%", height: "100%", position: "relative" };
+  const chartStylePie = { width: "100%", height: "100%" };
+
+  const monthlyTrendList = ref([]);
+  const receivablePayableList = ref([]);
+
+  // --- 1. 鏀舵敮鏋勬垚鍒嗘瀽 (绠�鍖栫増閫昏緫) ---
+  const gaugeTooltip = { show: false };
+
+  const profitGaugeSeries = computed(() => {
+    const rate =
+      pageInfo.totalIncome > 0
+        ? (pageInfo.netRevenue / pageInfo.totalIncome) * 100
+        : 0;
+    return [
+      {
+        type: "gauge",
+        startAngle: 210,
+        endAngle: -30,
+        min: 0,
+        max: 100,
+        splitNumber: 10,
+        radius: "100%",
+        progress: {
+          show: true,
+          width: 14,
+          itemStyle: { color: pageInfo.netRevenue >= 0 ? "#10b981" : "#f43f5e" },
+        },
+        pointer: { show: false },
+        axisLine: { lineStyle: { width: 14, color: [[1, "#f1f5f9"]] } },
+        axisTick: { show: false },
+        splitLine: { show: false },
+        axisLabel: { show: false },
+        anchor: { show: false },
+        title: { show: false },
+        detail: { show: false },
+        data: [{ value: Math.max(0, Math.min(100, rate)) }],
+      },
+    ];
+  });
+
+  // --- 2. 搴旀敹/搴斾粯姒傝 (鏌辩姸鍥�) ---
+  const barGrid = { left: "3%", right: "4%", bottom: "3%", containLabel: true };
+  const barLegend = { top: "0", right: "center" };
+  const barXAxis = computed(() => [
+    {
+      type: "category",
+      data: receivablePayableList.value.map(item => item.month || ""),
+      axisTick: { alignWithLabel: true },
+    },
+  ]);
+  const barYAxis = [{ type: "value", name: "閲戦 (鍏�)" }];
+  const barTooltip = { trigger: "axis", axisPointer: { type: "shadow" } };
+  const barSeries = computed(() => [
+    {
+      name: "搴旀敹璐︽",
+      type: "bar",
+      barWidth: "30%",
+      data: receivablePayableList.value.map(item => item.receivable || 0),
+      itemStyle: { color: "#10b981" },
+    },
+    {
+      name: "搴斾粯璐︽",
+      type: "bar",
+      barWidth: "30%",
+      data: receivablePayableList.value.map(item => item.payable || 0),
+      itemStyle: { color: "#ef4444" },
+    },
+  ]);
+
+  // --- 3. 璐㈠姟缁煎悎瓒嬪娍鍒嗘瀽 (鎶樼嚎鍥�) ---
+  const trendGrid = { left: "3%", right: "4%", bottom: "3%", containLabel: true };
+  const trendLegend = { top: "0", right: "center" };
+  const trendXAxis = computed(() => [
+    {
+      type: "category",
+      boundaryGap: false,
+      data: monthlyTrendList.value.map(item => item.month || ""),
+    },
+  ]);
+  const trendYAxis = [{ type: "value", name: "閲戦 (鍏�)" }];
+  const trendTooltip = { trigger: "axis" };
+  const trendSeries = computed(() => [
+    {
+      name: "鎬昏惀鏀�",
+      type: "line",
+      smooth: true,
+      data: monthlyTrendList.value.map(item => item.income || 0),
+      itemStyle: { color: "#4f46e5" },
+      areaStyle: { opacity: 0.1 },
+    },
+    {
+      name: "鎬绘敮鍑�",
+      type: "line",
+      smooth: true,
+      data: monthlyTrendList.value.map(item => item.expense || 0),
+      itemStyle: { color: "#f97316" },
+    },
+    {
+      name: "鍑�鍒╂鼎",
+      type: "line",
+      smooth: true,
+      data: monthlyTrendList.value.map(item => item.profit || 0),
+      lineStyle: { width: 4, type: "dashed" },
+      itemStyle: { color: "#10b981" },
+    },
+  ]);
+
+  // --- 鍏敤閫昏緫 ---
+  const formatMoney = val => {
+    return val;
+  };
+
+  const handleDateChange = val => {
+    if (val) getData();
+  };
+
+  const resetDateRange = () => {
+    dateRange.value = [
+      dayjs().subtract(5, "month").format("YYYY-MM"),
+      dayjs().format("YYYY-MM"),
+    ];
+    getData();
+  };
+
+  const disabledDate = time => dayjs(time).isAfter(dayjs(), "month");
+
+  const getData = async () => {
+    if (!dateRange.value || dateRange.value.length !== 2) return;
+
+    const params = {
+      entryDateStart: dayjs(dateRange.value[0])
+        .startOf("month")
+        .format("YYYY-MM-DD"),
+      entryDateEnd: dayjs(dateRange.value[1]).endOf("month").format("YYYY-MM-DD"),
+    };
+
+    try {
+      const res = await accountStatementDetailsByMonth(params);
+      if (res.code === 200 && res.data) {
+        const data = res.data;
+        // 鏇存柊椤堕儴姹囨�诲崱鐗囨暟鎹�
+        pageInfo.totalIncome = data.totalIncome || 0;
+        pageInfo.totalExpense = data.totalExpense || 0;
+        pageInfo.totalReceivable = data.accountsReceivable || 0;
+        pageInfo.totalPayable = data.accountsPayable || 0;
+        pageInfo.netRevenue = data.netRevenue || 0;
+
+        // 鏇存柊鍥捐〃鏁版嵁
+        monthlyTrendList.value = data.monthlyTrendList || [];
+        receivablePayableList.value = data.receivablePayableList || [];
+      }
+    } catch (error) {
+      console.error("鑾峰彇璐㈠姟鎶ヨ〃鏁版嵁澶辫触锛�", error);
+    }
+  };
+
+  onMounted(() => {
+    resetDateRange();
+  });
+</script>
+
+<style scoped lang="scss">
+  .stats-cards {
+    display: grid;
+    grid-template-columns: repeat(auto-fit, minmax(240px, 1fr));
+    gap: 20px;
+    margin-bottom: 24px;
+  }
+
+  .stat-card {
+    background: #fff;
+    border: 1px solid #edf2f7;
+    border-radius: 12px;
+    padding: 24px;
+    display: flex;
+    align-items: center;
+    gap: 16px;
+    box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.05);
+    transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
+
+    &:hover {
+      box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.1);
+      transform: translateY(-4px);
+    }
+
+    .stat-icon {
+      width: 56px;
+      height: 56px;
+      background: #f7fafc;
+      border-radius: 12px;
+      display: flex;
+      align-items: center;
+      justify-content: center;
+      img {
+        width: 32px;
+        height: 32px;
+      }
+    }
+
+    .stat-content {
+      .stat-label {
+        font-size: 14px;
+        color: #718096;
+        margin-bottom: 4px;
+      }
+      .stat-value {
+        font-size: 20px;
+        font-weight: 700;
+        color: #2d3748;
+      }
+    }
+  }
+
+  .charts-row {
+    display: grid;
+    grid-template-columns: repeat(auto-fit, minmax(320px, 1fr));
+    gap: 24px;
+    margin-bottom: 24px;
+  }
+
+  @media (min-width: 1200px) {
+    .charts-row {
+      grid-template-columns: repeat(2, 1fr);
+    }
+  }
+
+  .chart-card,
+  .trend-chart-card {
+    border-radius: 16px;
+    border: none;
+    box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.05);
+
+    .card-header {
+      display: flex;
+      align-items: center;
+      gap: 8px;
+      .header-title {
+        font-size: 16px;
+        font-weight: 600;
+        color: #1a202c;
+      }
+      .el-icon {
+        color: #a0aec0;
+        cursor: help;
+      }
+    }
+  }
+
+  .financial-overview-container {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    flex-wrap: nowrap;
+    gap: 10px;
+    padding: 20px 0;
+    width: 100%;
+    overflow: hidden;
+
+    .overview-item {
+      flex: 1;
+      min-width: 0; // 鍏佽鍦� flex 瀹瑰櫒涓缉鍐欙紝闃叉鍐呭鎾戝紑
+      display: flex;
+      justify-content: center;
+
+      .overview-box {
+        position: relative;
+        width: 100%;
+        max-width: 320px;
+        height: 110px;
+        background: #f8fafc;
+        border-radius: 12px;
+        padding: 12px 16px;
+        display: flex;
+        align-items: center;
+        gap: 12px;
+        overflow: hidden;
+        transition: all 0.3s ease;
+
+        &:hover {
+          transform: translateY(-5px);
+          box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.1);
+        }
+
+        .icon-circle {
+          flex-shrink: 0;
+          width: 42px;
+          height: 42px;
+          border-radius: 10px;
+          display: flex;
+          align-items: center;
+          justify-content: center;
+          font-size: 20px;
+          z-index: 2;
+        }
+
+        .data-content {
+          z-index: 2;
+          min-width: 0;
+          .label {
+            font-size: 13px;
+            color: #718096;
+            margin-bottom: 2px;
+            font-weight: 500;
+            white-space: nowrap;
+          }
+          .value {
+            font-size: 18px;
+            font-weight: 800;
+            color: #1a202c;
+            line-height: 1.2;
+            overflow: hidden;
+            text-overflow: ellipsis;
+            white-space: nowrap;
+          }
+          .unit {
+            font-size: 11px;
+            color: #a0aec0;
+          }
+        }
+
+        .bg-decoration {
+          position: absolute;
+          right: -5px;
+          bottom: -5px;
+          font-size: 32px;
+          font-weight: 950;
+          color: rgba(0, 0, 0, 0.03);
+          font-style: italic;
+          user-select: none;
+          z-index: 1;
+        }
+      }
+
+      &.income {
+        .icon-circle {
+          background: #eef2ff;
+          color: #4f46e5;
+        }
+        .overview-box {
+          border-left: 5px solid #4f46e5;
+        }
+      }
+
+      &.expense {
+        .icon-circle {
+          background: #fff7ed;
+          color: #f97316;
+        }
+        .overview-box {
+          border-left: 5px solid #f97316;
+        }
+      }
+    }
+
+    .profit-indicator {
+      flex: 0 40%; // 鍥哄畾瀹藉害锛屼笉鍙備笌寮规�х缉鏀句互淇濊瘉浠〃鐩樺畬鏁�
+      display: flex;
+      justify-content: center;
+      align-items: center;
+
+      .profit-gauge-wrapper {
+        position: relative;
+        display: flex;
+        justify-content: center;
+        align-items: center;
+        width: 100%;
+        // max-width: 180px;
+
+        .profit-center-text {
+          position: absolute;
+          top: 50%;
+          left: 50%;
+          transform: translate(-50%, -50%);
+          text-align: center;
+          width: 100%;
+
+          .label {
+            font-size: 12px;
+            color: #718096;
+            font-weight: 500;
+          }
+
+          .value {
+            font-size: 20px;
+            font-weight: 800;
+            margin: 2px 0;
+            overflow: hidden;
+            text-overflow: ellipsis;
+            white-space: nowrap;
+
+            &.plus {
+              color: #10b981;
+            }
+
+            &.minus {
+              color: #f43f5e;
+            }
+          }
+
+          .rate {
+            font-size: 11px;
+            color: #a0aec0;
+            font-weight: 500;
+          }
+        }
+      }
+    }
+
+    // 閽堝闈炲父绐勭殑灞忓箷杩涜鏁翠綋缂╂斁
+    @media (max-width: 1400px) {
+      transform-origin: center;
+      // 濡傛灉瀹瑰櫒澶獎锛岄�氳繃缂╁皬鍐呴儴鍏冪礌鏉ラ�傚簲
+      // 杩欓噷涓嶄娇鐢� transform: scale 鍥犱负浼氬奖鍝嶅竷灞�娴侊紝鏀圭敤鍐呴儴灏哄寰皟
+      .overview-item .overview-box {
+        padding: 10px;
+        gap: 8px;
+        .value {
+          font-size: 16px;
+        }
+        .icon-circle {
+          width: 36px;
+          height: 36px;
+          font-size: 18px;
+        }
+      }
+      .profit-indicator {
+        flex: 0 40%;
+        .profit-gauge-wrapper .value {
+          font-size: 18px;
+        }
+      }
+    }
+  }
+</style>

--
Gitblit v1.9.3