gasSupplyAir.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. <template>
  2. <a-modal
  3. v-model:visible="visible"
  4. width="1200px"
  5. height="500px"
  6. class="supplyAir-modal"
  7. title="风窗自主调控风量监测"
  8. @ok="handleOk"
  9. @cancel="handleOk"
  10. >
  11. <div class="modal-box">
  12. <div class="left-box"> </div>
  13. <div class="right-box">
  14. <BarAndLine
  15. class="echarts-line"
  16. xAxisPropType="readTime"
  17. height="400px"
  18. :dataSource="echartsData"
  19. :chartsColumns="chartsColumnList"
  20. :option="echatsOption"
  21. />
  22. </div>
  23. </div>
  24. <div class="setting-box">
  25. <div class="right-inputs">
  26. <div class="vent-flex-row" v-if="!isMock">
  27. <template v-if="monitorData['nwindownum'] == 1">
  28. <div class="input-box"
  29. >风窗实时开度(°):<span>{{ monitorData['forntArea'] }}</span></div
  30. >
  31. </template>
  32. <template v-else>
  33. <div class="input-box"
  34. >前窗实时开度(°):<span>{{ monitorData['forntArea'] }}</span></div
  35. >
  36. <div class="input-box"
  37. >后窗实时开度(°):<span>{{ monitorData['rearArea'] }}</span></div
  38. >
  39. </template>
  40. <div class="input-box"
  41. >风窗实时瓦斯含量(%):<span>{{ monitorData['gas'] }}</span></div
  42. >
  43. </div>
  44. <div class="vent-flex-row" v-else>
  45. <template v-if="monitorData['nwindownum'] == 1">
  46. <div class="input-box"
  47. >风窗实时开度(°):<span>{{ frontWindowAngle }}</span></div
  48. >
  49. </template>
  50. <template v-else>
  51. <div class="input-box"
  52. >前窗实时开度(°):<span>{{ frontWindowAngle }}</span></div
  53. >
  54. <div class="input-box"
  55. >后窗实时开度(°):<span>{{ rearWindowAngle }}</span></div
  56. >
  57. </template>
  58. <div class="input-box"
  59. >风窗实时瓦斯含量(%):<span>{{ gasTemp ? gasTemp.toFixed(2) : gas?.toFixed(2) }}</span></div
  60. >
  61. <div class="input-box"
  62. >风窗实时风量(m³/min):<span>{{ fWindowM3Temp ? fWindowM3Temp.toFixed(2) : fWindowM3?.toFixed(2) }}</span></div
  63. >
  64. </div>
  65. </div>
  66. </div>
  67. </a-modal>
  68. </template>
  69. <script lang="ts" setup>
  70. import { ref, nextTick, onMounted, watch, computed, reactive } from 'vue';
  71. import BarAndLine from '/@/components/chart/BarAndLine.vue';
  72. import { number } from 'vue-types';
  73. const props = defineProps({
  74. modalIsShow: {
  75. type: Boolean,
  76. default: false,
  77. },
  78. modalType: {
  79. type: String,
  80. },
  81. data: {
  82. type: Object,
  83. default: () => {},
  84. },
  85. gasVal: {
  86. type: Number,
  87. },
  88. isMock: {
  89. type: Boolean,
  90. },
  91. });
  92. const emit = defineEmits(['handleOk', 'handleCancel']);
  93. const visible = computed(() => props.modalIsShow);
  94. const changeEchart = computed(() => props.modalType);
  95. const echartsData = ref<
  96. {
  97. gasVal: string | number;
  98. gas: string | number;
  99. readTime: string;
  100. fWindowM3: string | number;
  101. forntArea: string | number;
  102. rearArea?: string | number;
  103. }[]
  104. >([]);
  105. const monitorData = ref({});
  106. const echatsOption = {
  107. legend: {
  108. top: 10,
  109. },
  110. grid: {
  111. top: '80',
  112. left: '30',
  113. right: '35',
  114. bottom: '10',
  115. containLabel: true,
  116. },
  117. toolbox: {
  118. feature: {
  119. saveAsImage: {
  120. show: false,
  121. },
  122. },
  123. },
  124. xAxis: {
  125. type: 'category',
  126. axisLabel: {
  127. margin: 10,
  128. color: '#f1f1f199',
  129. },
  130. name: '',
  131. },
  132. yAxis: {
  133. axisLabel: {
  134. color: '#0071A5',
  135. },
  136. },
  137. };
  138. const chartsColumnListTemp = [
  139. {
  140. legend: '瓦斯超限值',
  141. seriesName: '(%)',
  142. ymax: 1100,
  143. yname: '%',
  144. linetype: 'line',
  145. yaxispos: 'left',
  146. color: '#00FFA8',
  147. sort: 1,
  148. xRotate: 0,
  149. dataIndex: 'gasVal',
  150. },
  151. {
  152. legend: '瓦斯实时值',
  153. seriesName: '(%)',
  154. ymax: 1100,
  155. yname: '%',
  156. linetype: 'line',
  157. yaxispos: 'left',
  158. color: '#00FFA8',
  159. sort: 1,
  160. xRotate: 0,
  161. dataIndex: 'gas',
  162. },
  163. {
  164. legend: '风窗开度',
  165. seriesName: '(°)',
  166. ymax: 1100,
  167. yname: '°',
  168. linetype: 'line',
  169. yaxispos: 'right',
  170. color: '#FDB146',
  171. sort: 2,
  172. xRotate: 0,
  173. dataIndex: 'forntArea',
  174. },
  175. {
  176. legend: '前窗开度',
  177. seriesName: '(°)',
  178. ymax: 1100,
  179. yname: '°',
  180. linetype: 'line',
  181. yaxispos: 'right',
  182. color: '#FDB146',
  183. sort: 2,
  184. xRotate: 0,
  185. dataIndex: 'forntArea',
  186. },
  187. {
  188. legend: '后窗开度',
  189. seriesName: '(°)',
  190. ymax: 1100,
  191. yname: '°',
  192. linetype: 'line',
  193. yaxispos: 'right',
  194. color: '#FDB146',
  195. sort: 2,
  196. xRotate: 0,
  197. dataIndex: 'rearArea',
  198. },
  199. ];
  200. const chartsColumnList = ref(chartsColumnListTemp);
  201. let frontWindowAngle = ref<undefined | number>(undefined);
  202. let rearWindowAngle = ref<undefined | number>(undefined);
  203. let fWindowM3 = ref<undefined | number>(undefined);
  204. let gas = ref<undefined | number>(undefined);
  205. let fWindowM3Temp = ref<undefined | number>(undefined);
  206. let gasTemp = ref<undefined | number>(undefined);
  207. let index = 1;
  208. watch(
  209. () => props.data,
  210. (newVal) => {
  211. if (!visible.value) return;
  212. if (newVal['nwindownum'] == 1) {
  213. // 单道风窗
  214. if (chartsColumnList.value.length != 3) chartsColumnList.value = [chartsColumnListTemp[0], chartsColumnListTemp[1], chartsColumnListTemp[2]];
  215. } else {
  216. // 多道风窗
  217. if (chartsColumnList.value.length != 4)
  218. chartsColumnList.value = [chartsColumnListTemp[0], chartsColumnListTemp[1], chartsColumnListTemp[3], chartsColumnListTemp[4]];
  219. }
  220. monitorData.value = newVal;
  221. if (echartsData.value.length > 20) {
  222. echartsData.value.shift();
  223. }
  224. if (props.isMock) {
  225. // 这里开启风窗开度模拟
  226. if (newVal['nwindownum'] == 1 && frontWindowAngle.value === undefined) {
  227. frontWindowAngle.value = newVal['forntArea'];
  228. } else if (newVal['nwindownum'] == 2) {
  229. if (frontWindowAngle.value === undefined) frontWindowAngle.value = (newVal['forntArea'] / newVal['maxarea']) * 90;
  230. if (rearWindowAngle.value === undefined) rearWindowAngle.value = (newVal['rearArea'] / newVal['maxarea']) * 90;
  231. }
  232. if (fWindowM3.value === undefined) fWindowM3.value = newVal['fWindowM3'];
  233. // if (gas == undefined) gas = Number(newVal['gas']) > Number(props.gasVal) ? newVal['gas'] : props.gasVal + 0.2;
  234. if (gas.value == undefined) gas.value = props.gasVal + 0.2;
  235. // 开启模拟
  236. if (gas.value > props.gasVal - 0.3) {
  237. mock(newVal['maxarea']);
  238. echartsData.value = [
  239. ...echartsData.value,
  240. {
  241. gasVal: props.gasVal,
  242. gas: gas.value as number,
  243. readTime: newVal['readTime'].substring(11),
  244. forntArea: frontWindowAngle.value as number,
  245. rearArea: rearWindowAngle.value as number,
  246. fWindowM3: fWindowM3.value as number,
  247. },
  248. ];
  249. } else {
  250. const random = Math.random() - 0.5;
  251. fWindowM3Temp.value = fWindowM3.value + random * 50;
  252. gasTemp.value = gas.value + random * 0.01;
  253. echartsData.value = [
  254. ...echartsData.value,
  255. {
  256. gasVal: props.gasVal,
  257. gas: gasTemp.value,
  258. readTime: newVal['readTime'].substring(11),
  259. forntArea: frontWindowAngle.value as number,
  260. rearArea: rearWindowAngle.value as number,
  261. fWindowM3: fWindowM3Temp.value,
  262. },
  263. ];
  264. }
  265. } else {
  266. echartsData.value = [
  267. ...echartsData.value,
  268. {
  269. gasVal: props.gasVal,
  270. gas: newVal['gas'],
  271. readTime: newVal['readTime'].substring(11),
  272. forntArea: (newVal['forntArea'] / newVal['maxarea']) * 90,
  273. rearArea: (newVal['rearArea'] / newVal['maxarea']) * 90,
  274. fWindowM3: newVal['fWindowM3'],
  275. },
  276. ];
  277. }
  278. }
  279. );
  280. function mock(maxArea?: number) {
  281. debugger;
  282. if (maxArea) {
  283. // 每调用一次数据就增减一些
  284. // frontWindowAngle 增加一些
  285. if (frontWindowAngle.value !== undefined && frontWindowAngle.value <= 90) {
  286. frontWindowAngle.value = Math.min(frontWindowAngle.value + 3, 90);
  287. }
  288. if (rearWindowAngle.value !== undefined && rearWindowAngle.value <= 90) {
  289. rearWindowAngle.value = Math.min(rearWindowAngle.value + 3, 90);
  290. }
  291. if (fWindowM3.value !== undefined) {
  292. fWindowM3.value += 40 * index;
  293. }
  294. if (gas.value !== undefined) {
  295. gas.value -= 0.001 * index;
  296. }
  297. ++index;
  298. }
  299. }
  300. function handleOk() {
  301. emit('handleCancel');
  302. echartsData.value = [];
  303. frontWindowAngle.value = undefined;
  304. rearWindowAngle.value = undefined;
  305. fWindowM3.value = undefined;
  306. gas.value = undefined;
  307. fWindowM3Temp.value = undefined;
  308. gasTemp.value = undefined;
  309. }
  310. onMounted(() => {});
  311. </script>
  312. <style scoped lang="less">
  313. .modal-box {
  314. display: flex;
  315. flex-direction: row;
  316. background-color: #ffffff05;
  317. padding: 10px 8px 0 8px;
  318. border: 1px solid #00d8ff22;
  319. position: relative;
  320. // min-height: 600px;
  321. .left-box {
  322. flex: 1; /* 占据 3/4 的空间 */
  323. background-image: url(../../../../../assets/images/supplyAir.svg);
  324. background-repeat: no-repeat;
  325. background-size: contain; /* 确保背景图片完整显示 */
  326. background-position: center; /* 确保背景图片居中 */
  327. }
  328. .right-box {
  329. flex: 1; /* 占据 3/4 的空间 */
  330. height: 400px;
  331. width: 100%;
  332. }
  333. }
  334. .setting-box {
  335. width: 1570px;
  336. height: 70px;
  337. margin: 10px 0;
  338. background-color: #ffffff05;
  339. border: 1px solid #00d8ff22;
  340. display: flex;
  341. align-items: center;
  342. justify-content: space-between;
  343. .right-inputs {
  344. width: 100%;
  345. display: flex;
  346. height: 40px;
  347. margin: 0 10px;
  348. justify-content: space-between;
  349. }
  350. .left-buttons {
  351. display: flex;
  352. height: 40px;
  353. .btn {
  354. margin: 0 10px;
  355. }
  356. }
  357. .border-clip {
  358. width: 1px;
  359. height: 25px;
  360. border-right: 1px solid #8b8b8b77;
  361. }
  362. .input-box {
  363. width: 300px;
  364. span {
  365. color: aqua;
  366. padding: 0 20px;
  367. }
  368. }
  369. }
  370. @keyframes open {
  371. 0% {
  372. height: 0px;
  373. }
  374. 100% {
  375. height: fit-content;
  376. }
  377. }
  378. @keyframes close {
  379. 0% {
  380. height: fit-content;
  381. }
  382. 100% {
  383. height: 0px;
  384. }
  385. }
  386. :deep(.zxm-divider-inner-text) {
  387. color: #cacaca88 !important;
  388. }
  389. :deep(.zxm-form-item) {
  390. margin-bottom: 10px;
  391. }
  392. </style>