From b12b55a5ee1b34b5a3f9d21533fa9fc909207285 Mon Sep 17 00:00:00 2001
From: spring <2396852758@qq.com>
Date: 星期四, 05 二月 2026 09:40:13 +0800
Subject: [PATCH] Merge branch 'dev_New' of http://114.132.189.42:9002/r/product-inventory-management into dev_New

---
 src/views/reportAnalysis/qualityAnalysis/components/left-top.vue |  448 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++
 1 files changed, 448 insertions(+), 0 deletions(-)

diff --git a/src/views/reportAnalysis/qualityAnalysis/components/left-top.vue b/src/views/reportAnalysis/qualityAnalysis/components/left-top.vue
new file mode 100644
index 0000000..8237a3f
--- /dev/null
+++ b/src/views/reportAnalysis/qualityAnalysis/components/left-top.vue
@@ -0,0 +1,448 @@
+<template>
+  <div>
+    <PanelHeader title="璐ㄩ噺鎸囨爣鍚堟牸鍒嗘瀽" />
+    <div class="main-panel panel-item-customers">
+      <div v-for="section in sections" :key="section.key" class="inspect-block">
+        <div class="filters-row">
+          <div class="filters-row-left">
+            <span></span>
+            <p>{{ section.title }}</p>
+          </div>
+          <DateTypeSwitch v-model="section.dateType" @change="(v) => handleDateTypeChange(section.key, v)" />
+        </div>
+
+        <div class="inspect-body">
+          <div class="ring">
+            <Echarts :chartStyle="ringChartStyle" :series="buildRingSeries(section)" :tooltip="ringTooltip"
+              :legend="{ show: false }" :options="ringOptions" />
+          </div>
+
+          <div class="stats">
+            <div class="stat-row">
+              <div class="stat-left">
+                <span class="dot dot-qualified"></span>
+                <span class="stat-label">鍚堟牸鏁�</span>
+              </div>
+              <div class="stat-right">
+                <span class="stat-value">{{ section.qualifiedCount }}</span>
+                <span class="stat-percent">{{ formatPercent(section.qualifiedRate) }}</span>
+              </div>
+            </div>
+            <div class="stat-row">
+              <div class="stat-left">
+                <span class="dot dot-unqualified"></span>
+                <span class="stat-label">涓嶅悎鏍兼暟</span>
+              </div>
+              <div class="stat-right">
+                <span class="stat-value">{{ section.unqualifiedCount }}</span>
+                <span class="stat-percent">{{ formatPercent(section.unqualifiedRate) }}</span>
+              </div>
+            </div>
+          </div>
+        </div>
+      </div>
+    </div>
+  </div>
+</template>
+
+<script setup>
+import { reactive, onMounted } from 'vue'
+import Echarts from '@/components/Echarts/echarts.vue'
+import PanelHeader from './PanelHeader.vue'
+import DateTypeSwitch from './DateTypeSwitch.vue'
+import { rawMaterialDetection, processDetection, factoryDetection } from '@/api/viewIndex.js'
+
+const QUALIFIED_COLOR = '#4EE4FF'
+const UNQUALIFIED_COLOR = '#3378FF'
+const TRACK_COLOR = 'rgba(78, 228, 255, 0.12)'
+
+const apiMap = {
+  raw: rawMaterialDetection,
+  process: processDetection,
+  final: factoryDetection,
+}
+
+
+const fetchSectionData = async (section) => {
+  const api = apiMap[section.key]
+  if (!api) return
+
+  try {
+    const res = await api({
+      type: section.dateType,
+    })
+
+    if (res?.code === 200 && res?.data) {
+      const data = res.data
+      section.qualifiedCount = Number(data.qualifiedCount || 0)
+      section.unqualifiedCount = Number(data.unqualifiedCount || 0)
+      section.qualifiedRate = Number(data.qualifiedRate || 0)
+      section.unqualifiedRate = Number(data.unqualifiedRate || 0)
+    }
+  } catch (err) {
+    console.error(`${section.key} 鎺ュ彛璇锋眰澶辫触`, err)
+  }
+}
+
+
+const sections = reactive([
+  {
+    key: 'raw',
+    title: '鍘熸潗鏂欐娴�',
+    dateType: 1,
+    qualifiedCount: 0,
+    unqualifiedCount: 0,
+    qualifiedRate: 0,
+    unqualifiedRate: 0,
+  },
+  {
+    key: 'process',
+    title: '杩囩▼妫�娴�',
+    dateType: 1,
+    qualifiedCount: 0,
+    unqualifiedCount: 0,
+    qualifiedRate: 0,
+    unqualifiedRate: 0,
+  },
+  {
+    key: 'final',
+    title: '鎴愬搧鍑哄巶妫�娴�',
+    dateType: 1,
+    qualifiedCount: 0,
+    unqualifiedCount: 0,
+    qualifiedRate: 0,
+    unqualifiedRate: 0,
+  },
+])
+
+const ringChartStyle = {
+  width: '110px',
+  height: '110px',
+}
+
+const ringOptions = {
+  backgroundColor: 'transparent',
+  textStyle: { color: '#B8C8E0' },
+}
+
+const ringTooltip = {
+  show: false,
+}
+
+const calcRates = (qualifiedCount, unqualifiedCount) => {
+  const total = Number(qualifiedCount || 0) + Number(unqualifiedCount || 0)
+  if (total <= 0) return { qualifiedRate: 0, unqualifiedRate: 0 }
+  const qualifiedRate = Math.round((Number(qualifiedCount || 0) / total) * 100)
+  const unqualifiedRate = Math.max(0, 100 - qualifiedRate)
+  return { qualifiedRate, unqualifiedRate }
+}
+
+const formatPercent = (v) => `${Number(v || 0)}%`
+
+const buildRingSeries = (section) => {
+  const qualified = Number(section.qualifiedCount || 0)
+  const unqualified = Number(section.unqualifiedCount || 0)
+  const total = qualified + unqualified
+
+  return [
+    {
+      type: 'pie',
+      radius: ['68%', '82%'],
+      center: ['50%', '50%'],
+      silent: true,
+      label: { show: false },
+      labelLine: { show: false },
+      itemStyle: { color: TRACK_COLOR },
+      data: [1],
+    },
+    {
+      name: section.title,
+      type: 'pie',
+      radius: ['68%', '82%'],
+      center: ['50%', '50%'],
+      silent: true,
+      label: { show: false },
+      labelLine: { show: false },
+      startAngle: 90,
+      clockwise: true,
+      minAngle: total > 0 ? 8 : 0,
+      itemStyle: {
+        borderColor: 'rgba(10, 28, 58, 0.95)',
+        borderWidth: 2,
+      },
+      data: [
+        {
+          value: qualified,
+          name: '鍚堟牸鏁�',
+          itemStyle: {
+            color: QUALIFIED_COLOR,
+            shadowBlur: 16,
+            shadowColor: 'rgba(78, 228, 255, 0.45)',
+          },
+        },
+        {
+          value: unqualified,
+          name: '涓嶅悎鏍兼暟',
+          itemStyle: {
+            color: UNQUALIFIED_COLOR,
+            shadowBlur: 10,
+            shadowColor: 'rgba(51, 120, 255, 0.35)',
+          },
+        },
+      ],
+    },
+    {
+      type: 'pie',
+      radius: ['52%', '56%'],
+      center: ['50%', '50%'],
+      silent: true,
+      label: { show: false },
+      labelLine: { show: false },
+      itemStyle: { color: 'rgba(0, 127, 255, 0.22)' },
+      data: [1],
+    },
+  ]
+}
+
+const handleDateTypeChange = (key, dateType) => {
+  const section = sections.find((s) => s.key === key)
+  if (!section) return
+  section.dateType = dateType
+  // 鍒囨崲鏃ユ湡绫诲瀷鏃堕噸鏂拌幏鍙栨暟鎹�
+  fetchSectionData(section)
+}
+
+// 缁勪欢鎸傝浇鏃惰幏鍙栨墍鏈塻ection鐨勬暟鎹�
+onMounted(() => {
+  sections.forEach((section) => {
+    fetchSectionData(section)
+  })
+})
+</script>
+
+<style scoped>
+.main-panel {
+  display: flex;
+  flex-direction: column;
+  height: 100%;
+  gap: 0;
+}
+
+.filters-row {
+  display: flex;
+  justify-content: space-between;
+  align-items: center;
+  gap: 12px;
+  margin-bottom: 10px;
+
+  .filters-row-left {
+    width: 50%;
+    color: white;
+    /* 鐢╢lex鏇夸唬float锛岃瀛愬厓绱犲榻愭洿绋冲畾 */
+    display: flex;
+    align-items: center;
+
+    span {
+      /* 鏍稿績锛氱埗绾х浉瀵瑰畾浣嶏紝浣滀负浼厓绱犲熀鍑� */
+      position: relative;
+      display: inline-block;
+      /* 缁欎吉鍏冪礌鍜屾枃瀛楃暀绌洪棿 */
+      padding-left: 22px;
+      /* 鏂囧瓧鍨傜洿灞呬腑 */
+      line-height: 23px;
+      margin-right: 8px;
+
+      &::after {
+        content: '';
+        display: inline-block;
+        width: 16px;
+        height: 16px;
+        clip-path: polygon(50% 0%, 100% 50%, 50% 100%, 0% 50%);
+        background: #217AFF;
+        position: absolute;
+        top: 50%;
+        left: 0;
+        transform: translateY(-50%);
+        /* 纭繚鑿卞舰鍦ㄦ笎鍙樺潡涓婃柟 */
+        z-index: 1;
+      }
+
+      &::before {
+        content: '';
+        display: inline-block;
+        width: 18px;
+        height: 7px;
+        border-radius: 8px;
+        background: linear-gradient(360deg, rgba(33, 133, 255, 0.4) 0%, rgba(33, 221, 255, 0) 100%);
+        position: absolute;
+        top: 50%;
+        left: -1px;
+        /* 绮惧噯璐村湪鑿卞舰姝d笅鏂� */
+        transform: translateY(calc(0% + 8px));
+        z-index: 0;
+      }
+    }
+
+    p {
+      width: 100px;
+      height: 23px;
+      /* 娓愬彉璧峰鑹插拰鑿卞舰缁熶竴锛屾洿鍗忚皟 */
+      background: linear-gradient(90deg, #217AFF 0%, rgba(33, 221, 255, 0) 100%);
+      /* 绮惧噯鍨傜洿灞呬腑 */
+      line-height: 26px;
+      text-align: center;
+      color: white;
+      /* 鐢ㄩ珮搴︾殑涓�鍗婂仛鍦嗚锛岀‘淇濆乏杈规槸瀹岀編鍗婂渾 */
+      border-radius: 12px 0 0 12px;
+      /* 鍙�夛細鍔犱竴鐐瑰乏鍐呰竟璺濓紝璁╂枃瀛椾笉璐磋竟 */
+      padding-left: 4px;
+    }
+  }
+}
+
+.panel-item-customers {
+  border: 1px solid #1a58b0;
+  padding: 14px 18px;
+  width: 100%;
+  height: 960px;
+  box-sizing: border-box;
+}
+
+.inspect-block {
+  flex: 1 1 0;
+  min-height: 0;
+  display: flex;
+  flex-direction: column;
+  padding: 8px 0;
+  gap: 6px;
+  position: relative;
+}
+
+.inspect-block:not(:last-child)::after {
+  content: '';
+  position: absolute;
+  left: 0;
+  right: 0;
+  bottom: 0;
+  height: 1px;
+  background: linear-gradient(90deg, rgba(33, 122, 255, 0) 0%, rgba(33, 122, 255, 0.55) 50%, rgba(33, 122, 255, 0) 100%);
+  pointer-events: none;
+}
+
+.inspect-body {
+  flex: 1 1 auto;
+  min-height: 0;
+  display: flex;
+  justify-content: space-around;
+  align-items: center;
+  gap: 18px;
+}
+
+.ring {
+  width: 120px;
+  height: 120px;
+  flex: 0 0 120px;
+  position: relative;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+}
+
+/* 澶栧湀鍒诲害锛堢偣鐘剁幆锛� */
+.ring::before {
+  content: '';
+  position: absolute;
+  inset: -8px;
+  border-radius: 50%;
+  background: repeating-conic-gradient(from 0deg,
+      rgba(78, 228, 255, 0.75) 0 1deg,
+      rgba(78, 228, 255, 0) 1deg 9deg);
+  -webkit-mask: radial-gradient(circle, transparent 62%, #000 63%);
+  mask: radial-gradient(circle, transparent 62%, #000 63%);
+  opacity: 0.35;
+  pointer-events: none;
+}
+
+/* 鏌斿拰鍙戝厜鑳屾櫙 */
+.ring::after {
+  content: '';
+  position: absolute;
+  inset: -20px;
+  border-radius: 50%;
+  background: radial-gradient(circle, rgba(78, 228, 255, 0.18) 0%, rgba(78, 228, 255, 0.06) 40%, rgba(0, 0, 0, 0) 70%);
+  filter: blur(0.2px);
+  pointer-events: none;
+}
+
+.stats {
+  width: 240px;
+  flex: 0 0 240px;
+  display: grid;
+  grid-template-rows: 1fr 1fr;
+  gap: 10px;
+}
+
+.stat-row {
+  display: flex;
+  justify-content: space-between;
+  align-items: center;
+  height: 100%;
+  padding: 10px 14px;
+  border-radius: 4px;
+  border: 1px solid rgba(78, 228, 255, 0.22);
+  background: linear-gradient(90deg, rgba(33, 122, 255, 0.28) 0%, rgba(10, 28, 58, 0.35) 55%, rgba(10, 28, 58, 0.2) 100%);
+  box-shadow: inset 0 0 18px rgba(16, 45, 95, 0.25);
+}
+
+.stat-left {
+  display: inline-flex;
+  align-items: center;
+  gap: 10px;
+  color: #b8c8e0;
+  font-size: 12px;
+}
+
+.dot {
+  width: 10px;
+  height: 10px;
+  border-radius: 2px;
+  display: inline-block;
+  box-shadow: 0 0 10px rgba(78, 228, 255, 0.25);
+}
+
+.dot-qualified {
+  background: rgba(184, 200, 224, 0.85);
+}
+
+.dot-unqualified {
+  background: #4ee4ff;
+}
+
+.stat-right {
+  display: inline-flex;
+  align-items: baseline;
+  gap: 14px;
+}
+
+.stat-value {
+  color: #ffffff;
+  font-size: 14px;
+  font-weight: 600;
+  min-width: 40px;
+  text-align: right;
+  text-shadow: 0 0 10px rgba(78, 228, 255, 0.15);
+}
+
+.stat-percent {
+  color: rgba(184, 200, 224, 0.95);
+  font-size: 12px;
+  min-width: 40px;
+  text-align: right;
+}
+
+/* 璁╁垏鎹㈡寜閽洿璐磋繎鎴浘锛堟洿绱у噾锛� */
+:deep(.date-type-switch .el-radio-button__inner) {
+  padding: 4px 16px;
+  font-size: 12px;
+}
+</style>

--
Gitblit v1.9.3