Quellcode durchsuchen

Merge branch 'master' of http://182.92.126.35:3000/hrx/mky-vent-base

lxh vor 2 Monaten
Ursprung
Commit
d6099ebfa1

BIN
public/model/glft/jbfj/jbfj-dual_2025-01-20.glb → public/model/glft/jbfj/jbfj-dual_2025-01-22.glb


+ 1 - 1
src/utils/threejs/main.worker.ts

@@ -46,7 +46,7 @@ export function initModalWorker() {
     'jbfj/jbfj-hd_2025-01-09.glb',
     'jbfj/jbfj-fm_2023-06-02.glb',
     'jbfj/jbfj-fc_2023-06-02.glb',
-    'jbfj/jbfj-dual_2025-01-20.glb',
+    'jbfj/jbfj-dual_2025-01-22.glb',
     'ztfj/dj1_2023-06-02.glb',
     'ztfj/dj2_2023-06-02.glb',
     'ztfj/bg_2023-06-02.glb',

+ 337 - 330
src/views/vent/bundle/bundleMonitorTable/index.vue

@@ -15,6 +15,11 @@
             <template v-if="column.dataIndex === 'action'">
               <a class="action-link" @click="toDetail(record)">数据分析</a>
             </template>
+            <template v-else>
+              <template v-if="record[column.dataIndex] === null">
+                <span>-</span>
+              </template>
+            </template>
           </template>
         </a-table>
         <div class="data-container">
@@ -22,12 +27,14 @@
           <div class="data-content">
             <div class="title">煤自燃阶段统计分析</div>
             <div class="explain">测点共计{{ combustionCount + selfHeatingCount + latentCount }}个</div>
-            <div class="progress-label">剧烈氧化阶段(燃烧期):{{ combustionCount }}</div>
-            <Progress :percent="combustionPercent" size="default" strokeColor="red" :show-info="true" :format="() => combustionCount" />
-            <div class="progress-label">加速氧化阶段(自热期):{{ selfHeatingCount }}</div>
-            <Progress :percent="selfHeatingPercent" size="default" strokeColor="yellow" :show-info="true" :format="() => selfHeatingCount" />
-            <div class="progress-label">缓慢氧化阶段(潜伏期):{{ latentCount }}</div>
+            <div class="progress-label">潜伏期阶段:{{ latentCount }}</div>
             <Progress :percent="latentPercent" size="default" strokeColor="green" :show-info="true" :format="() => latentCount" />
+            <div class="progress-label">缓慢氧化阶段:{{ latentCount }}</div>
+            <Progress :percent="latentPercent" size="default" strokeColor="green" :show-info="true" :format="() => latentCount" />
+            <div class="progress-label">加速氧化阶段:{{ selfHeatingCount }}</div>
+            <Progress :percent="selfHeatingPercent" size="default" strokeColor="yellow" :show-info="true" :format="() => selfHeatingCount" />
+            <div class="progress-label">剧烈氧化阶段:{{ combustionCount }}</div>
+            <Progress :percent="combustionPercent" size="default" strokeColor="red" :show-info="true" :format="() => combustionCount" />
           </div>
         </div>
       </div>
@@ -42,362 +49,362 @@
 </template>
 
 <script setup lang="ts">
-  import { ref, onMounted, shallowRef, reactive, nextTick } from 'vue';
-  import { columns } from './bundle-table.data';
-  import { getBundleInfoList, getAllFileList } from './bundle-table.api';
-  import customHeader from '/@/components/vent/customHeader.vue';
-  // import { blastDelta } from './modal/blastDelta.vue';
-  import BlastDelta from './modal/blastDelta.vue';
-  import BlastDelta1 from './modal/blastDelta1.vue';
-  import * as echarts from 'echarts';
-  import { Progress } from 'ant-design-vue';
-  // import 'ant-design-vue/dist/antd.css'; // 引入样式
-  let selectList = ref<any[]>([]);
-  let jcddArr = ref<any[]>([]);
-  let formSearch = reactive({
-    pageNum: 1,
-    pageSize: 1000,
-    fileId: '',
-    fileName: '',
-  });
-  const latentCount = ref(0); // 缓慢氧化阶段(潜伏期)
-  const selfHeatingCount = ref(0); // 加速氧化阶段(自热期)
-  const combustionCount = ref(0); // 剧烈氧化阶段(燃烧期)
-  const latentPercent = ref(0); // 缓慢氧化阶段(潜伏期)
-  const selfHeatingPercent = ref(0); // 加速氧化阶段(自热期)
-  const combustionPercent = ref(0); // 剧烈氧化阶段(燃烧期)
-  let tableData = ref<any[]>([]);
-  let modalVisible = ref(false);
-  let selectedFileId = ref<string | null>(null);
-  const posMonitor = shallowRef({});
-  function updateChart(data: any) {
-    const chartDom = document.getElementById('barChart');
-    const myChart = echarts.init(chartDom);
-    const categories = data.map((item: any) => item.jcdd);
-    const c2h2MaxValues = data.map((item: any) => parseFloat(item.c2h2_max));
-    const c2h2AveValues = data.map((item: any) => parseFloat(item.c2h2_ave));
-    const c2h4MaxValues = data.map((item: any) => parseFloat(item.c2h4_max));
-    const c2h4AveValues = data.map((item: any) => parseFloat(item.c2h4_ave));
-    const ch4MaxValues = data.map((item: any) => parseFloat(item.ch4_max));
-    const ch4AveValues = data.map((item: any) => parseFloat(item.ch4_ave));
-    const co2MaxValues = data.map((item: any) => parseFloat(item.co2_max));
-    const co2AveValues = data.map((item: any) => parseFloat(item.co2_ave));
-    const coMaxValues = data.map((item: any) => parseFloat(item.co_max));
-    const coAveValues = data.map((item: any) => parseFloat(item.co_ave));
-    const o2MinValues = data.map((item: any) => parseFloat(item.o2_min));
-    const o2AveValues = data.map((item: any) => parseFloat(item.o2_ave));
+import { ref, onMounted, shallowRef, reactive, nextTick } from 'vue';
+import { columns } from './bundle-table.data';
+import { getBundleInfoList, getAllFileList } from './bundle-table.api';
+import customHeader from '/@/components/vent/customHeader.vue';
+// import { blastDelta } from './modal/blastDelta.vue';
+import BlastDelta from './modal/blastDelta.vue';
+import BlastDelta1 from './modal/blastDelta1.vue';
+import * as echarts from 'echarts';
+import { Progress } from 'ant-design-vue';
+// import 'ant-design-vue/dist/antd.css'; // 引入样式
+let selectList = ref<any[]>([]);
+let jcddArr = ref<any[]>([]);
+let formSearch = reactive({
+  pageNum: 1,
+  pageSize: 1000,
+  fileId: '',
+  fileName: '',
+});
+const latentCount = ref(0); // 缓慢氧化阶段(潜伏期)
+const selfHeatingCount = ref(0); // 加速氧化阶段(自热期)
+const combustionCount = ref(0); // 剧烈氧化阶段(燃烧期)
+const latentPercent = ref(0); // 缓慢氧化阶段(潜伏期)
+const selfHeatingPercent = ref(0); // 加速氧化阶段(自热期)
+const combustionPercent = ref(0); // 剧烈氧化阶段(燃烧期)
+let tableData = ref<any[]>([]);
+let modalVisible = ref(false);
+let selectedFileId = ref<string | null>(null);
+const posMonitor = shallowRef({});
+function updateChart(data: any) {
+  const chartDom = document.getElementById('barChart');
+  const myChart = echarts.init(chartDom);
+  const categories = data.map((item: any) => item.jcdd);
+  const c2h2MaxValues = data.map((item: any) => parseFloat(item.c2h2_max));
+  const c2h2AveValues = data.map((item: any) => parseFloat(item.c2h2_ave));
+  const c2h4MaxValues = data.map((item: any) => parseFloat(item.c2h4_max));
+  const c2h4AveValues = data.map((item: any) => parseFloat(item.c2h4_ave));
+  const ch4MaxValues = data.map((item: any) => parseFloat(item.ch4_max));
+  const ch4AveValues = data.map((item: any) => parseFloat(item.ch4_ave));
+  const co2MaxValues = data.map((item: any) => parseFloat(item.co2_max));
+  const co2AveValues = data.map((item: any) => parseFloat(item.co2_ave));
+  const coMaxValues = data.map((item: any) => parseFloat(item.co_max));
+  const coAveValues = data.map((item: any) => parseFloat(item.co_ave));
+  const o2MinValues = data.map((item: any) => parseFloat(item.o2_min));
+  const o2AveValues = data.map((item: any) => parseFloat(item.o2_ave));
 
-    const option = {
-      title: {
-        text: '束管日报分析',
-        textStyle: {
-          color: '#ffffff', // 设置标题颜色
-        },
-        left: 'center', // 水平居中
-        top: '0', // 设置标题距离顶部的距离
+  const option = {
+    title: {
+      text: '束管日报分析',
+      textStyle: {
+        color: '#ffffff', // 设置标题颜色
       },
-      tooltip: {
-        trigger: 'axis',
-        backgroundColor: 'rgba(28, 72, 105, 0.5)', // 设置 tooltip 背景为透明
-        textStyle: {
-          color: '#ffffff', // 设置 tooltip 字体颜色为白色
-        },
-        axisPointer: {
-          type: 'shadow',
-          label: {
-            show: true,
-            backgroundColor: '#1c4869',
-          },
-        },
+      left: 'center', // 水平居中
+      top: '0', // 设置标题距离顶部的距离
+    },
+    tooltip: {
+      trigger: 'axis',
+      backgroundColor: 'rgba(28, 72, 105, 0.5)', // 设置 tooltip 背景为透明
+      textStyle: {
+        color: '#ffffff', // 设置 tooltip 字体颜色为白色
       },
-      legend: {
-        top: '8%',
-        textStyle: {
-          color: '#ffffffff',
-        },
-        width: '80%', // 设置图例的宽度
-        orient: 'horizontal', // 水平布局
-        pageIconColor: '#ffffff', // 设置翻页图标颜色
-        pageTextStyle: {
-          color: '#ffffff', // 设置翻页文字颜色
+      axisPointer: {
+        type: 'shadow',
+        label: {
+          show: true,
+          backgroundColor: '#1c4869',
         },
       },
-      xAxis: {
-        type: 'category',
-        data: categories,
-        splitLine: { show: true, lineStyle: { color: 'rgba(28, 72, 105, 0.5)' } },
-        axisLabel: {
-          interval: 0, // 显示所有标签
-          color: '#ffffff', // 设置 x 轴字体颜色
-          formatter: function (value: string) {
-            return value.length > 8 ? value.slice(0, 8) + '...' : value; // 截断长标签
-          },
-        },
+    },
+    legend: {
+      top: '8%',
+      textStyle: {
+        color: '#ffffffff',
       },
-      yAxis: {
-        type: 'value',
-        splitLine: { show: true, lineStyle: { color: 'rgba(28, 72, 105, 0.5)' } },
-        axisLabel: {
-          color: '#ffffff',
+      width: '80%', // 设置图例的宽度
+      orient: 'horizontal', // 水平布局
+      pageIconColor: '#ffffff', // 设置翻页图标颜色
+      pageTextStyle: {
+        color: '#ffffff', // 设置翻页文字颜色
+      },
+    },
+    xAxis: {
+      type: 'category',
+      data: categories,
+      splitLine: { show: true, lineStyle: { color: 'rgba(28, 72, 105, 0.5)' } },
+      axisLabel: {
+        interval: 0, // 显示所有标签
+        color: '#ffffff', // 设置 x 轴字体颜色
+        formatter: function (value: string) {
+          return value.length > 8 ? value.slice(0, 8) + '...' : value; // 截断长标签
         },
       },
-      grid: {
-        top: '21%', // 设置 grid 距离顶部的距离,增加间隔
-        left: '3%',
-        right: '4%',
-        bottom: '3%',
-        containLabel: true,
+    },
+    yAxis: {
+      type: 'value',
+      splitLine: { show: true, lineStyle: { color: 'rgba(28, 72, 105, 0.5)' } },
+      axisLabel: {
+        color: '#ffffff',
       },
-      series: [
-        {
-          name: 'C₂H₂最大值',
-          data: c2h2MaxValues,
-          type: 'bar',
-        },
-        {
-          name: 'C₂H₂平均值',
-          data: c2h2AveValues,
-          type: 'bar',
-        },
-        {
-          name: 'C₂H₄最大值',
-          data: c2h4MaxValues,
-          type: 'bar',
-        },
-        {
-          name: 'C₂H₄平均值',
-          data: c2h4AveValues,
-          type: 'bar',
-        },
-        {
-          name: 'CH₄最大值',
-          data: ch4MaxValues,
-          type: 'bar',
-        },
-        {
-          name: 'CH₄平均值',
-          data: ch4AveValues,
-          type: 'bar',
-        },
-        {
-          name: 'CO₂最大值',
-          data: co2MaxValues,
-          type: 'bar',
-        },
-        {
-          name: 'CO₂平均值',
-          data: co2AveValues,
-          type: 'bar',
-        },
-        {
-          name: 'CO最大值',
-          data: coMaxValues,
-          type: 'bar',
-        },
-        {
-          name: 'CO平均值',
-          data: coAveValues,
-          type: 'bar',
-        },
-        {
-          name: 'O₂最小值',
-          data: o2MinValues,
-          type: 'bar',
-        },
-        {
-          name: 'O₂平均值',
-          data: o2AveValues,
-          type: 'bar',
-        },
-      ],
-    };
-    myChart.setOption(option);
-  }
-  //跳转到爆炸三角形
-  function toDetail(record: any) {
-    posMonitor.value = record;
-    modalVisible.value = true;
-  }
-  //获取束管日报
-  async function getTableList(params: any) {
-    let res = await getBundleInfoList({ type: 'bundle', ...params });
-    const content = res.content;
-    let contentArr = JSON.parse(content);
-    latentCount.value = contentArr.filter((item: any) => item.internalFireWarnLevel === '缓慢氧化阶段(潜伏期)').length;
-    selfHeatingCount.value = contentArr.filter((item: any) => item.internalFireWarnLevel === '加速氧化阶段(自热期)').length;
-    combustionCount.value = contentArr.filter((item: any) => item.internalFireWarnLevel === '剧烈氧化阶段(燃烧期)').length;
-    const total = contentArr.length;
-    latentPercent.value = (latentCount.value / total) * 100;
-    selfHeatingPercent.value = (selfHeatingCount.value / total) * 100;
-    combustionPercent.value = (combustionCount.value / total) * 100;
-    tableData.value = contentArr;
-    nextTick(() => {
-      updateChart(contentArr);
-    });
-  }
+    },
+    grid: {
+      top: '21%', // 设置 grid 距离顶部的距离,增加间隔
+      left: '3%',
+      right: '4%',
+      bottom: '3%',
+      containLabel: true,
+    },
+    series: [
+      {
+        name: 'C₂H₂最大值',
+        data: c2h2MaxValues,
+        type: 'bar',
+      },
+      {
+        name: 'C₂H₂平均值',
+        data: c2h2AveValues,
+        type: 'bar',
+      },
+      {
+        name: 'C₂H₄最大值',
+        data: c2h4MaxValues,
+        type: 'bar',
+      },
+      {
+        name: 'C₂H₄平均值',
+        data: c2h4AveValues,
+        type: 'bar',
+      },
+      {
+        name: 'CH₄最大值',
+        data: ch4MaxValues,
+        type: 'bar',
+      },
+      {
+        name: 'CH₄平均值',
+        data: ch4AveValues,
+        type: 'bar',
+      },
+      {
+        name: 'CO₂最大值',
+        data: co2MaxValues,
+        type: 'bar',
+      },
+      {
+        name: 'CO₂平均值',
+        data: co2AveValues,
+        type: 'bar',
+      },
+      {
+        name: 'CO最大值',
+        data: coMaxValues,
+        type: 'bar',
+      },
+      {
+        name: 'CO平均值',
+        data: coAveValues,
+        type: 'bar',
+      },
+      {
+        name: 'O₂最小值',
+        data: o2MinValues,
+        type: 'bar',
+      },
+      {
+        name: 'O₂平均值',
+        data: o2AveValues,
+        type: 'bar',
+      },
+    ],
+  };
+  myChart.setOption(option);
+}
+//跳转到爆炸三角形
+function toDetail(record: any) {
+  posMonitor.value = record;
+  modalVisible.value = true;
+}
+//获取束管日报
+async function getTableList(params: any) {
+  let res = await getBundleInfoList({ type: 'bundle', ...params });
+  const content = res.content;
+  let contentArr = JSON.parse(content);
+  latentCount.value = contentArr.filter((item: any) => item.internalFireWarnLevel === '缓慢氧化阶段(潜伏期)').length;
+  selfHeatingCount.value = contentArr.filter((item: any) => item.internalFireWarnLevel === '加速氧化阶段(自热期)').length;
+  combustionCount.value = contentArr.filter((item: any) => item.internalFireWarnLevel === '剧烈氧化阶段(燃烧期)').length;
+  const total = contentArr.length;
+  latentPercent.value = (latentCount.value / total) * 100;
+  selfHeatingPercent.value = (selfHeatingCount.value / total) * 100;
+  combustionPercent.value = (combustionCount.value / total) * 100;
+  tableData.value = contentArr;
+  nextTick(() => {
+    updateChart(contentArr);
+  });
+}
 
-  //获取所有文件列表
-  async function getAllFile() {
-    let res = await getAllFileList({ type: 'bundle' });
-    selectList.value = res.records.map((item: any) => ({
-      fileId: item.fileId,
-      fileName: item.fileName,
-    }));
-    jcddArr.value = res.records.map((item: any) => ({
-      fileId: item.jcdd,
-    }));
+//获取所有文件列表
+async function getAllFile() {
+  let res = await getAllFileList({ type: 'bundle' });
+  selectList.value = res.records.map((item: any) => ({
+    fileId: item.fileId,
+    fileName: item.fileName,
+  }));
+  jcddArr.value = res.records.map((item: any) => ({
+    fileId: item.jcdd,
+  }));
+  if (selectList.value.length > 0) {
+    formSearch.fileId = selectList.value[0].fileId;
+    getSearch();
+  }
+}
+// 处理文件点击事件
+function handleFileClick(item: any) {
+  formSearch.fileId = item.fileId;
+  formSearch.fileName = item.fileName;
+  selectedFileId.value = item.fileId;
+  getSearch();
+}
+//查询
+function getSearch() {
+  const selectedFile = selectList.value.find((item) => item.fileId === formSearch.fileId);
+  const params = {
+    fileId: formSearch.fileId,
+    fileName: selectedFile ? selectedFile.fileName : '',
+  };
+  getTableList(params);
+}
+onMounted(() => {
+  getTableList({ type: 'bundle' });
+  getAllFile().then(() => {
     if (selectList.value.length > 0) {
       formSearch.fileId = selectList.value[0].fileId;
+      selectedFileId.value = selectList.value[0].fileId;
       getSearch();
     }
-  }
-  // 处理文件点击事件
-  function handleFileClick(item: any) {
-    formSearch.fileId = item.fileId;
-    formSearch.fileName = item.fileName;
-    selectedFileId.value = item.fileId;
-    getSearch();
-  }
-  //查询
-  function getSearch() {
-    const selectedFile = selectList.value.find((item) => item.fileId === formSearch.fileId);
-    const params = {
-      fileId: formSearch.fileId,
-      fileName: selectedFile ? selectedFile.fileName : '',
-    };
-    getTableList(params);
-  }
-  onMounted(() => {
-    getTableList({ type: 'bundle' });
-    getAllFile().then(() => {
-      if (selectList.value.length > 0) {
-        formSearch.fileId = selectList.value[0].fileId;
-        selectedFileId.value = selectList.value[0].fileId;
-        getSearch();
-      }
-    });
   });
+});
 </script>
 
 <style lang="less" scoped>
-  @import '/@/design/theme.less';
+@import '/@/design/theme.less';
 
-  .content-container {
-    display: flex;
-    width: 100%;
-    height: 100%;
-    padding-top: 54px;
-  }
+.content-container {
+  display: flex;
+  width: 100%;
+  height: 100%;
+  padding-top: 54px;
+}
 
-  .file-list {
-    width: 20%;
-    padding: 10px;
-    margin-right: 10px;
-    margin-bottom: 50px;
-    border: 1px solid #99e8ff66;
-    background: #27546e1a;
-    box-shadow: 0px 0px 20px 7px rgba(145, 233, 254, 0.7) inset;
-    -moz-box-shadow: 0px 0px 20px 7px rgba(145, 233, 254, 0.7) inset;
-    -webkit-box-shadow: 0px 0px 50px 1px rgb(149 235 255 / 5%) inset;
-  }
+.file-list {
+  width: 20%;
+  padding: 10px;
+  margin-right: 10px;
+  margin-bottom: 50px;
+  border: 1px solid #99e8ff66;
+  background: #27546e1a;
+  box-shadow: 0px 0px 20px 7px rgba(145, 233, 254, 0.7) inset;
+  -moz-box-shadow: 0px 0px 20px 7px rgba(145, 233, 254, 0.7) inset;
+  -webkit-box-shadow: 0px 0px 50px 1px rgb(149 235 255 / 5%) inset;
+}
 
-  .file-list ul {
-    list-style: none;
-    padding: 0;
-  }
+.file-list ul {
+  list-style: none;
+  padding: 0;
+}
 
-  .file-list li {
-    color: #fff;
-    padding: 5px;
-    cursor: pointer;
-  }
+.file-list li {
+  color: #fff;
+  padding: 5px;
+  cursor: pointer;
+}
 
-  .file-list li:hover,
-  .file-list li.selected {
-    background: #1c4869;
-  }
+.file-list li:hover,
+.file-list li.selected {
+  background: #1c4869;
+}
 
-  .table-container {
-    margin-top: 10px;
-    width: 80%;
-    box-sizing: border-box;
-  }
-  .data-container {
-    display: flex;
-    width: 100%;
-    height: 100%;
-  }
+.table-container {
+  margin-top: 10px;
+  width: 80%;
+  box-sizing: border-box;
+}
+.data-container {
+  display: flex;
+  width: 100%;
+  height: 100%;
+}
 
-  .bar-chart {
-    flex: 3; /* 占据 3/4 的空间 */
-    width: 100%;
-    height: 400px;
-  }
+.bar-chart {
+  flex: 3; /* 占据 3/4 的空间 */
+  width: 100%;
+  height: 400px;
+}
 
-  .data-content {
-    flex: 1; /* 占据 1/4 的空间 */
-    height: 400px;
-    display: flex;
-    flex-direction: column; /* 垂直排列进度条 */
-    // align-items: center; /* 水平居中 */
-    margin: 10px;
-    .title {
-      font-size: 18px;
-      font-weight: 600;
-      color: #fff;
-      margin-bottom: 20px;
-    }
-    .explain {
-      color: var(--vent-table-action-link);
-      margin-top: 18px;
-    }
+.data-content {
+  flex: 1; /* 占据 1/4 的空间 */
+  height: 400px;
+  display: flex;
+  flex-direction: column; /* 垂直排列进度条 */
+  // align-items: center; /* 水平居中 */
+  margin: 10px;
+  .title {
+    font-size: 18px;
+    font-weight: 600;
+    color: #fff;
+    margin-bottom: 20px;
   }
-
-  .dustMonitor {
-    width: 100%;
-    height: 100%;
-    padding: 10px 10px 15px 10px;
-    box-sizing: border-box;
-    position: relative;
+  .explain {
+    color: var(--vent-table-action-link);
+    margin-top: 18px;
   }
+}
 
-  .yellow-progress .ant-progress-bg {
-    background-color: yellow !important;
-  }
+.dustMonitor {
+  width: 100%;
+  height: 100%;
+  padding: 10px 10px 15px 10px;
+  box-sizing: border-box;
+  position: relative;
+}
 
-  :deep(.zxm-table-thead > tr > th:last-child) {
-    border-right: 1px solid #91e9fe !important;
-  }
+.yellow-progress .ant-progress-bg {
+  background-color: yellow !important;
+}
 
-  :deep(.zxm-picker-input > input) {
-    color: #fff;
-  }
+:deep(.zxm-table-thead > tr > th:last-child) {
+  border-right: 1px solid #91e9fe !important;
+}
 
-  :deep(.zxm-select:not(.zxm-select-customize-input) .zxm-select-selector) {
-    border: 1px solid var(--vent-form-item-border) !important;
-    background-color: #ffffff00 !important;
-  }
+:deep(.zxm-picker-input > input) {
+  color: #fff;
+}
 
-  :deep(.zxm-select-selection-item) {
-    color: #fff !important;
-  }
-  .blast-delta-container {
-    margin: 50px;
-    display: flex;
-    justify-content: space-between;
-  }
-  .progress {
-    width: 100%;
-    height: 20px;
-    margin-top: 10px;
-  }
-  .progress-label {
-    margin-top: 20px;
-    text-align: left;
-    margin-bottom: 5px;
-    color: #fff;
-  }
-  ::deep .progress-text {
-    color: #fff !important; /* 自定义百分比文字颜色 */
-  }
+:deep(.zxm-select:not(.zxm-select-customize-input) .zxm-select-selector) {
+  border: 1px solid var(--vent-form-item-border) !important;
+  background-color: #ffffff00 !important;
+}
+
+:deep(.zxm-select-selection-item) {
+  color: #fff !important;
+}
+.blast-delta-container {
+  margin: 50px;
+  display: flex;
+  justify-content: space-between;
+}
+.progress {
+  width: 100%;
+  height: 20px;
+  margin-top: 10px;
+}
+.progress-label {
+  margin-top: 20px;
+  text-align: left;
+  margin-bottom: 5px;
+  color: #fff;
+}
+::deep .progress-text {
+  color: #fff !important; /* 自定义百分比文字颜色 */
+}
 </style>

+ 12 - 5
src/views/vent/bundleSpy/bundleSpyTable/index.vue

@@ -15,6 +15,11 @@
             <template v-if="column.dataIndex === 'action'">
               <a class="action-link" @click="toDetail(record)">数据分析</a>
             </template>
+            <template v-else>
+              <template v-if="record[column.dataIndex] === null">
+                <span>-</span>
+              </template>
+            </template>
           </template>
         </a-table>
         <div class="data-container">
@@ -22,12 +27,14 @@
           <div class="data-content">
             <div class="title">煤自燃阶段统计分析</div>
             <div class="explain">测点共计{{ combustionCount + selfHeatingCount + latentCount }}个</div>
-            <div class="progress-label">剧烈氧化阶段(燃烧期):{{ combustionCount }}</div>
-            <Progress :percent="combustionPercent" size="default" strokeColor="red" :show-info="true" :format="() => combustionCount" />
-            <div class="progress-label">加速氧化阶段(自热期):{{ selfHeatingCount }}</div>
-            <Progress :percent="selfHeatingPercent" size="default" strokeColor="yellow" :show-info="true" :format="() => selfHeatingCount" />
-            <div class="progress-label">缓慢氧化阶段(潜伏期):{{ latentCount }}</div>
+            <div class="progress-label">潜伏期阶段:{{ latentCount }}</div>
             <Progress :percent="latentPercent" size="default" strokeColor="green" :show-info="true" :format="() => latentCount" />
+            <div class="progress-label">缓慢氧化阶段:{{ latentCount }}</div>
+            <Progress :percent="latentPercent" size="default" strokeColor="green" :show-info="true" :format="() => latentCount" />
+            <div class="progress-label">加速氧化阶段:{{ selfHeatingCount }}</div>
+            <Progress :percent="selfHeatingPercent" size="default" strokeColor="yellow" :show-info="true" :format="() => selfHeatingCount" />
+            <div class="progress-label">剧烈氧化阶段:{{ combustionCount }}</div>
+            <Progress :percent="combustionPercent" size="default" strokeColor="red" :show-info="true" :format="() => combustionCount" />
           </div>
         </div>
       </div>

+ 8 - 2
src/views/vent/dust/dustMonitorTable/index.vue

@@ -10,11 +10,17 @@
         </ul>
       </div>
       <div class="table-container">
-        <a-table :columns="columns" :data-source="tableData" size="small" :scroll="{ y: 250 }" class="tableW" />
+        <a-table :columns="columns" :data-source="tableData" size="small" :scroll="{ y: 250 }" class="tableW">
+          <template #bodyCell="{ record, column }">
+            <template v-if="record[column.dataIndex] === null">
+              <span>-</span>
+            </template>
+          </template>
+        </a-table>
         <div class="tab-container">
           <a-tabs v-model:activeKey="activeTab">
             <a-tab-pane key="workplace" class="tab1" tab="监测地点粉尘情况分析">
-              <div class="filter-container">
+              <div class="filter-container" v-if="workplaceList.length > 0">
                 <a-select
                   :key="DefaultValue"
                   :default-value="DefaultValue"

+ 8 - 9
src/views/vent/home/clique/components/echart-scene.vue

@@ -57,7 +57,6 @@
           opacity: 0.1,
         },
       };
-      console.log(scene.value, 'scene');
       const myChart = echarts.init(scene.value);
       let option = {
         color: ['#4edaff', '#53ffde', '#a696ed'],
@@ -246,14 +245,14 @@
 
   watch(
     () => props.echartData,
-    (newV, oldV) => {
-      console.log(newV, 'newV-----------');
-      percent.jf = newV.jfq;
-      percent.yf = newV.yfq;
-      percent.hf = newV.hfq;
-      percent.jfEcharts = Math.ceil(newV.zf - percent.jf);
-      percent.yfEcharts = Math.ceil(newV.zf - percent.yf);
-      percent.hfEcharts = Math.ceil(newV.zf - percent.hf);
+    ({ jfq, yfq, hfq, zf }) => {
+      const getEchartsNumber = (v: number) => Math.ceil(isNaN(v) ? 0 : v);
+      percent.jf = isNaN(jfq) ? '-' : jfq;
+      percent.yf = isNaN(yfq) ? '-' : yfq;
+      percent.hf = isNaN(hfq) ? '-' : hfq;
+      percent.jfEcharts = getEchartsNumber(zf - percent.jf);
+      percent.yfEcharts = getEchartsNumber(zf - percent.yf);
+      percent.hfEcharts = getEchartsNumber(zf - percent.hf);
       getOption();
     },
     {

+ 17 - 8
src/views/vent/home/clique/components/scene-key.vue

@@ -49,7 +49,7 @@
 
   let sceneTitle = ref('回风系统通风状态监测');
 
-  let sceneList = reactive([
+  let sceneList = reactive<{ label: string; value: number | string }[]>([
     { label: '总阻力', value: 0 },
     // { label: '总进风', value: 0 },
     { label: '总风量', value: 0 },
@@ -79,14 +79,23 @@
       }
       return false;
     });
-    echartData.jfq = data.majorpath.drag_1;
-    echartData.yfq = data.majorpath.drag_2;
-    echartData.hfq = data.majorpath.drag_3;
-    echartData.zf = data.majorpath.drag_total;
+    // 下面的每一个变量可能是:正常的 number、NaN
+    const { drag_1, drag_2, drag_3, drag_total, m3_total } = data.majorpath;
+    echartData.jfq = drag_1;
+    echartData.yfq = drag_2;
+    echartData.hfq = drag_3;
+    echartData.zf = drag_total;
 
-    sceneList[0].value = data.majorpath.drag_total;
-    sceneList[1].value = data.majorpath.m3_total;
-    sceneList[2].value = Math.round(((1.19 * sceneList[1].value) / 60 / Math.sqrt(sceneList[0].value)) * 100) / 100;
+    if (isNaN(drag_total) || isNaN(m3_total)) {
+      // 如果数据是0,也会展示 '-'
+      sceneList[0].value = drag_total || '-';
+      sceneList[1].value = m3_total || '-';
+      sceneList[2].value = '-';
+    } else {
+      sceneList[0].value = drag_total;
+      sceneList[1].value = m3_total;
+      sceneList[2].value = Math.round(((1.19 * m3_total) / 60 / Math.sqrt(drag_total)) * 100) / 100;
+    }
   }
 
   watch(

+ 22 - 17
src/views/vent/home/clique/index.vue

@@ -156,36 +156,41 @@
       warningList.value = warningListTemp;
       compositeData.value = res.reduce((arr, e) => {
         if (isDataRealTime.value) {
-          return [...arr, ...e.majorpath_data];
+          arr.push(...e.majorpath_data);
         } else {
-          return [
-            ...arr,
-            ...e.majorpath_data.map(({ majorpath, readData }) => {
+          arr.push(
+            ...e.majorpath_data.map((ele) => {
+              const { majorpath, readData } = ele;
+              const { drag_1, drag_2, drag_3, drag_total } = majorpath;
+              const { retM3_merge, fy_merge } = readData;
               // 报表数据只有总数据,按实时数据计算比例然后乘以报表数据
-              if (readData.fy_merge && readData.retM3_merge) {
+              if (fy_merge && retM3_merge) {
                 return {
+                  ...ele,
                   majorpath: {
-                    drag_1: Math.round((majorpath.drag_1 / majorpath.drag_total) * parseInt(readData.fy_merge.value)),
-                    drag_2: Math.round((majorpath.drag_2 / majorpath.drag_total) * parseInt(readData.fy_merge.value)),
-                    drag_3: Math.round((majorpath.drag_3 / majorpath.drag_total) * parseInt(readData.fy_merge.value)),
-                    drag_total: readData.fy_merge.value,
-                    m3_total: readData.retM3_merge.value,
+                    drag_1: Math.round((drag_1 / drag_total) * parseInt(fy_merge.value)),
+                    drag_2: Math.round((drag_2 / drag_total) * parseInt(fy_merge.value)),
+                    drag_3: Math.round((drag_3 / drag_total) * parseInt(fy_merge.value)),
+                    drag_total: parseInt(fy_merge.value),
+                    m3_total: parseInt(retM3_merge.value),
                   },
                 };
               } else {
                 return {
+                  ...ele,
                   majorpath: {
-                    drag_1: '-',
-                    drag_2: '-',
-                    drag_3: '-',
-                    drag_total: '-',
-                    m3_total: '-',
+                    drag_1: NaN,
+                    drag_2: NaN,
+                    drag_3: NaN,
+                    drag_total: NaN,
+                    m3_total: NaN,
                   },
                 };
               }
-            }),
-          ];
+            })
+          );
         }
+        return arr;
       }, []);
       centerDetail.value = res.filter((v) => v.orgcode == orgcode.value)[0];
     }

+ 761 - 761
src/views/vent/monitorManager/fanLocalMonitor/components/conditionAssistance.vue

@@ -80,862 +80,862 @@
 </template>
 
 <script lang="ts" setup>
-  //ts语法
-  import { ref, onMounted, reactive, nextTick, computed } from 'vue';
-  import echarts from '/@/utils/lib/echarts';
-  import { option, initData, fanInfoData, chartsColumnList, echatsOption } from '../fanLocal.data';
-  import { BasicModal, useModalInner } from '/@/components/Modal';
-  import { useForm } from '/@/components/Form/index';
-  import { Input, InputNumber } from 'ant-design-vue';
-  import { Decoration7 as DvDecoration7 } from '@kjgl77/datav-vue3';
-  import { message } from 'ant-design-vue';
-  import { formatNum } from '/@/utils/ventutil';
-  import BarAndLine from '/@/components/chart/BarAndLine.vue';
-  import { cloneDeep } from 'lodash-es';
-  import dayjs from 'dayjs';
-  import { SendOutlined } from '@ant-design/icons-vue';
-
-  const emit = defineEmits(['close', 'register', 'openModal']);
-  const props = defineProps({
-    dataSource: {
-      type: Array,
-      default: () => [],
-    },
-    frequency: {
-      type: Number,
-      default: 30,
-    },
-    m3: {
-      type: Number,
-      default: 670.8,
-    },
-    // gasWarningMax: { type: Number, default: 0.5 },
-    // gasWarningVal: { type: Number, default: 0.6 },
-    // windQuantity: { type: Number, default: 635.84 },
-  });
-  type AssistanceItemType = {
-    angle: number;
-    Hz: number;
-    a: number;
-    b: number;
-    c: number;
-    min: number;
-    max: number;
-  };
-
-  // 注册 modal
-  const [register, { closeModal }] = useModalInner((data) => {
-    nextTick(() => {
-      computeAssistance();
-      if (option['xAxis']) option['xAxis']['data'] = xData;
-      option['series'] = yDataList;
-      if (JSON.stringify(data) !== '{}') {
-        uQ1.value = Number(data['m3']);
-        uHz.value = Math.ceil(data['frequency']);
-        gasWarningVal.value = data['gasWarningVal'];
-        isComputeGas.value = true;
-        nextTick(() => {
-          computeUH(data['frequency'], data['m3']);
-          initEcharts();
-          setTimeout(() => {
-            // 根据频率计算uH
-            makeLine();
-          }, 2000);
-        });
-      } else {
+//ts语法
+import { ref, onMounted, reactive, nextTick, computed } from 'vue';
+import echarts from '/@/utils/lib/echarts';
+import { option, initData, fanInfoData, chartsColumnList, echatsOption } from '../fanLocal.data';
+import { BasicModal, useModalInner } from '/@/components/Modal';
+import { useForm } from '/@/components/Form/index';
+import { Input, InputNumber } from 'ant-design-vue';
+import { Decoration7 as DvDecoration7 } from '@kjgl77/datav-vue3';
+import { message } from 'ant-design-vue';
+import { formatNum } from '/@/utils/ventutil';
+import BarAndLine from '/@/components/chart/BarAndLine.vue';
+import { cloneDeep } from 'lodash-es';
+import dayjs from 'dayjs';
+import { SendOutlined } from '@ant-design/icons-vue';
+
+const emit = defineEmits(['close', 'register', 'openModal']);
+const props = defineProps({
+  dataSource: {
+    type: Array,
+    default: () => [],
+  },
+  frequency: {
+    type: Number,
+    default: 30,
+  },
+  m3: {
+    type: Number,
+    default: 670.8,
+  },
+  // gasWarningMax: { type: Number, default: 0.5 },
+  // gasWarningVal: { type: Number, default: 0.6 },
+  // windQuantity: { type: Number, default: 635.84 },
+});
+type AssistanceItemType = {
+  angle: number;
+  Hz: number;
+  a: number;
+  b: number;
+  c: number;
+  min: number;
+  max: number;
+};
+
+// 注册 modal
+const [register, { closeModal }] = useModalInner((data) => {
+  nextTick(() => {
+    computeAssistance();
+    if (option['xAxis']) option['xAxis']['data'] = xData;
+    option['series'] = yDataList;
+    if (JSON.stringify(data) !== '{}') {
+      uQ1.value = Number(data['m3']);
+      uHz.value = Math.ceil(data['frequency']);
+      gasWarningVal.value = data['gasWarningVal'];
+      isComputeGas.value = true;
+      nextTick(() => {
+        computeUH(data['frequency'], data['m3']);
         initEcharts();
-        isComputeGas.value = false;
-      }
-    });
-  });
-  const loadding = ref<boolean>(false);
-  const formShow = ref(false);
-  const formType = ref('');
-  const ChartRef = ref();
-  const myChart = ref();
-  const refresh = ref(true);
-  const xDataMax = 1000;
-  let xDataMin = 0;
-  const xData: any[] = [];
-  const yDataList: [] = [];
-  let lineNum = 0;
-  const lineEquation = ref<string[]>([]);
-  const assistanceData = ref([]);
-  const monitorData = ref([]);
-  const gasWarningVal = ref(0);
-  const gasWarningMax = ref(0.5);
-  const isComputeGas = ref(false);
-  const isStartCompute = ref(0);
-  const uHz = ref(0);
-  const uQ1 = ref(0);
-  const uQ = computed(() => {
-    if (uQ1.value) {
-      if (gasWarningVal.value) {
-        return ((uQ1.value * gasWarningVal.value) / gasWarningMax.value) * 1.08;
-      }
-      return uQ1.value;
+        setTimeout(() => {
+          // 根据频率计算uH
+          makeLine();
+        }, 2000);
+      });
     } else {
-      return 0;
+      initEcharts();
+      isComputeGas.value = false;
     }
   });
-  const uH = ref<number | undefined>(undefined); //  - 1000
-  const isHaCross = ref(true);
-  const resultObj = ref<{ x: number; y: number; Hz: number } | null>(null);
-
-  const [registerForm, {}] = useForm({
-    labelWidth: 120,
-    actionColOptions: {
-      span: 24,
-    },
-    compact: true,
-    showSubmitButton: true,
-    submitButtonOptions: {
-      text: '提交',
-      preIcon: '',
-    },
-    showResetButton: true,
-    resetButtonOptions: {
-      text: '关闭',
-      preIcon: '',
-    },
-    resetFunc: async () => {
-      formShow.value = false;
-    },
-  });
-
-  function computeAssistance() {
-    assistanceData.value = initData();
-    lineNum = 0;
-    const assistanceDataList = [];
-    const lineEquationList: string[] = [];
-    for (const key in assistanceData.value) {
-      const item = assistanceData.value[key];
-      assistanceDataList.push(item);
-      lineEquationList.push(
-        `H${parseInt(item['Hz'])} = ${item['a']}Q² ${Number(item['b']) > 0 ? '+' : '-'} ${Math.abs(Number(item['b'])).toFixed(4)}Q ${
-          Number(item['c']) > 0 ? '+' : '-'
-        } ${Math.abs(Number(item['c'])).toFixed(4)}`
-      );
-    }
-    lineEquation.value = lineEquationList;
-    lineNum = assistanceDataList.length;
-    xDataMin =
-      Math.min.apply(
-        Math,
-        assistanceDataList.map((item) => {
-          return item.min;
-        })
-      ) - 100;
-    // const xDataMax = Math.max.apply(Math, assistanceDataList.map(item => { return item.max }))
-    fanInfoData.flfw = `${xDataMin}~${xDataMax}`;
-    const computeItem = (item: AssistanceItemType) => {
-      const min = item.min;
-      const max = item.max;
-      const HList: number[] = [];
-      for (let i = xDataMin; i <= xDataMax; i++) {
-        if (i < min) {
-          HList.push(null);
-        } else if (i > max) {
-          HList.push(null);
-        } else {
-          HList.push(item.a * i * i + item.b * i + item.c);
-        }
+});
+const loadding = ref<boolean>(false);
+const formShow = ref(false);
+const formType = ref('');
+const ChartRef = ref();
+const myChart = ref();
+const refresh = ref(true);
+const xDataMax = 1000;
+let xDataMin = 0;
+const xData: any[] = [];
+const yDataList: [] = [];
+let lineNum = 0;
+const lineEquation = ref<string[]>([]);
+const assistanceData = ref([]);
+const monitorData = ref([]);
+const gasWarningVal = ref(0);
+const gasWarningMax = ref(0.5);
+const isComputeGas = ref(false);
+const isStartCompute = ref(0);
+const uHz = ref(0);
+const uQ1 = ref(0);
+const uQ = computed(() => {
+  if (uQ1.value) {
+    if (gasWarningVal.value) {
+      return ((uQ1.value * gasWarningVal.value) / gasWarningMax.value) * 1.08;
+    }
+    return uQ1.value;
+  } else {
+    return 0;
+  }
+});
+const uH = ref<number | undefined>(undefined); //  - 1000
+const isHaCross = ref(true);
+const resultObj = ref<{ x: number; y: number; Hz: number } | null>(null);
+
+const [registerForm, {}] = useForm({
+  labelWidth: 120,
+  actionColOptions: {
+    span: 24,
+  },
+  compact: true,
+  showSubmitButton: true,
+  submitButtonOptions: {
+    text: '提交',
+    preIcon: '',
+  },
+  showResetButton: true,
+  resetButtonOptions: {
+    text: '关闭',
+    preIcon: '',
+  },
+  resetFunc: async () => {
+    formShow.value = false;
+  },
+});
+
+function computeAssistance() {
+  assistanceData.value = initData();
+  lineNum = 0;
+  const assistanceDataList = [];
+  const lineEquationList: string[] = [];
+  for (const key in assistanceData.value) {
+    const item = assistanceData.value[key];
+    assistanceDataList.push(item);
+    lineEquationList.push(
+      `H${parseInt(item['Hz'])} = ${item['a']}Q² ${Number(item['b']) > 0 ? '+' : '-'} ${Math.abs(Number(item['b'])).toFixed(4)}Q ${
+        Number(item['c']) > 0 ? '+' : '-'
+      } ${Math.abs(Number(item['c'])).toFixed(4)}`
+    );
+  }
+  lineEquation.value = lineEquationList;
+  lineNum = assistanceDataList.length;
+  xDataMin =
+    Math.min.apply(
+      Math,
+      assistanceDataList.map((item) => {
+        return item.min;
+      })
+    ) - 100;
+  // const xDataMax = Math.max.apply(Math, assistanceDataList.map(item => { return item.max }))
+  fanInfoData.flfw = `${xDataMin}~${xDataMax}`;
+  const computeItem = (item: AssistanceItemType) => {
+    const min = item.min;
+    const max = item.max;
+    const HList: number[] = [];
+    for (let i = xDataMin; i <= xDataMax; i++) {
+      if (i < min) {
+        HList.push(null);
+      } else if (i > max) {
+        HList.push(null);
+      } else {
+        HList.push(item.a * i * i + item.b * i + item.c);
       }
-      return HList;
-    };
-
-    for (const key in assistanceData.value) {
-      const element: AssistanceItemType = assistanceData.value[key];
-      const yData: number[] = computeItem(element);
-      const series = {
-        type: 'line',
-        name: `${element['Hz']}Hz`,
-        smooth: true,
-        showSymbol: false,
-        symbol: 'none',
-        emphasis: {
-          focus: 'series',
-        },
-        itemStyle: { normal: { label: { show: true } } },
-        lineStyle: {
-          width: 1,
-          color: '#ffffff88',
-        },
-        zlevel: 0,
-        z: 1,
-        endLabel: {
-          show: true,
-          formatter: '{a}',
-          distance: 0,
-          color: '#39E9FE99',
-          backgroundColor: 'transparent',
-          padding: [3, 3, 2, 3],
-        },
-        data: yData,
-      };
-
-      yDataList.push(series);
     }
+    return HList;
+  };
 
-    for (let i = xDataMin; i <= xDataMax; i++) {
-      xData.push(i);
-    }
+  for (const key in assistanceData.value) {
+    const element: AssistanceItemType = assistanceData.value[key];
+    const yData: number[] = computeItem(element);
+    const series = {
+      type: 'line',
+      name: `${element['Hz']}Hz`,
+      smooth: true,
+      showSymbol: false,
+      symbol: 'none',
+      emphasis: {
+        focus: 'series',
+      },
+      itemStyle: { normal: { label: { show: true } } },
+      lineStyle: {
+        width: 1,
+        color: '#ffffff88',
+      },
+      zlevel: 0,
+      z: 1,
+      endLabel: {
+        show: true,
+        formatter: '{a}',
+        distance: 0,
+        color: '#39E9FE99',
+        backgroundColor: 'transparent',
+        padding: [3, 3, 2, 3],
+      },
+      data: yData,
+    };
+
+    yDataList.push(series);
   }
 
-  function computeUH(Hz: number, uQ: number) {
-    debugger;
-    const equation = assistanceData.value.find((item) => {
-      return Math.ceil(Hz) == item['Hz'];
-    });
-    if (equation) {
-      const uHMax = Math.round((equation['a'] * equation['min'] * equation['min'] + equation['b'] * equation['min'] + equation['c']) * 100) / 100;
-      const uHMin = Math.round((equation['a'] * equation['max'] * equation['max'] + equation['b'] * equation['max'] + equation['c']) * 100) / 100;
-      const uH1 = Math.round((equation['a'] * uQ * uQ + equation['b'] * uQ + equation['c']) * 100) / 100;
-      if (uH1 >= uHMin && uH1 <= uHMax) {
-        uH.value = uH1;
-        isHaCross.value = true;
-      } else {
-        isHaCross.value = false;
-      }
-    }
+  for (let i = xDataMin; i <= xDataMax; i++) {
+    xData.push(i);
   }
+}
 
-  function computeRLine() {
-    console.log('计算后的风量为------------>', uQ.value);
-    if (uH.value && uQ.value) {
-      const R = uH.value / Number(uQ.value) / Number(uQ.value);
-      const yAxis: number[] = [];
-      for (let i = 0; i < xData.length; i++) {
-        const x = xData[i];
-        const y = R * x * x;
-        if (x == uQ.value) {
-          uH.value = y;
-        }
-        yAxis.push(y);
-      }
-      const series = {
-        name: 'R',
-        type: 'line',
-        smooth: true,
-        showSymbol: false,
-        zlevel: 0,
-        emphasis: {
-          focus: 'series',
-        },
-        itemStyle: { normal: { label: { show: true } } },
-        lineStyle: {
-          width: 2,
-          color: '#D0A343',
-        },
-        endLabel: {
-          show: true,
-          formatter: '{a}',
-          distance: 0,
-          color: '#D0A343',
-        },
-        data: yAxis,
-      };
-      yDataList[lineNum] = series;
+function computeUH(Hz: number, uQ: number) {
+  debugger;
+  const equation = assistanceData.value.find((item) => {
+    return Math.ceil(Hz) == item['Hz'];
+  });
+  if (equation) {
+    const uHMax = Math.round((equation['a'] * equation['min'] * equation['min'] + equation['b'] * equation['min'] + equation['c']) * 100) / 100;
+    const uHMin = Math.round((equation['a'] * equation['max'] * equation['max'] + equation['b'] * equation['max'] + equation['c']) * 100) / 100;
+    const uH1 = Math.round((equation['a'] * uQ * uQ + equation['b'] * uQ + equation['c']) * 100) / 100;
+    if (uH1 >= uHMin && uH1 <= uHMax) {
+      uH.value = uH1;
+      isHaCross.value = true;
+    } else {
+      isHaCross.value = false;
     }
   }
+}
 
-  function reSetLine() {
-    let minIndex = -1;
-    for (let i = 0; i < yDataList.length; i++) {
-      if (i !== lineNum && i != lineNum + 1) {
-        if (yDataList[i]['lineStyle']) yDataList[i]['lineStyle']['color'] = '#ffffff88';
-        if (yDataList[i]['lineStyle']) yDataList[i]['lineStyle']['width'] = 1;
-        if (yDataList[i]['endLabel'] && yDataList[i]['endLabel']['color']) {
-          yDataList[i]['endLabel']['color'] = '#39E9FE99';
-        }
-        if (yDataList[i]['endLabel'] && yDataList[i]['endLabel']['backgroundColor']) yDataList[i]['endLabel']['backgroundColor'] = 'transparent';
-        if (yDataList[i]['z']) yDataList[i]['z'] = 1;
+function computeRLine() {
+  console.log('计算后的风量为------------>', uQ.value);
+  if (uH.value && uQ.value) {
+    const R = uH.value / Number(uQ.value) / Number(uQ.value);
+    const yAxis: number[] = [];
+    for (let i = 0; i < xData.length; i++) {
+      const x = xData[i];
+      const y = R * x * x;
+      if (x == uQ.value) {
+        uH.value = y;
       }
-
-      if (resultObj.value && `${resultObj.value.Hz}Hz` == yDataList[i]['name']) {
-        minIndex = i;
+      yAxis.push(y);
+    }
+    const series = {
+      name: 'R',
+      type: 'line',
+      smooth: true,
+      showSymbol: false,
+      zlevel: 0,
+      emphasis: {
+        focus: 'series',
+      },
+      itemStyle: { normal: { label: { show: true } } },
+      lineStyle: {
+        width: 2,
+        color: '#D0A343',
+      },
+      endLabel: {
+        show: true,
+        formatter: '{a}',
+        distance: 0,
+        color: '#D0A343',
+      },
+      data: yAxis,
+    };
+    yDataList[lineNum] = series;
+  }
+}
+
+function reSetLine() {
+  let minIndex = -1;
+  for (let i = 0; i < yDataList.length; i++) {
+    if (i !== lineNum && i != lineNum + 1) {
+      if (yDataList[i]['lineStyle']) yDataList[i]['lineStyle']['color'] = '#ffffff88';
+      if (yDataList[i]['lineStyle']) yDataList[i]['lineStyle']['width'] = 1;
+      if (yDataList[i]['endLabel'] && yDataList[i]['endLabel']['color']) {
+        yDataList[i]['endLabel']['color'] = '#39E9FE99';
       }
+      if (yDataList[i]['endLabel'] && yDataList[i]['endLabel']['backgroundColor']) yDataList[i]['endLabel']['backgroundColor'] = 'transparent';
+      if (yDataList[i]['z']) yDataList[i]['z'] = 1;
     }
-    if (minIndex != -1) {
-      yDataList[minIndex]['lineStyle']['color'] = '#9A60B4';
-      yDataList[minIndex]['lineStyle']['width'] = 2;
-      yDataList[minIndex]['endLabel']['color'] = '#9A60B4';
-      yDataList[minIndex]['endLabel']['backgroundColor'] = '#111';
-      yDataList[minIndex]['z'] = 999;
+
+    if (resultObj.value && `${resultObj.value.Hz}Hz` == yDataList[i]['name']) {
+      minIndex = i;
     }
   }
-
-  // 根据风量计算压差
-  function computePa() {
-    debugger;
-    const R = uH.value / Number(uQ.value) / Number(uQ.value);
-    const pointX = Number(uQ.value);
-    const pointY = Number(uH.value);
-    type ItemType = {
-      x: number;
-      y: number;
-      Hz: number;
-    };
-    const paList = new Map<number, ItemType>(); // key 是最近距离
-    const getIntersectionPoint = (a, b, c, R, min, max) => {
-      const obj: { x: undefined | number; y: undefined | number; min: number; max: number } = { x: undefined, y: undefined, min: min, max: max };
-      // 计算二次方程的判别式
-      const discriminant = b * b - 4 * (a - R) * c;
-
-      if (discriminant > 0) {
-        // 有两个实根
-        const x1 = (-b + Math.sqrt(discriminant)) / (2 * (a - R));
-        const x2 = (-b - Math.sqrt(discriminant)) / (2 * (a - R));
-
-        const y1 = R * x1 * x1;
-        const y2 = R * x2 * x2;
-        if (x1 >= min && x1 <= max) {
-          obj.x = x1;
-          obj.y = y1;
-        } else {
-          obj.x = x2;
-          obj.y = y2;
-        }
-      } else if (discriminant === 0) {
-        // 有一个实根
-        const x = -b / (2 * (a - R));
-        const y = R * x * x;
-        if (x >= min && x <= max) {
-          obj.x = x;
-          obj.y = y;
-        }
-        // console.log(`唯一交点: (${x}, ${y})`);
+  if (minIndex != -1) {
+    yDataList[minIndex]['lineStyle']['color'] = '#9A60B4';
+    yDataList[minIndex]['lineStyle']['width'] = 2;
+    yDataList[minIndex]['endLabel']['color'] = '#9A60B4';
+    yDataList[minIndex]['endLabel']['backgroundColor'] = '#111';
+    yDataList[minIndex]['z'] = 999;
+  }
+}
+
+// 根据风量计算压差
+function computePa() {
+  debugger;
+  const R = uH.value / Number(uQ.value) / Number(uQ.value);
+  const pointX = Number(uQ.value);
+  const pointY = Number(uH.value);
+  type ItemType = {
+    x: number;
+    y: number;
+    Hz: number;
+  };
+  const paList = new Map<number, ItemType>(); // key 是最近距离
+  const getIntersectionPoint = (a, b, c, R, min, max) => {
+    const obj: { x: undefined | number; y: undefined | number; min: number; max: number } = { x: undefined, y: undefined, min: min, max: max };
+    // 计算二次方程的判别式
+    const discriminant = b * b - 4 * (a - R) * c;
+
+    if (discriminant > 0) {
+      // 有两个实根
+      const x1 = (-b + Math.sqrt(discriminant)) / (2 * (a - R));
+      const x2 = (-b - Math.sqrt(discriminant)) / (2 * (a - R));
+
+      const y1 = R * x1 * x1;
+      const y2 = R * x2 * x2;
+      if (x1 >= min && x1 <= max) {
+        obj.x = x1;
+        obj.y = y1;
       } else {
-        // 没有实根,交点在虚数域
-        console.log('没有实数交点');
-        isHaCross.value = false;
+        obj.x = x2;
+        obj.y = y2;
       }
-      return obj;
-    };
-
-    for (let key in assistanceData.value) {
-      const item: AssistanceItemType = assistanceData.value[key];
-      paList.set(item.Hz, getIntersectionPoint(item.a, item.b, item.c, R, item.min, item.max));
+    } else if (discriminant === 0) {
+      // 有一个实根
+      const x = -b / (2 * (a - R));
+      const y = R * x * x;
+      if (x >= min && x <= max) {
+        obj.x = x;
+        obj.y = y;
+      }
+      // console.log(`唯一交点: (${x}, ${y})`);
+    } else {
+      // 没有实根,交点在虚数域
+      console.log('没有实数交点');
+      isHaCross.value = false;
     }
+    return obj;
+  };
 
-    const min = (points: Map<number, ItemType>) => {
-      const targetX = uQ.value;
-      const targetY = uH.value;
-      let minDistance = Number.POSITIVE_INFINITY;
-      let closestPoint = null;
-      let keyVal = '';
-      // 遍历已知点数组,计算距离并更新最小距离和对应的点
-      for (const [key, point] of points) {
-        const distance = Math.sqrt((targetX - point.x) ** 2 + (targetY - point.y) ** 2);
-        if (distance < minDistance) {
-          minDistance = distance;
-          closestPoint = point;
-          keyVal = key;
-        }
+  for (let key in assistanceData.value) {
+    const item: AssistanceItemType = assistanceData.value[key];
+    paList.set(item.Hz, getIntersectionPoint(item.a, item.b, item.c, R, item.min, item.max));
+  }
+
+  const min = (points: Map<number, ItemType>) => {
+    const targetX = uQ.value;
+    const targetY = uH.value;
+    let minDistance = Number.POSITIVE_INFINITY;
+    let closestPoint = null;
+    let keyVal = '';
+    // 遍历已知点数组,计算距离并更新最小距离和对应的点
+    for (const [key, point] of points) {
+      const distance = Math.sqrt((targetX - point.x) ** 2 + (targetY - point.y) ** 2);
+      if (distance < minDistance) {
+        minDistance = distance;
+        closestPoint = point;
+        keyVal = key;
       }
+    }
 
-      if (closestPoint !== null) {
-        // console.log(`距离最小的点是 (${closestPoint.x}, ${closestPoint.y}), 距离为 ${minDistance}`);
-        if (closestPoint.x < closestPoint.min || closestPoint.x > closestPoint.max) {
-          resultObj.value = null;
-          isHaCross.value = false;
-          console.log('没有找到最小距离的点');
-        } else {
-          resultObj.value = { x: closestPoint.x, y: closestPoint.y, Hz: keyVal };
-          isHaCross.value = true;
-        }
-      } else {
+    if (closestPoint !== null) {
+      // console.log(`距离最小的点是 (${closestPoint.x}, ${closestPoint.y}), 距离为 ${minDistance}`);
+      if (closestPoint.x < closestPoint.min || closestPoint.x > closestPoint.max) {
         resultObj.value = null;
         isHaCross.value = false;
         console.log('没有找到最小距离的点');
+      } else {
+        resultObj.value = { x: closestPoint.x, y: closestPoint.y, Hz: keyVal };
+        isHaCross.value = true;
       }
-    };
-    min(paList);
-  }
-
-  function computeR() {
-    if (uQ.value && uH.value) {
-      computeRLine();
-      computePa();
-      const x = resultObj.value && resultObj.value.x ? resultObj.value.x.toFixed(0) : -100;
-      const y = resultObj.value && resultObj.value.y ? Number(resultObj.value.y.toFixed(0)) : -100;
-      const series = {
-        type: 'effectScatter',
-        symbolSize: 5,
-        // symbolOffset:[1, 1],
-        showEffectOn: 'render',
-        // 涟漪特效相关配置。
-        rippleEffect: {
-          // 波纹的绘制方式,可选 'stroke' 和 'fill'。
-          brushType: 'stroke',
-        },
-        zlevel: 1,
-        z: 999,
-        itemStyle: {
-          color: '#C60000',
-        },
-        data: [[x, y]],
-      };
-      yDataList[lineNum + 1] = series;
+    } else {
+      resultObj.value = null;
+      isHaCross.value = false;
+      console.log('没有找到最小距离的点');
     }
-    // // 根据计算后的得到的频率,风量,瓦斯浓度
-    // getMonitor();
+  };
+  min(paList);
+}
+
+function computeR() {
+  if (uQ.value && uH.value) {
+    computeRLine();
+    computePa();
+    const x = resultObj.value && resultObj.value.x ? resultObj.value.x.toFixed(0) : -100;
+    const y = resultObj.value && resultObj.value.y ? Number(resultObj.value.y.toFixed(0)) : -100;
+    const series = {
+      type: 'effectScatter',
+      symbolSize: 5,
+      // symbolOffset:[1, 1],
+      showEffectOn: 'render',
+      // 涟漪特效相关配置。
+      rippleEffect: {
+        // 波纹的绘制方式,可选 'stroke' 和 'fill'。
+        brushType: 'stroke',
+      },
+      zlevel: 1,
+      z: 999,
+      itemStyle: {
+        color: '#C60000',
+      },
+      data: [[x, y]],
+    };
+    yDataList[lineNum + 1] = series;
   }
+  // // 根据计算后的得到的频率,风量,瓦斯浓度
+  // getMonitor();
+}
+
+function getMonitor() {
+  clearTimeout(timer);
+  timer = undefined;
+  const n = 10;
+  const windQuantity = uQ1.value;
+  const obj = {
+    m3: windQuantity,
+    gas: gasWarningVal.value,
+    time: '',
+    Hz: uHz.value,
+  };
+  const monitorList: { m3: number; gas: number; Hz: number; time: string }[] = [];
+  for (let i = 0; i < n; i++) {
+    const item = cloneDeep(obj);
+    const m3Temp = (Math.random() * 2 - 1) * Math.random() * 20;
+    const gas = m3Temp * 0.0002;
+    item.time = dayjs(new Date().getTime() - (n + i) * 3000).format('HH:mm:ss');
+    item.m3 = (obj.m3 + m3Temp).toFixed(2);
+    item.gas = (obj.gas + gas).toFixed(4);
+    item.Hz = (obj.Hz + Math.random()).toFixed(2);
+    monitorList.unshift(item);
+  }
+  monitorData.value = cloneDeep(monitorList);
 
-  function getMonitor() {
+  setTimeout(() => {
     clearTimeout(timer);
     timer = undefined;
-    const n = 10;
-    const windQuantity = uQ1.value;
-    const obj = {
-      m3: windQuantity,
-      gas: gasWarningVal.value,
-      time: '',
-      Hz: uHz.value,
-    };
-    const monitorList: { m3: number; gas: number; Hz: number; time: string }[] = [];
-    for (let i = 0; i < n; i++) {
-      const item = cloneDeep(obj);
-      const m3Temp = (Math.random() * 2 - 1) * Math.random() * 20;
-      const gas = m3Temp * 0.0002;
-      item.time = dayjs(new Date().getTime() - (n + i) * 3000).format('HH:mm:ss');
-      item.m3 = (obj.m3 + m3Temp).toFixed(2);
-      item.gas = (obj.gas + gas).toFixed(4);
-      item.Hz = (obj.Hz + Math.random()).toFixed(2);
-      monitorList.unshift(item);
-    }
-    monitorData.value = cloneDeep(monitorList);
-
-    setTimeout(() => {
-      clearTimeout(timer);
-      timer = undefined;
 
-      timer = 0;
-      if (uQ1.value && uH.value && uQ.value) {
-        openTimer(cloneDeep(obj), 0);
-      } else {
-        openTimer(cloneDeep(obj), 0);
-      }
-    }, 1000);
-  }
-
-  function startCompute() {
-    setTimeout(() => {
-      message.success('指令下发成功!');
-      isStartCompute.value = 1;
-    }, 800);
-  }
-
-  function resetCompute() {
-    setTimeout(() => {
-      message.success('指令下发成功!');
-      isStartCompute.value = -1;
-      tempList.length = 0;
-      n = 0;
-    }, 800);
-  }
-
-  let timer = undefined;
-  let n = 0;
-  let tempList = [];
-  function openTimer(obj: { m3: number; gas: number; Hz: number; time: string }) {
-    // 打开定时器
-    const monitorList = cloneDeep(monitorData.value);
-    if (timer !== undefined) {
-      timer = setTimeout(() => {
-        obj.m3 = Number(obj.m3);
-        obj.gas = Number(obj.gas);
-        obj.Hz = Number(obj.Hz);
-        const item = cloneDeep(obj);
-        item.time = dayjs(new Date().getTime()).format('HH:mm:ss');
-        if (resultObj.value && resultObj.value.x && resultObj.value.y && isStartCompute.value != 0) {
-          if (isStartCompute.value == 1 && Number(obj.m3) > uQ.value && Number(obj.gas) < gasWarningMax.value) {
-            isStartCompute.value = 0;
-            n = 0;
-          } else if (isStartCompute.value == -1 && Number(obj.m3) < uQ1.value) {
-            isStartCompute.value = 0;
-            n = 0;
-          }
+    timer = 0;
+    if (uQ1.value && uH.value && uQ.value) {
+      openTimer(cloneDeep(obj), 0);
+    } else {
+      openTimer(cloneDeep(obj), 0);
+    }
+  }, 1000);
+}
+
+function startCompute() {
+  setTimeout(() => {
+    message.success('指令下发成功!');
+    isStartCompute.value = 1;
+  }, 800);
+}
+
+function resetCompute() {
+  setTimeout(() => {
+    message.success('指令下发成功!');
+    isStartCompute.value = -1;
+    tempList.length = 0;
+    n = 0;
+  }, 800);
+}
+
+let timer = undefined;
+let n = 0;
+let tempList = [];
+function openTimer(obj: { m3: number; gas: number; Hz: number; time: string }) {
+  // 打开定时器
+  const monitorList = cloneDeep(monitorData.value);
+  if (timer !== undefined) {
+    timer = setTimeout(() => {
+      obj.m3 = Number(obj.m3);
+      obj.gas = Number(obj.gas);
+      obj.Hz = Number(obj.Hz);
+      const item = cloneDeep(obj);
+      item.time = dayjs(new Date().getTime()).format('HH:mm:ss');
+      if (resultObj.value && resultObj.value.x && resultObj.value.y && isStartCompute.value != 0) {
+        if (isStartCompute.value == 1 && Number(obj.m3) > uQ.value && Number(obj.gas) < gasWarningMax.value) {
+          isStartCompute.value = 0;
+          n = 0;
+        } else if (isStartCompute.value == -1 && Number(obj.m3) < uQ1.value) {
+          isStartCompute.value = 0;
+          n = 0;
         }
+      }
 
-        if (!isStartCompute.value) {
-          const m3Temp = (Math.random() * 2 - 1) * Math.random() * Math.random() * 10;
+      if (!isStartCompute.value) {
+        const m3Temp = (Math.random() * 2 - 1) * Math.random() * Math.random() * 10;
+        const gas = m3Temp * 0.0002;
+        item.m3 = (obj.m3 + m3Temp).toFixed(2);
+        item.gas = (obj.gas + gas).toFixed(4);
+        item.Hz = (obj.Hz + Math.random()).toFixed(2);
+        n = 0;
+      } else {
+        if (n < 2) {
+          const item1 = cloneDeep(obj);
+          const m3Temp = Math.random() * Math.random() * 10;
           const gas = m3Temp * 0.0002;
-          item.m3 = (obj.m3 + m3Temp).toFixed(2);
-          item.gas = (obj.gas + gas).toFixed(4);
-          item.Hz = (obj.Hz + Math.random()).toFixed(2);
-          n = 0;
-        } else {
-          if (n < 2) {
-            const item1 = cloneDeep(obj);
-            const m3Temp = Math.random() * Math.random() * 10;
-            const gas = m3Temp * 0.0002;
-            item1.m3 = (obj.m3 + m3Temp).toFixed(2);
-            item1.gas = (obj.gas + gas).toFixed(4);
-            item1.Hz = (obj.Hz + (Math.random() * 2 - 1) * Math.random()).toFixed(2);
-            tempList.push(item1);
-          }
+          item1.m3 = (obj.m3 + m3Temp).toFixed(2);
+          item1.gas = (obj.gas + gas).toFixed(4);
+          item1.Hz = (obj.Hz + (Math.random() * 2 - 1) * Math.random()).toFixed(2);
+          tempList.push(item1);
+        }
 
-          // 计算  uQ1 -> uQ  gas -> gasWarningMax.value Hz -> resultObj.value.Hz
-          if (isStartCompute.value == 1) {
-            if (resultObj.value.Hz - obj.Hz > 0) {
-              item.Hz = (uHz.value + (2 * n * (resultObj.value.Hz - uHz.value)) / 20).toFixed(2);
-              if (resultObj.value.Hz - obj.Hz < 0) {
-                item.Hz = (resultObj.value.Hz + Math.random() * 0.5).toFixed(2);
-              }
-            } else {
-              // item.Hz = (resultObj.value.Hz + (Math.random() * 2 - 1) * Math.random() * 2).toFixed(2);
+        // 计算  uQ1 -> uQ  gas -> gasWarningMax.value Hz -> resultObj.value.Hz
+        if (isStartCompute.value == 1) {
+          if (resultObj.value.Hz - obj.Hz > 0) {
+            item.Hz = (uHz.value + (2 * n * (resultObj.value.Hz - uHz.value)) / 20).toFixed(2);
+            if (resultObj.value.Hz - obj.Hz < 0) {
               item.Hz = (resultObj.value.Hz + Math.random() * 0.5).toFixed(2);
             }
-            item.m3 = (obj.m3 + (0.027 * n * item.Hz * (item.Hz - uHz.value) * (uQ.value - uQ1.value)) / 600).toFixed(2);
-            if (Number(item.m3) > uQ.value) {
-              item.m3 = (uQ.value + Math.random() * 10).toFixed(2);
-            }
-            item.gas = (
-              gasWarningVal.value -
-              (0.015 * item.Hz * n * (item.Hz - uHz.value) * (gasWarningVal.value - gasWarningMax.value)) / 200
-            ).toFixed(4);
           } else {
-            // 复位
-            if (obj.Hz - uHz.value > 0) {
-              item.Hz = (obj.Hz - (2 * n * (resultObj.value.Hz - uHz.value)) / 30).toFixed(2);
-              if (item.Hz - uHz.value < 0) {
-                item.Hz = uHz.value - 0.3;
-              }
-            } else {
-              item.Hz = (uHz.value + (Math.random() * 2 - 1) * Math.random() * 2).toFixed(2);
-            }
-            item.m3 = (obj.m3 - (0.02 * n * n * item.Hz * (resultObj.value.Hz - uHz.value) * (uQ.value - uQ1.value)) / 400).toFixed(2);
-            if (item.m3 < uQ1.value) {
-              item.m3 = uQ1.value - Math.random() * 10;
-            }
-            item.gas = (obj.gas + (0.015 * item.Hz * n * (uHz.value - item.Hz) * (gasWarningVal.value - gasWarningMax.value)) / 800).toFixed(4);
+            // item.Hz = (resultObj.value.Hz + (Math.random() * 2 - 1) * Math.random() * 2).toFixed(2);
+            item.Hz = (resultObj.value.Hz + Math.random() * 0.5).toFixed(2);
           }
-        }
-        if (monitorList.length >= 10) {
-          monitorList.shift();
-          const monitor = cloneDeep(item);
-          const data = tempList.shift();
-          if (data) {
-            item['m3'] = data['m3'];
-            item['gas'] = data['gas'];
+          item.m3 = (obj.m3 + (0.027 * n * item.Hz * (item.Hz - uHz.value) * (uQ.value - uQ1.value)) / 600).toFixed(2);
+          if (Number(item.m3) > uQ.value) {
+            item.m3 = (uQ.value + Math.random() * 10).toFixed(2);
           }
-          tempList.push(monitor);
-          monitorList.push(item);
+          item.gas = (
+            gasWarningVal.value -
+            (0.015 * item.Hz * n * (item.Hz - uHz.value) * (gasWarningVal.value - gasWarningMax.value)) / 200
+          ).toFixed(4);
         } else {
-          monitorList.push(item);
-        }
-        monitorData.value = monitorList;
-        // console.log('瓦斯监测数据-------------->', monitorData.value);
-        if (timer) {
-          if (!isStartCompute.value) {
-            // n++;
-            openTimer(cloneDeep(obj));
+          // 复位
+          if (obj.Hz - uHz.value > 0) {
+            item.Hz = (obj.Hz - (2 * n * (resultObj.value.Hz - uHz.value)) / 30).toFixed(2);
+            if (item.Hz - uHz.value < 0) {
+              item.Hz = uHz.value - 0.3;
+            }
           } else {
-            n++;
-            openTimer(cloneDeep(item));
+            item.Hz = (uHz.value + (Math.random() * 2 - 1) * Math.random() * 2).toFixed(2);
+          }
+          item.m3 = (obj.m3 - (0.02 * n * n * item.Hz * (resultObj.value.Hz - uHz.value) * (uQ.value - uQ1.value)) / 400).toFixed(2);
+          if (item.m3 < uQ1.value) {
+            item.m3 = uQ1.value - Math.random() * 10;
           }
+          item.gas = (obj.gas + (0.015 * item.Hz * n * (uHz.value - item.Hz) * (gasWarningVal.value - gasWarningMax.value)) / 800).toFixed(4);
         }
-      }, 3000);
-    }
+      }
+      if (monitorList.length >= 10) {
+        monitorList.shift();
+        const monitor = cloneDeep(item);
+        const data = tempList.shift();
+        if (data) {
+          item['m3'] = data['m3'];
+          item['gas'] = data['gas'];
+        }
+        tempList.push(monitor);
+        monitorList.push(item);
+      } else {
+        monitorList.push(item);
+      }
+      monitorData.value = monitorList;
+      // console.log('瓦斯监测数据-------------->', monitorData.value);
+      if (timer) {
+        if (!isStartCompute.value) {
+          // n++;
+          openTimer(cloneDeep(obj));
+        } else {
+          n++;
+          openTimer(cloneDeep(item));
+        }
+      }
+    }, 3000);
   }
+}
 
-  function edit(flag) {
-    if (flag == 'info') {
-      formType.value = '编辑风机信息';
-    }
-    if (flag == 'line') {
-      formType.value = '编辑特性曲线';
-    }
-    if (formShow.value == true) {
-      formShow.value = false;
-      nextTick(() => {
-        formShow.value = true;
-      });
-    } else {
-      formShow.value = true;
-    }
+function edit(flag) {
+  if (flag == 'info') {
+    formType.value = '编辑风机信息';
   }
-
-  function onSubmit() {
-    emit('close');
-    closeModal();
-    ChartRef.value = null;
-    uQ.value = undefined;
-    uH.value = undefined;
-    formType.value = '';
-    refresh.value = true;
-    xData.length = 0;
-    yDataList.length = 0;
-    lineNum = 0;
-    lineEquation.value = [];
-    resultObj.value = null;
-    monitorData.value = [];
-    clearTimeout(timer);
-    timer = undefined;
+  if (flag == 'line') {
+    formType.value = '编辑特性曲线';
   }
-
-  async function onCancel() {
-    return new Promise((resolve) => {
-      if (isStartCompute.value == 0) {
-        onSubmit();
-        resolve(true);
-      } else {
-        message.warning('为保障矿井安全生产,请确保已复位!!!');
-        resolve(false);
-      }
+  if (formShow.value == true) {
+    formShow.value = false;
+    nextTick(() => {
+      formShow.value = true;
     });
+  } else {
+    formShow.value = true;
   }
-
-  function initEcharts() {
-    if (ChartRef.value) {
-      reSetLine();
-      myChart.value = echarts.init(ChartRef.value);
-      option && myChart.value.setOption(option);
-      refresh.value = false;
-      getMonitor();
-      nextTick(() => {
-        setTimeout(() => {
-          refresh.value = true;
-        }, 0);
-      });
+}
+
+function onSubmit() {
+  emit('close');
+  closeModal();
+  ChartRef.value = null;
+  uQ.value = undefined;
+  uH.value = undefined;
+  formType.value = '';
+  refresh.value = true;
+  xData.length = 0;
+  yDataList.length = 0;
+  lineNum = 0;
+  lineEquation.value = [];
+  resultObj.value = null;
+  monitorData.value = [];
+  clearTimeout(timer);
+  timer = undefined;
+}
+
+async function onCancel() {
+  return new Promise((resolve) => {
+    if (isStartCompute.value == 0) {
+      onSubmit();
+      resolve(true);
+    } else {
+      message.warning('为保障矿井安全生产,请确保已复位!!!');
+      resolve(false);
     }
-  }
-
-  function makeLine() {
-    if (uQ.value && uH.value) {
-      loadding.value = true;
+  });
+}
+
+function initEcharts() {
+  if (ChartRef.value) {
+    reSetLine();
+    myChart.value = echarts.init(ChartRef.value);
+    option && myChart.value.setOption(option);
+    refresh.value = false;
+    getMonitor();
+    nextTick(() => {
       setTimeout(() => {
-        computeR();
-        reSetLine();
-        option && myChart.value.setOption(option);
-        loadding.value = false;
-      }, 1200);
-    }
+        refresh.value = true;
+      }, 0);
+    });
   }
-  function handleSubmit() {
-    message.success('提交成功');
+}
+
+function makeLine() {
+  if (uQ.value && uH.value) {
+    loadding.value = true;
     setTimeout(() => {
-      formShow.value = false;
-    }, 800);
+      computeR();
+      reSetLine();
+      option && myChart.value.setOption(option);
+      loadding.value = false;
+    }, 1200);
   }
-  onMounted(() => {
-    timer = undefined;
-  });
+}
+function handleSubmit() {
+  message.success('提交成功');
+  setTimeout(() => {
+    formShow.value = false;
+  }, 800);
+}
+onMounted(() => {
+  timer = undefined;
+});
 </script>
 
 <style scoped lang="less">
-  .modal-box {
+.modal-box {
+  display: flex;
+  flex-direction: row;
+  background-color: #ffffff05;
+  padding: 20px 8px 0 8px;
+  border: 1px solid #00d8ff22;
+  position: relative;
+  // min-height: 600px;
+  .box-title {
+    width: calc(100% - 40px);
+    text-align: center;
+    background-color: #1dc1f522;
+  }
+  .info-item {
     display: flex;
-    flex-direction: row;
-    background-color: #ffffff05;
-    padding: 20px 8px 0 8px;
-    border: 1px solid #00d8ff22;
-    position: relative;
-    // min-height: 600px;
-    .box-title {
-      width: calc(100% - 40px);
-      text-align: center;
-      background-color: #1dc1f522;
+    justify-content: space-between;
+    align-items: center;
+    padding: 2px 0px;
+    margin: 10px 0;
+    line-height: 30px;
+    background-image: linear-gradient(to right, #39deff15, #3977e500);
+    &:first-child {
+      margin-top: 0;
+    }
+    .title {
+      width: 200px;
+      text-align: left;
+      padding-left: 20px;
+      color: #f1f1f1cc;
+    }
+    .value {
+      width: 150px;
+      color: #00d8ff;
+      padding-right: 20px;
+      text-align: right;
     }
-    .info-item {
-      display: flex;
-      justify-content: space-between;
-      align-items: center;
-      padding: 2px 0px;
-      margin: 10px 0;
-      line-height: 30px;
-      background-image: linear-gradient(to right, #39deff15, #3977e500);
-      &:first-child {
-        margin-top: 0;
-      }
+  }
+  .right-box {
+    width: 320px;
+    .info-lines {
+      width: calc(100% - 2px);
+      height: 450px;
+      box-shadow: 0px 0px 50px #86baff08 inset;
+      overflow-y: auto;
+      margin-top: 5px;
       .title {
-        width: 200px;
-        text-align: left;
-        padding-left: 20px;
+        width: 100%;
         color: #f1f1f1cc;
       }
-      .value {
-        width: 150px;
-        color: #00d8ff;
-        padding-right: 20px;
-        text-align: right;
-      }
     }
-    .right-box {
-      width: 320px;
-      .info-lines {
-        width: calc(100% - 2px);
-        height: 450px;
-        box-shadow: 0px 0px 50px #86baff08 inset;
-        overflow-y: auto;
-        margin-top: 5px;
-        .title {
-          width: 100%;
-          color: #f1f1f1cc;
-        }
-      }
+  }
+  .center-box {
+    flex: 1;
+    margin: 0 10px;
+    display: flex;
+    .info-echarts {
+      // background-color: #ffffff11;
     }
-    .center-box {
-      flex: 1;
-      margin: 0 10px;
+    .result-tip {
+      text-align: center;
+      background-color: #00000011;
+      line-height: 28px;
+      margin: 10px 50px 0 50px;
+      border: 1px solid #00d8ff22;
+      border-radius: 2px;
+    }
+  }
+  .tip-box {
+    width: 1040px;
+    height: 44px;
+    position: absolute;
+    top: 417px;
+    display: flex;
+    padding: 0 20px;
+    .title {
+      width: 142px;
+      height: 43px;
       display: flex;
-      .info-echarts {
-        // background-color: #ffffff11;
-      }
-      .result-tip {
-        text-align: center;
-        background-color: #00000011;
-        line-height: 28px;
-        margin: 10px 50px 0 50px;
-        border: 1px solid #00d8ff22;
-        border-radius: 2px;
+      align-items: center;
+      padding-left: 30px;
+      background-image: url('@/assets/images/fanlocal-tip/tip-title.png');
+      position: relative;
+      &::before {
+        content: '';
+        display: inline-block;
+        position: absolute;
+        width: 31px;
+        height: 31px;
+        top: 5px;
+        left: -8px;
+        background-image: url('@/assets/images/fanlocal-tip/tip-icon.png');
       }
     }
-    .tip-box {
-      width: 1040px;
+    .tip-container {
+      width: 898px;
       height: 44px;
-      position: absolute;
-      top: 417px;
+      line-height: 44px;
       display: flex;
-      padding: 0 20px;
-      .title {
-        width: 142px;
-        height: 43px;
-        display: flex;
-        align-items: center;
-        padding-left: 30px;
-        background-image: url('@/assets/images/fanlocal-tip/tip-title.png');
-        position: relative;
-        &::before {
-          content: '';
-          display: inline-block;
-          position: absolute;
-          width: 31px;
-          height: 31px;
-          top: 5px;
-          left: -8px;
-          background-image: url('@/assets/images/fanlocal-tip/tip-icon.png');
-        }
-      }
-      .tip-container {
-        width: 898px;
-        height: 44px;
-        line-height: 44px;
-        display: flex;
-        background-image: url('@/assets/images/fanlocal-tip/tip-bg.png');
-        background-size: cover;
-      }
+      background-image: url('@/assets/images/fanlocal-tip/tip-bg.png');
+      background-size: cover;
     }
   }
-  .setting-box {
-    width: 1170px;
-    height: 70px;
-    margin: 10px 0;
-    background-color: #ffffff05;
-    border: 1px solid #00d8ff22;
+}
+.setting-box {
+  width: 1170px;
+  height: 70px;
+  margin: 10px 0;
+  background-color: #ffffff05;
+  border: 1px solid #00d8ff22;
+  display: flex;
+  align-items: center;
+  justify-content: space-between;
+
+  .right-inputs {
+    width: 100%;
     display: flex;
-    align-items: center;
+    height: 40px;
+    margin: 0 10px;
     justify-content: space-between;
+  }
+  .left-buttons {
+    display: flex;
+    height: 40px;
 
-    .right-inputs {
-      width: 100%;
-      display: flex;
-      height: 40px;
+    .btn {
       margin: 0 10px;
-      justify-content: space-between;
     }
-    .left-buttons {
-      display: flex;
-      height: 40px;
+  }
+  .border-clip {
+    width: 1px;
+    height: 25px;
+    border-right: 1px solid #8b8b8b77;
+  }
+  .input-title {
+    max-width: 150px;
+  }
+  .input-box {
+    width: 220px !important;
+    background: transparent !important;
+    border-color: #00d8ff44 !important;
+    margin-right: 20px;
+    color: #fff !important;
+  }
+  .btn {
+    padding: 8px 20px;
+    position: relative;
+    border-radius: 2px;
+    color: #fff;
+    width: fit-content;
+    cursor: pointer;
 
-      .btn {
-        margin: 0 10px;
-      }
-    }
-    .border-clip {
-      width: 1px;
-      height: 25px;
-      border-right: 1px solid #8b8b8b77;
-    }
-    .input-title {
-      max-width: 150px;
-    }
-    .input-box {
-      width: 220px !important;
-      background: transparent !important;
-      border-color: #00d8ff44 !important;
-      margin-right: 20px;
-      color: #fff !important;
-    }
-    .btn {
-      padding: 8px 20px;
-      position: relative;
+    &::before {
+      position: absolute;
+      display: block;
+      content: '';
+      width: calc(100% - 4px);
+      height: calc(100% - 4px);
+      top: 2px;
+      left: 2px;
       border-radius: 2px;
-      color: #fff;
-      width: fit-content;
-      cursor: pointer;
-
-      &::before {
-        position: absolute;
-        display: block;
-        content: '';
-        width: calc(100% - 4px);
-        height: calc(100% - 4px);
-        top: 2px;
-        left: 2px;
-        border-radius: 2px;
-        z-index: -1;
-      }
+      z-index: -1;
     }
+  }
 
-    .btn1 {
-      border: 1px solid #5cfaff;
+  .btn1 {
+    border: 1px solid #5cfaff;
 
-      &::before {
-        background-image: linear-gradient(#2effee92, #0cb1d592);
-      }
+    &::before {
+      background-image: linear-gradient(#2effee92, #0cb1d592);
+    }
 
-      &:hover {
-        border: 1px solid #5cfaffaa;
+    &:hover {
+      border: 1px solid #5cfaffaa;
 
-        &::before {
-          background-image: linear-gradient(#2effee72, #0cb1d572);
-        }
+      &::before {
+        background-image: linear-gradient(#2effee72, #0cb1d572);
       }
     }
   }
-  .is-open {
-    animation: open 0.5s;
-    animation-iteration-count: 1;
-    animation-fill-mode: forwards;
-    animation-timing-function: ease-in;
-  }
-  .is-close {
+}
+.is-open {
+  animation: open 0.5s;
+  animation-iteration-count: 1;
+  animation-fill-mode: forwards;
+  animation-timing-function: ease-in;
+}
+.is-close {
+  height: 0px;
+}
+
+@keyframes open {
+  0% {
     height: 0px;
   }
-
-  @keyframes open {
-    0% {
-      height: 0px;
-    }
-    100% {
-      height: fit-content;
-    }
+  100% {
+    height: fit-content;
   }
+}
 
-  @keyframes close {
-    0% {
-      height: fit-content;
-    }
-    100% {
-      height: 0px;
-    }
-  }
-  :deep(.zxm-divider-inner-text) {
-    color: #cacaca88 !important;
+@keyframes close {
+  0% {
+    height: fit-content;
   }
-  :deep(.zxm-form-item) {
-    margin-bottom: 10px;
+  100% {
+    height: 0px;
   }
+}
+:deep(.zxm-divider-inner-text) {
+  color: #cacaca88 !important;
+}
+:deep(.zxm-form-item) {
+  margin-bottom: 10px;
+}
 </style>

+ 220 - 178
src/views/vent/monitorManager/fanLocalMonitor/components/dischargeGas.vue

@@ -40,217 +40,259 @@
         </div>
       </div>
     </div>
+    <a-modal v-model:visible="modalIsShow" :title="modalTitle" :maskStyle="{ backgroundColor: '#000000aa', backdropFilter: 'blur(3px)' }">
+      <template #footer>
+        <div>
+          <a-button key="back" @click="cancel">返回</a-button>
+          <a-button key="submit" type="primary" @click="handleOk">确定</a-button>
+        </div>
+      </template>
+      <div class="modal-container">
+        <div class="vent-flex-row">
+          <ExclamationCircleFilled style="color: #ffb700; font-size: 30px" />
+          <div class="warning-text">您是否要进行紧急停止操作?</div>
+        </div>
+        <div class="vent-flex-row input-box">
+          <div class="label">操作密码:</div>
+          <a-input size="small" type="password" v-model:value="passWord" style="width: 50%" />
+        </div>
+      </div>
+    </a-modal>
   </BasicModal>
 </template>
 
 <script lang="ts" setup>
-  import { BasicModal, useModalInner } from '/@/components/Modal';
-  import { ref, nextTick, onMounted, watch } from 'vue';
-  import { option, chartsColumnListGas, echatsOption1 } from '../fanLocal.data';
-  import BarAndLine from '/@/components/chart/BarAndLine.vue';
-  import { autoAdjust } from '../fanLocal.api';
-  import { message } from 'ant-design-vue';
-  const props = defineProps({
-    data: {
-      type: Object,
-      default: () => {},
-    },
-    gasMax: {
-      type: Number,
-    },
-    fanlocalId: {
-      type: Number,
-    },
-  });
-  const emit = defineEmits(['close', 'register', 'openModal']);
-  // 注册 modal
-  const [register, { closeModal }] = useModalInner((data) => {
-    nextTick(() => {
-      if (option['xAxis']) option['xAxis']['data'] = xData;
-      option['series'] = yDataList;
-      initEcharts();
-    });
+import { BasicModal, useModalInner } from '/@/components/Modal';
+import { ref, nextTick, onMounted, watch } from 'vue';
+import { option, chartsColumnListGas, echatsOption1 } from '../fanLocal.data';
+import BarAndLine from '/@/components/chart/BarAndLine.vue';
+import { autoAdjust } from '../fanLocal.api';
+import { message } from 'ant-design-vue';
+const props = defineProps({
+  data: {
+    type: Object,
+    default: () => {},
+  },
+  gasMax: {
+    type: Number,
+  },
+  fanlocalId: {
+    type: Number,
+  },
+});
+const emit = defineEmits(['close', 'register', 'openModal']);
+// 注册 modal
+const [register, { closeModal }] = useModalInner((data) => {
+  nextTick(() => {
+    if (option['xAxis']) option['xAxis']['data'] = xData;
+    option['series'] = yDataList;
+    initEcharts();
   });
+});
 
-  const ChartRef = ref();
-  const myChart = ref();
-  const refresh = ref(true);
-  const xData: any[] = [];
-  const yDataList: [] = [];
-  const echartsData = ref<Record<string, any>[]>([]);
-  const monitorData = ref<Record<string, any>>({});
-  watch(
-    () => props.data,
-    (newVal) => {
-      // 创建新对象,合并 newVal 和 targetVolume
-      const combinedData = {
-        gasT1: newVal.gasT1,
-        gasT2: newVal.gasT2,
-        gasT3: newVal.gasT3,
-        gasT4: newVal.gasT4,
-        readTime: newVal.readTime || new Date().toISOString(), // 确保有时间字段
-      };
-      monitorData.value = combinedData;
-      if (echartsData.value.length > 20) {
-        echartsData.value.shift();
-      }
-      echartsData.value = [...echartsData.value, combinedData];
-    }
-  );
-  function initEcharts() {
-    if (ChartRef.value) {
-      myChart.value = echarts.init(ChartRef.value);
-      option && myChart.value.setOption(option);
-      refresh.value = false;
-      nextTick(() => {
-        setTimeout(() => {
-          refresh.value = true;
-        }, 0);
-      });
+const ChartRef = ref();
+const myChart = ref();
+const refresh = ref(true);
+const xData: any[] = [];
+const yDataList: [] = [];
+const echartsData = ref<Record<string, any>[]>([]);
+const monitorData = ref<Record<string, any>>({});
+const modalIsShow = ref<boolean>(false); // 是否显示模态框
+const modalTitle = ref('');
+const passWord = ref('');
+watch(
+  () => props.data,
+  (newVal) => {
+    // 创建新对象,合并 newVal 和 targetVolume
+    const combinedData = {
+      gasT1: newVal.gasT1,
+      gasT2: newVal.gasT2,
+      gasT3: newVal.gasT3,
+      gasT4: newVal.gasT4,
+      readTime: newVal.readTime || new Date().toISOString(), // 确保有时间字段
+    };
+    monitorData.value = combinedData;
+    if (echartsData.value.length > 20) {
+      echartsData.value.shift();
     }
+    echartsData.value = [...echartsData.value, combinedData];
   }
-  function onHide() {
-    closeModal();
+);
+function initEcharts() {
+  if (ChartRef.value) {
+    myChart.value = echarts.init(ChartRef.value);
+    option && myChart.value.setOption(option);
+    refresh.value = false;
+    nextTick(() => {
+      setTimeout(() => {
+        refresh.value = true;
+      }, 0);
+    });
   }
-  function stop() {
-    const params = { auto: 0, fanlocalId: props.fanlocalId };
-    autoAdjust(params)
-      .then(() => {
-        message.success('指令已下发成功!');
-      })
-      .catch(() => {
-        message.error('指令下发失败');
-      });
+}
+
+function onHide() {
+  closeModal();
+}
+function stop() {
+  modalIsShow.value = true;
+  passWord.value == '';
+}
+function cancel() {
+  modalIsShow.value = false;
+}
+function handleOk() {
+  if (passWord.value == '') {
+    message.warning('请输入密码!');
+    return;
   }
-  onMounted(() => {});
-  function onSubmit() {
-    emit('close');
-    closeModal();
+  handerFn();
+}
+function handerFn() {
+  if (passWord.value != '123456') {
+    message.error('密码错误,请重新输入!');
+    return;
   }
+  const params = { auto: 0, fanlocalId: props.fanlocalId };
+  autoAdjust(params)
+    .then(() => {
+      message.success('指令已下发成功!');
+      modalIsShow.value = false;
+      passWord.value == '';
+    })
+    .catch(() => {
+      message.error('指令下发失败');
+    });
+}
+onMounted(() => {});
+function onSubmit() {
+  emit('close');
+  closeModal();
+}
 </script>
 
 <style scoped lang="less">
-  .modal-box {
-    display: flex;
-    flex-direction: row;
-    background-color: #ffffff05;
-    padding: 10px 8px 0 8px;
-    border: 1px solid #00d8ff22;
-    position: relative;
-    // min-height: 600px;
-    .left-box {
-      flex: 1; /* 占据 3/4 的空间 */
-      background-image: url(../../../../../assets/images/dischargeGas.svg);
-      background-repeat: no-repeat;
-      background-size: contain; /* 确保背景图片完整显示 */
-      background-position: center; /* 确保背景图片居中 */
-    }
-    .right-box {
-      flex: 1; /* 占据 3/4 的空间 */
-      height: 400px;
-      width: 100%;
-    }
+.modal-box {
+  display: flex;
+  flex-direction: row;
+  background-color: #ffffff05;
+  padding: 10px 8px 0 8px;
+  border: 1px solid #00d8ff22;
+  position: relative;
+  // min-height: 600px;
+  .left-box {
+    flex: 1; /* 占据 3/4 的空间 */
+    background-image: url(../../../../../assets/images/dischargeGas.svg);
+    background-repeat: no-repeat;
+    background-size: contain; /* 确保背景图片完整显示 */
+    background-position: center; /* 确保背景图片居中 */
+  }
+  .right-box {
+    flex: 1; /* 占据 3/4 的空间 */
+    height: 400px;
+    width: 100%;
   }
-  .setting-box {
+}
+.setting-box {
+  width: 100%;
+  height: 70px;
+  margin: 10px 0;
+  background-color: #ffffff05;
+  border: 1px solid #00d8ff22;
+  display: flex;
+  align-items: center;
+  justify-content: space-between;
+
+  .right-inputs {
     width: 100%;
-    height: 70px;
-    margin: 10px 0;
-    background-color: #ffffff05;
-    border: 1px solid #00d8ff22;
     display: flex;
-    align-items: center;
+    height: 40px;
+    margin: 0 10px;
     justify-content: space-between;
+  }
+  .left-buttons {
+    display: flex;
+    height: 40px;
 
-    .right-inputs {
-      width: 100%;
-      display: flex;
-      height: 40px;
+    .btn {
       margin: 0 10px;
-      justify-content: space-between;
     }
-    .left-buttons {
-      display: flex;
-      height: 40px;
-
-      .btn {
-        margin: 0 10px;
-      }
-    }
-    .border-clip {
-      width: 1px;
-      height: 25px;
-      border-right: 1px solid #8b8b8b77;
-    }
-    .input-title {
-      max-width: 150px;
-    }
-    .input-box {
-      width: 120px !important;
-      background: transparent !important;
-      border-color: #00d8ff44 !important;
-      margin-right: 20px;
-      color: #fff !important;
-    }
-    .btn {
-      padding: 8px 20px;
-      position: relative;
-      margin: 10px;
+  }
+  .border-clip {
+    width: 1px;
+    height: 25px;
+    border-right: 1px solid #8b8b8b77;
+  }
+  .input-title {
+    max-width: 150px;
+  }
+  .input-box {
+    width: 120px !important;
+    background: transparent !important;
+    border-color: #00d8ff44 !important;
+    margin-right: 20px;
+    color: #fff !important;
+  }
+  .btn {
+    padding: 8px 20px;
+    position: relative;
+    margin: 10px;
+    border-radius: 2px;
+    color: #fff;
+    width: fit-content;
+    cursor: pointer;
+    float: right;
+    &::before {
+      position: absolute;
+      display: block;
+      content: '';
+      width: calc(100% - 4px);
+      height: calc(100% - 4px);
+      top: 2px;
+      left: 2px;
       border-radius: 2px;
-      color: #fff;
-      width: fit-content;
-      cursor: pointer;
-      float: right;
-      &::before {
-        position: absolute;
-        display: block;
-        content: '';
-        width: calc(100% - 4px);
-        height: calc(100% - 4px);
-        top: 2px;
-        left: 2px;
-        border-radius: 2px;
-        z-index: -1;
-      }
+      z-index: -1;
     }
+  }
 
-    .btn1 {
-      border: 1px solid #5cfaff;
+  .btn1 {
+    border: 1px solid #5cfaff;
 
-      &::before {
-        background-image: linear-gradient(#2effee92, #0cb1d592);
-      }
+    &::before {
+      background-image: linear-gradient(#2effee92, #0cb1d592);
+    }
 
-      &:hover {
-        border: 1px solid #5cfaffaa;
+    &:hover {
+      border: 1px solid #5cfaffaa;
 
-        &::before {
-          background-image: linear-gradient(#2effee72, #0cb1d572);
-        }
+      &::before {
+        background-image: linear-gradient(#2effee72, #0cb1d572);
       }
     }
   }
+}
 
-  @keyframes open {
-    0% {
-      height: 0px;
-    }
-    100% {
-      height: fit-content;
-    }
+@keyframes open {
+  0% {
+    height: 0px;
   }
-
-  @keyframes close {
-    0% {
-      height: fit-content;
-    }
-    100% {
-      height: 0px;
-    }
+  100% {
+    height: fit-content;
   }
-  :deep(.zxm-divider-inner-text) {
-    color: #cacaca88 !important;
+}
+
+@keyframes close {
+  0% {
+    height: fit-content;
   }
-  :deep(.zxm-form-item) {
-    margin-bottom: 10px;
+  100% {
+    height: 0px;
   }
+}
+:deep(.zxm-divider-inner-text) {
+  color: #cacaca88 !important;
+}
+:deep(.zxm-form-item) {
+  margin-bottom: 10px;
+}
 </style>

+ 224 - 183
src/views/vent/monitorManager/fanLocalMonitor/components/supplyAir.vue

@@ -39,222 +39,263 @@
         </div>
       </div>
     </div>
+    <a-modal v-model:visible="modalIsShow" :title="modalTitle" :maskStyle="{ backgroundColor: '#000000aa', backdropFilter: 'blur(3px)' }">
+      <template #footer>
+        <div>
+          <a-button key="back" @click="cancel">返回</a-button>
+          <a-button key="submit" type="primary" @click="handleOk">确定</a-button>
+        </div>
+      </template>
+      <div class="modal-container">
+        <div class="vent-flex-row">
+          <ExclamationCircleFilled style="color: #ffb700; font-size: 30px" />
+          <div class="warning-text">您是否要进行紧急停止操作?</div>
+        </div>
+        <div class="vent-flex-row input-box">
+          <div class="label">操作密码:</div>
+          <a-input size="small" type="password" v-model:value="passWord" style="width: 50%" />
+        </div>
+      </div>
+    </a-modal>
   </BasicModal>
 </template>
 
 <script lang="ts" setup>
-  import { BasicModal, useModalInner } from '/@/components/Modal';
-  import { ref, onMounted, nextTick, watch } from 'vue';
-  import echarts from '/@/utils/lib/echarts';
-  import BarAndLine from '/@/components/chart/BarAndLine.vue';
-  import { option, chartsColumnList1, echatsOption1 } from '../fanLocal.data';
-  import { autoAdjust } from '../fanLocal.api';
-  import { message } from 'ant-design-vue';
-  const emit = defineEmits(['close', 'register', 'openModal']);
-  const props = defineProps({
-    data: {
-      type: Object,
-      default: () => {},
-    },
-    targetVolume: {
-      type: Number,
-    },
-    fanlocalId: {
-      type: String,
-    },
-  });
-  // 注册 modal
-  const [register, { closeModal }] = useModalInner((data) => {
-    nextTick(() => {
-      if (option['xAxis']) option['xAxis']['data'] = xData;
-      option['series'] = yDataList;
-      initEcharts();
-    });
+import { BasicModal, useModalInner } from '/@/components/Modal';
+import { ref, onMounted, nextTick, watch } from 'vue';
+import echarts from '/@/utils/lib/echarts';
+import BarAndLine from '/@/components/chart/BarAndLine.vue';
+import { option, chartsColumnList1, echatsOption1 } from '../fanLocal.data';
+import { autoAdjust } from '../fanLocal.api';
+import { message } from 'ant-design-vue';
+import { ExclamationCircleFilled } from '@ant-design/icons-vue';
+
+const emit = defineEmits(['close', 'register', 'openModal']);
+const props = defineProps({
+  data: {
+    type: Object,
+    default: () => {},
+  },
+  targetVolume: {
+    type: Number,
+  },
+  fanlocalId: {
+    type: String,
+  },
+});
+// 注册 modal
+const [register, { closeModal }] = useModalInner((data) => {
+  nextTick(() => {
+    if (option['xAxis']) option['xAxis']['data'] = xData;
+    option['series'] = yDataList;
+    initEcharts();
   });
-  const ChartRef = ref();
-  const myChart = ref();
-  const refresh = ref(true);
-  const xData: any[] = [];
-  const yDataList: [] = [];
-  // const echartsData = ref([]);
-  // const monitorData = ref({});
-  const echartsData = ref<Record<string, any>[]>([]);
-  const monitorData = ref<Record<string, any>>({});
-  watch(
-    () => props.data,
-    (newVal) => {
-      // 创建新对象,合并 newVal 和 targetVolume
-      const combinedData = {
-        inletAirVolume_merge: newVal.inletAirVolume_merge,
-        targetVolume: props.targetVolume, // 添加目标风量
-        readTime: newVal.readTime || new Date().toISOString(), // 确保有时间字段
-      };
-      monitorData.value = combinedData;
-      if (echartsData.value.length > 20) {
-        echartsData.value.shift();
-      }
-      echartsData.value = [...echartsData.value, combinedData];
+});
+const ChartRef = ref();
+const myChart = ref();
+const refresh = ref(true);
+const xData: any[] = [];
+const yDataList: [] = [];
+const modalIsShow = ref<boolean>(false); // 是否显示模态框
+const modalTitle = ref('');
+const passWord = ref('');
+const echartsData = ref<Record<string, any>[]>([]);
+const monitorData = ref<Record<string, any>>({});
+watch(
+  () => props.data,
+  (newVal) => {
+    // 创建新对象,合并 newVal 和 targetVolume
+    const combinedData = {
+      inletAirVolume_merge: newVal.inletAirVolume_merge,
+      targetVolume: props.targetVolume, // 添加目标风量
+      readTime: newVal.readTime || new Date().toISOString(), // 确保有时间字段
+    };
+    monitorData.value = combinedData;
+    if (echartsData.value.length > 20) {
+      echartsData.value.shift();
     }
-  );
-  function onSubmit() {
-    emit('close');
-    closeModal();
+    echartsData.value = [...echartsData.value, combinedData];
   }
-  function onCancel() {
-    //
-  }
-  function initEcharts() {
-    if (ChartRef.value) {
-      myChart.value = echarts.init(ChartRef.value);
-      option && myChart.value.setOption(option);
-      refresh.value = false;
-      nextTick(() => {
-        setTimeout(() => {
-          refresh.value = true;
-        }, 0);
-      });
-    }
+);
+function initEcharts() {
+  if (ChartRef.value) {
+    myChart.value = echarts.init(ChartRef.value);
+    option && myChart.value.setOption(option);
+    refresh.value = false;
+    nextTick(() => {
+      setTimeout(() => {
+        refresh.value = true;
+      }, 0);
+    });
   }
-  function onHide() {
-    closeModal();
+}
+function onSubmit() {
+  emit('close');
+  closeModal();
+}
+function onCancel() {
+  //
+}
+function onHide() {
+  closeModal();
+}
+function stop() {
+  modalIsShow.value = true;
+  passWord.value == '';
+}
+function cancel() {
+  modalIsShow.value = false;
+}
+function handleOk() {
+  if (passWord.value == '') {
+    message.warning('请输入密码!');
+    return;
   }
-  function stop() {
-    const params = { auto: 0, fanlocalId: props.fanlocalId };
-    autoAdjust(params)
-      .then(() => {
-        message.success('指令已下发成功!');
-      })
-      .catch(() => {
-        message.error('指令下发失败');
-      });
+  handerFn();
+}
+function handerFn() {
+  if (passWord.value != '123456') {
+    message.error('密码错误,请重新输入!');
+    return;
   }
-  onMounted(() => {
-    // initEcharts();
-  });
+  const params = { auto: 0, fanlocalId: props.fanlocalId };
+  autoAdjust(params)
+    .then(() => {
+      message.success('指令已下发成功!');
+      modalIsShow.value = false;
+      passWord.value == '';
+    })
+    .catch(() => {
+      message.error('指令下发失败');
+    });
+}
+onMounted(() => {
+  // initEcharts();
+});
 </script>
 
 <style scoped lang="less">
-  .modal-box {
-    display: flex;
-    flex-direction: row;
-    background-color: #ffffff05;
-    padding: 10px 8px 0 8px;
-    border: 1px solid #00d8ff22;
-    position: relative;
-    // min-height: 600px;
-    .left-box {
-      flex: 1; /* 占据 3/4 的空间 */
-      background-image: url(../../../../../assets/images/supplyAir.svg);
-      background-repeat: no-repeat;
-      background-size: contain; /* 确保背景图片完整显示 */
-      background-position: center; /* 确保背景图片居中 */
-    }
-    .right-box {
-      flex: 1; /* 占据 3/4 的空间 */
-      height: 400px;
-      width: 100%;
-    }
+.modal-box {
+  display: flex;
+  flex-direction: row;
+  background-color: #ffffff05;
+  padding: 10px 8px 0 8px;
+  border: 1px solid #00d8ff22;
+  position: relative;
+  // min-height: 600px;
+  .left-box {
+    flex: 1; /* 占据 3/4 的空间 */
+    background-image: url(../../../../../assets/images/supplyAir.svg);
+    background-repeat: no-repeat;
+    background-size: contain; /* 确保背景图片完整显示 */
+    background-position: center; /* 确保背景图片居中 */
   }
-  .setting-box {
+  .right-box {
+    flex: 1; /* 占据 3/4 的空间 */
+    height: 400px;
+    width: 100%;
+  }
+}
+.setting-box {
+  width: 100%;
+  height: 70px;
+  margin: 10px 0;
+  background-color: #ffffff05;
+  border: 1px solid #00d8ff22;
+  display: flex;
+  align-items: center;
+  justify-content: space-between;
+
+  .right-inputs {
     width: 100%;
-    height: 70px;
-    margin: 10px 0;
-    background-color: #ffffff05;
-    border: 1px solid #00d8ff22;
     display: flex;
-    align-items: center;
+    height: 40px;
+    margin: 0 10px;
     justify-content: space-between;
+  }
+  .left-buttons {
+    display: flex;
+    height: 40px;
 
-    .right-inputs {
-      width: 100%;
-      display: flex;
-      height: 40px;
+    .btn {
       margin: 0 10px;
-      justify-content: space-between;
     }
-    .left-buttons {
-      display: flex;
-      height: 40px;
+  }
+  .border-clip {
+    width: 1px;
+    height: 25px;
+    border-right: 1px solid #8b8b8b77;
+  }
+  .input-title {
+    max-width: 150px;
+  }
+  .input-box {
+    width: 120px !important;
+    background: transparent !important;
+    border-color: #00d8ff44 !important;
+    margin-right: 20px;
+    color: #fff !important;
+  }
+  .btn {
+    padding: 8px 20px;
+    margin: 10px;
+    position: relative;
+    border-radius: 2px;
+    color: #fff;
+    width: fit-content;
+    cursor: pointer;
 
-      .btn {
-        margin: 0 10px;
-      }
-    }
-    .border-clip {
-      width: 1px;
-      height: 25px;
-      border-right: 1px solid #8b8b8b77;
-    }
-    .input-title {
-      max-width: 150px;
-    }
-    .input-box {
-      width: 120px !important;
-      background: transparent !important;
-      border-color: #00d8ff44 !important;
-      margin-right: 20px;
-      color: #fff !important;
-    }
-    .btn {
-      padding: 8px 20px;
-      margin: 10px;
-      position: relative;
+    &::before {
+      position: absolute;
+      display: block;
+      content: '';
+      width: calc(100% - 4px);
+      height: calc(100% - 4px);
+      top: 2px;
+      left: 2px;
       border-radius: 2px;
-      color: #fff;
-      width: fit-content;
-      cursor: pointer;
-
-      &::before {
-        position: absolute;
-        display: block;
-        content: '';
-        width: calc(100% - 4px);
-        height: calc(100% - 4px);
-        top: 2px;
-        left: 2px;
-        border-radius: 2px;
-        z-index: -1;
-      }
+      z-index: -1;
     }
+  }
 
-    .btn1 {
-      border: 1px solid #5cfaff;
+  .btn1 {
+    border: 1px solid #5cfaff;
 
-      &::before {
-        background-image: linear-gradient(#2effee92, #0cb1d592);
-      }
+    &::before {
+      background-image: linear-gradient(#2effee92, #0cb1d592);
+    }
 
-      &:hover {
-        border: 1px solid #5cfaffaa;
+    &:hover {
+      border: 1px solid #5cfaffaa;
 
-        &::before {
-          background-image: linear-gradient(#2effee72, #0cb1d572);
-        }
+      &::before {
+        background-image: linear-gradient(#2effee72, #0cb1d572);
       }
     }
   }
+}
 
-  @keyframes open {
-    0% {
-      height: 0px;
-    }
-    100% {
-      height: fit-content;
-    }
+@keyframes open {
+  0% {
+    height: 0px;
   }
-
-  @keyframes close {
-    0% {
-      height: fit-content;
-    }
-    100% {
-      height: 0px;
-    }
+  100% {
+    height: fit-content;
   }
-  :deep(.zxm-divider-inner-text) {
-    color: #cacaca88 !important;
+}
+
+@keyframes close {
+  0% {
+    height: fit-content;
   }
-  :deep(.zxm-form-item) {
-    margin-bottom: 10px;
+  100% {
+    height: 0px;
   }
+}
+:deep(.zxm-divider-inner-text) {
+  color: #cacaca88 !important;
+}
+:deep(.zxm-form-item) {
+  margin-bottom: 10px;
+}
 </style>

+ 158 - 42
src/views/vent/monitorManager/fanLocalMonitor/fanLocalDual.threejs.base.ts

@@ -1,8 +1,7 @@
 import * as THREE from 'three';
-import { setModalCenter } from '/@/utils/threejs/util';
+// import { setModalCenter } from '/@/utils/threejs/util';
 import Smoke from '../../comment/threejs/Smoke';
 import { CSS3DObject } from 'three/examples/jsm/renderers/CSS3DRenderer';
-// import { setModalCenter } from '/@/utils/threejs/util';
 // import * as dat from 'dat.gui';
 // const gui = new dat.GUI();
 // gui.domElement.style = 'position:absolute;top:100px;left:10px;z-index:99999999999999';
@@ -71,11 +70,10 @@ class ModelContext {
       this.model.setGLTFModel([this.modelName]).then(async (gltf) => {
         this.group = gltf[0];
         if (this.group) {
+          // setModalCenter(this.group);
           this.addLight();
-          this.setModalPosition();
-          this.initModules();
-          setModalCenter(this.group);
-          resolve(null);
+          this.setModelPosition();
+          this.initModules().then(resolve);
         }
       });
     });
@@ -89,10 +87,15 @@ class ModelContext {
   }
 
   // 设置模型位置
-  setModalPosition() {
+  setModelPosition() {
     if (!this.group) return;
-    this.group.scale.set(0.7, 0.7, 0.7);
-    this.group.position.set(0, 6, -50);
+    this.group.scale.set(0.6, 0.6, 0.6);
+
+    // const ff = gui.addFolder(`位置调整`);
+    // ff.add(this.group.position, 'x', -100, 100);
+    // ff.add(this.group.position, 'y', -100, 100);
+    // ff.add(this.group.position, 'z', -100, 100);
+    this.group.position.set(0, 0, -60);
     this.group.rotation.y = Math.PI / 2;
   }
 
@@ -110,7 +113,7 @@ class ModelContext {
   weakElements(eles: unknown[]) {
     eles.forEach((g) => {
       if (g instanceof Smoke) {
-        g.opacityFactor = 0.4;
+        g.oldOpacityFactor = 0.4;
       }
       if (g instanceof CSS3DObject) {
         g.element.style.setProperty('opacity', '0.5');
@@ -120,7 +123,7 @@ class ModelContext {
   strongElements(eles: unknown[]) {
     eles.forEach((g) => {
       if (g instanceof Smoke) {
-        g.opacityFactor = 0.75;
+        g.oldOpacityFactor = 0.75;
       }
       if (g instanceof CSS3DObject) {
         g.element.style.setProperty('opacity', '1');
@@ -146,45 +149,158 @@ class ModelContext {
   /** 核心方法,初始化本模型的各个模块,这些模块可以实现特定场景的展示、控制等功能 */
   async initModules() {
     if (this.elements.length > 0) return;
-    const curve1 = [
-      {
-        path0: new THREE.Vector3(10, 0, -10),
-        path1: new THREE.Vector3(20, 0, -10),
-        isSpread: false,
-        spreadDirection: 0,
-      },
-      {
-        path0: new THREE.Vector3(10, 0, -10),
-        path1: new THREE.Vector3(10, 0, -20),
-        isSpread: true,
-        spreadDirection: 1, // 1是由小变大,-1是由大变小
-      },
-    ];
+    // 右侧风机-主风机进风
+    const curveFan1Right = this.generateSmokePath(
+      [
+        new THREE.Vector3(-85.685, 4.208, 43.895),
+        new THREE.Vector3(-85.685, 2.208, 41.895),
+        new THREE.Vector3(-85.685, 2.188, 35.327),
+        new THREE.Vector3(-85.685, 0.784, 33.086),
+        new THREE.Vector3(-85.685, 0.784, 27.848),
+        new THREE.Vector3(-85.685, 4.724, 21.565),
+        new THREE.Vector3(-85.685, 4.724, -12.993),
+        new THREE.Vector3(-26.191, 4.724, -13.232),
+        new THREE.Vector3(-25.608, 4.724, -47.022),
+        new THREE.Vector3(80.038, 4.724, -47.022),
+      ],
+      true
+    );
+    // 右侧风机-备风机进风
+    const curveFan2Right = this.generateSmokePath(
+      [
+        new THREE.Vector3(-85.685, 1.475, 43.895),
+        new THREE.Vector3(-85.685, -0.525, 41.895),
+        new THREE.Vector3(-85.685, -0.525, 35.327),
+        new THREE.Vector3(-85.685, 0.784, 33.086),
+        new THREE.Vector3(-85.685, 0.784, 27.848),
+        new THREE.Vector3(-85.685, 4.724, 21.565),
+        new THREE.Vector3(-85.685, 4.724, -12.993),
+        new THREE.Vector3(-26.191, 4.724, -13.232),
+        new THREE.Vector3(-25.608, 4.724, -47.022),
+        new THREE.Vector3(80.038, 4.724, -47.022),
+      ],
+      true
+    );
+    // 左侧风机-主风机进风
+    const curveFan1Left = this.generateSmokePath(
+      [
+        new THREE.Vector3(-85.685, 4.188, 4.729),
+        new THREE.Vector3(-85.685, 2.188, 2.729),
+        new THREE.Vector3(-85.685, 2.188, -3.84),
+        new THREE.Vector3(-85.685, 0.784, -6.081),
+        new THREE.Vector3(-85.685, 0.784, -12.912),
+        new THREE.Vector3(80.251, 0.784, -12.912),
+      ],
+      true
+    );
+    // 左侧风机-备风机进风
+    const curveFan2Left = this.generateSmokePath(
+      [
+        new THREE.Vector3(-85.685, 1.475, 4.729),
+        new THREE.Vector3(-85.685, -0.525, 2.729),
+        new THREE.Vector3(-85.685, -0.508, -3.84),
+        new THREE.Vector3(-85.685, 0.784, -6.081),
+        new THREE.Vector3(-85.685, 0.784, -12.912),
+        new THREE.Vector3(80.251, 0.784, -12.912),
+      ],
+      true
+    );
     const group1 = new THREE.Group();
-    const smoke1 = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.75, 0.5, 400);
-    smoke1.setPath(curve1);
-    await smoke1.setPoints();
-    this.elements.push(smoke1);
-
-    const element = document.getElementById('inputBox') as HTMLElement;
-    if (element) {
-      const fanLocalCSS3D = new CSS3DObject(element);
-      fanLocalCSS3D.name = 'text1';
-      fanLocalCSS3D.scale.set(0.04, 0.04, 0.04);
-      fanLocalCSS3D.rotation.y = -Math.PI / 2;
-      fanLocalCSS3D.position.set(-85.68, 5.97, -3.39);
-      group1.add(fanLocalCSS3D);
-      this.elements.push(fanLocalCSS3D);
-    }
+    const smokeFan1Right = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.75, 0.5, 400);
+    smokeFan1Right.setPath(curveFan1Right);
+    this.elements.push(smokeFan1Right);
+    const smokeFan2Right = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.75, 0.5, 400);
+    smokeFan2Right.setPath(curveFan2Right);
+    this.elements.push(smokeFan2Right);
+    const smokeFan1Left = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.75, 0.5, 400);
+    smokeFan1Left.setPath(curveFan1Left);
+    this.elements.push(smokeFan1Left);
+    const smokeFan2Left = new Smoke('/model/img/texture-smoke.png', '#ffffff', 0, 0.75, 0.5, 400);
+    smokeFan2Left.setPath(curveFan2Left);
+    this.elements.push(smokeFan2Left);
+
+    await smokeFan1Right.setPoints();
+    this.group?.add(smokeFan1Right.points);
+    await smokeFan2Right.setPoints();
+    this.group?.add(smokeFan2Right.points);
+    await smokeFan1Left.setPoints();
+    this.group?.add(smokeFan1Left.points);
+    await smokeFan2Left.setPoints();
+    this.group?.add(smokeFan2Left.points);
+    // const element = document.getElementById('inputBox') as HTMLElement;
+    // if (element) {
+    //   const fanLocalCSS3D = new CSS3DObject(element);
+    //   fanLocalCSS3D.name = 'text1';
+    //   fanLocalCSS3D.scale.set(0.04, 0.04, 0.04);
+    //   fanLocalCSS3D.rotation.y = -Math.PI / 2;
+    //   fanLocalCSS3D.position.set(-85.68, 5.97, -3.39);
+    //   group1.add(fanLocalCSS3D);
+    //   this.elements.push(fanLocalCSS3D);
+    // }
 
     this.modules.push({
-      name: 'state1',
+      name: 'fan1RightOpen',
+      context: group1,
+      behavior: () => {
+        this.weakElements(this.elements);
+        this.startAnimation([smokeFan1Right]);
+        this.strongElements([smokeFan1Right]);
+      },
+    });
+    this.modules.push({
+      name: 'fan2RightOpen',
+      context: group1,
+      behavior: () => {
+        this.weakElements(this.elements);
+        this.startAnimation([smokeFan2Right]);
+        this.strongElements([smokeFan2Right]);
+      },
+    });
+    this.modules.push({
+      name: 'fan1LeftOpen',
       context: group1,
       behavior: () => {
         this.weakElements(this.elements);
-        this.strongElements([smoke1]);
+        this.startAnimation([smokeFan1Left]);
+        this.strongElements([smokeFan1Left]);
       },
     });
+    this.modules.push({
+      name: 'fan2LeftOpen',
+      context: group1,
+      behavior: () => {
+        this.weakElements(this.elements);
+        this.startAnimation([smokeFan2Left]);
+        this.strongElements([smokeFan2Left]);
+      },
+    });
+  }
+
+  /** 生成适用于 Smoke 的曲线数据,输入途径点,输出路径,如果是进风类型,首个线段将有扩散效果,出风则是末尾线段有扩散效果 */
+  generateSmokePath(points: THREE.Vector3[], airIn?: boolean, airOut?: boolean) {
+    const result: any[] = [];
+    for (let index = 1; index < points.length; index++) {
+      const path0 = points[index - 1];
+      const path1 = points[index];
+      const path = {
+        path0,
+        path1,
+        isSpread: false,
+        spreadDirection: 0,
+      };
+      if (airIn) {
+        // 首个线段需要扩散,由大变小
+        path.isSpread = index === 1;
+        path.spreadDirection = -1;
+      }
+      if (airOut) {
+        // 首个线段需要扩散,由小变大
+        path.isSpread = index === points.length - 1;
+        path.spreadDirection = 1;
+      }
+      result.push(path);
+    }
+    return result;
   }
 }
 export default ModelContext;

+ 4 - 4
src/views/vent/monitorManager/fanLocalMonitor/index.vue

@@ -172,7 +172,7 @@
       </div>
       <div class="top-right row">
         <template v-for="(item, index) in modalTypeArr.rightBtnArr" :key="index">
-          <div
+          <!-- <div
             v-if="
               (item.permission === 'fanLocal:gasAlarmSet' ||
                 item.permission === 'fanLocal:kkjc' ||
@@ -183,8 +183,8 @@
             :class="{ 'button-box': btnClick, 'button-disable': !btnClick }"
             @click="showModal(item)"
             >{{ item.value }}</div
-          >
-          <!-- <div
+          > -->
+          <div
             v-if="
               item.permission === 'fanLocal:gasAlarmSet' ||
               item.permission === 'fanLocal:kkjc' ||
@@ -194,7 +194,7 @@
             :class="{ 'button-box': btnClick, 'button-disable': !btnClick }"
             @click="showModal(item)"
             >{{ item.value }}</div
-          > -->
+          >
         </template>
       </div>
     </div>