index.vue 38 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052
  1. <template>
  2. <div class="bg" style="width: 100%; height: 100%; display: flex; justify-content: center; align-items: center; overflow: hidden">
  3. <a-spin :spinning="loading" >
  4. </a-spin>
  5. <div id="fanLocal3D" style="width: 100%; height: 100%; position: absolute; overflow: hidden"> </div>
  6. <!-- <div id="fanLocal3DCSS" class="threejs-Object-CSS" style="width: 100%; height: 100%; position: absolute; overflow: hidden; pointer-events: none;">
  7. <div style="z-index: -1; position: relative">
  8. <div class="elementTag" id="inputBox">
  9. <div class="elementContent">
  10. <p>风筒入口风速:{{ selectData.windSpeed1 ? selectData.windSpeed1 : '-' }}</p>
  11. <p>风筒入口瓦斯浓度 {{ selectData.gas1 ? selectData.gas1 : '-' }}</p>
  12. </div>
  13. </div>
  14. <div class="elementTag" id="outBox">
  15. <div class="elementContent elementContent-r">
  16. <p>迎头供风量:{{ selectData.windQuantity1 ? selectData.windQuantity2 : '-' }}</p>
  17. <p>迎头瓦斯浓度:{{ selectData.gas1 ? selectData.gas1 : '-' }}</p>
  18. </div>
  19. </div>
  20. <div class="elementTag" id="returnBox">
  21. <div class="elementContent">
  22. <p>回风流瓦斯浓度:{{ selectData.gas3 ? selectData.gas3 : '-' }}</p>
  23. </div>
  24. </div>
  25. <div class="elementTag" id="gateBox">
  26. <div class="elementContent">
  27. <p>风门状态:关</p>
  28. <p>风门过风面积:{{ selectData.gas1 ? selectData.gas1 : '-' }}</p>
  29. </div>
  30. </div>
  31. <div class="elementTag" id="windownBox">
  32. <div class="elementContent">
  33. <p>风窗全风压回风流:0</p>
  34. </div>
  35. </div>
  36. </div>
  37. </div> -->
  38. </div>
  39. <div class="scene-box">
  40. <div class="top-box" v-if="!loading">
  41. <div class="top-center row">
  42. <div class="vent-flex-row" id="fanLocalSelectDom" v-if="getDictItemsByCode('fanlocaltype')">
  43. <span style="color: #00f5fe; margin-left: 5px;">风机类型:</span>
  44. <JDictSelectTag v-model:value="devicekide" dictCode="fanlocaltype" :showChooseOption="false" :getPopupContainer="getPopupContainer" @change="changeDeviceKind" />
  45. </div>
  46. <div class="button-box" v-for="(item, index) in modalTypeArr.leftBtnArr" :key="index" @click="showModal(item)">{{ item.value }}</div>
  47. </div>
  48. <div class="top-right row">
  49. <div class="button-box" v-for="(item, index) in modalTypeArr.rightBtnArr" :key="index" @click="showModal(item)">{{ item.value }}</div>
  50. </div>
  51. </div>
  52. <div class="title-text">
  53. {{ selectData.stationname }}
  54. </div>
  55. <div class="data-show-box" v-if="!loading">
  56. <div class="data-item">
  57. <div class="item-header">环境监测</div>
  58. <div class="item-container">
  59. <div class="tab">
  60. <div class="tab-item" :class="{ 'tab-item-active-r': dataMonitorRowIndex == 0 }" @click="selectDevice('dataMonitorRowIndex', 0)"
  61. >1#风机</div
  62. >
  63. <div class="tab-item" :class="{ 'tab-item-active-r': dataMonitorRowIndex == 1 }" @click="selectDevice('dataMonitorRowIndex', 1)"
  64. >2#风机</div
  65. >
  66. </div>
  67. <div class="container-group container-group-l">
  68. <template v-if="!loading">
  69. <div class="container-item" v-for="(data, index) in leftColumns" :key="index">
  70. <div class="item-icon">
  71. <SvgIcon class="icon-style" size="18" name="temperature" />
  72. </div>
  73. <div class="item-name">{{ data.title }}</div>
  74. <div v-if="data.dataIndex.startsWith('Fan')">
  75. <div class="item-value" v-if="dataMonitorRowIndex == 0">{{
  76. selectData[data.dataIndex.replace('Fan', 'Fan1')] ? selectData[data.dataIndex.replace('Fan', 'Fan1')] : '-'
  77. }}</div>
  78. <div class="item-value" v-if="dataMonitorRowIndex == 1">{{
  79. selectData[data.dataIndex.replace('Fan', 'Fan2')] ? selectData[data.dataIndex.replace('Fan', 'Fan2')] : '-'
  80. }}</div>
  81. </div>
  82. <div v-else>
  83. <div class="item-value">{{ selectData[data.dataIndex] ? selectData[data.dataIndex] : '-' }}</div>
  84. </div>
  85. </div>
  86. </template>
  87. </div>
  88. </div>
  89. </div>
  90. <div class="data-item">
  91. <div class="item-header">设备状态</div>
  92. <div class="item-container">
  93. <div class="tab">
  94. <div class="tab-item" :class="{ 'tab-item-active-r': warningMonitorRowIndex === 0 }" @click="selectDevice('warningMonitorRowIndex', 0)"
  95. >1#风机</div
  96. >
  97. <div class="tab-item" :class="{ 'tab-item-active-r': warningMonitorRowIndex === 1 }" @click="selectDevice('warningMonitorRowIndex', 1)"
  98. >2#风机</div
  99. >
  100. </div>
  101. <div class="container-group">
  102. <div class="warning-header">
  103. <div class="header-item">
  104. <div class="header-title">报警总数</div>
  105. <div class="header-value">0</div>
  106. </div>
  107. <div class="header-item">
  108. <div class="header-title"> 未处理数</div>
  109. <div class="header-value">0</div>
  110. </div>
  111. </div>
  112. <div class="warning-group">
  113. <template v-if="selectData.deviceType">
  114. <div class="warning-item" v-for="(state, index) in rightColumns" :key="index">
  115. <div class="item-name"><div class="icon"></div> {{ state.title }}</div>
  116. <div v-if="state.dataIndex.startsWith('Fan')">
  117. <div class="signal-item" v-if="warningMonitorRowIndex == 0">
  118. <div
  119. class="signal-round"
  120. :class="{
  121. 'signal-round-run': selectData[state.dataIndex.replace('Fan', 'Fan1')],
  122. 'signal-round-warning':
  123. selectData[state.dataIndex.replace('Fan', 'Fan1')] !== undefined && !selectData[state.dataIndex.replace('Fan', 'Fan1')],
  124. 'signal-round-gry': selectData[state.dataIndex.replace('Fan', 'Fan1')] === undefined,
  125. }"
  126. ></div>
  127. <div class="vent-margin-l-8">{{
  128. selectData[state.dataIndex.replace('Fan', 'Fan1')] === undefined
  129. ? '无状态'
  130. : selectData[state.dataIndex.replace('Fan', 'Fan1')]
  131. ? '正常'
  132. : '异常'
  133. }}</div>
  134. </div>
  135. <div class="signal-item" v-if="warningMonitorRowIndex == 1">
  136. <div
  137. class="signal-round"
  138. :class="{
  139. 'signal-round-run': selectData[state.dataIndex.replace('Fan', 'Fan2')],
  140. 'signal-round-warning':
  141. selectData[state.dataIndex.replace('Fan', 'Fan2')] !== undefined && !selectData[state.dataIndex.replace('Fan', 'Fan2')],
  142. 'signal-round-gry': selectData[state.dataIndex.replace('Fan', 'Fan2')] === undefined,
  143. }"
  144. ></div>
  145. <div class="vent-margin-l-8">{{
  146. selectData[state.dataIndex.replace('Fan', 'Fan2')] === undefined
  147. ? '无状态'
  148. : selectData[state.dataIndex.replace('Fan', 'Fan2')]
  149. ? '正常'
  150. : '异常'
  151. }}</div>
  152. </div>
  153. </div>
  154. <div v-else>
  155. <div class="signal-item">
  156. <div
  157. class="signal-round vent-margin-l-8"
  158. :class="{
  159. 'signal-round-run': selectData[state.dataIndex],
  160. 'signal-round-warning': selectData[state.dataIndex] !== undefined && !selectData[state.dataIndex],
  161. 'signal-round-gry': selectData[state.dataIndex] === undefined,
  162. }"
  163. ></div>
  164. <div class="vent-margin-l-8">{{ selectData[state.dataIndex] === undefined ? '无状态' : selectData[state.dataIndex] ? '正常' : '异常' }}</div>
  165. </div>
  166. </div>
  167. </div>
  168. </template>
  169. </div>
  170. </div>
  171. </div>
  172. </div>
  173. </div>
  174. <div class="bottom-tabs-box" @mousedown="setDivHeight($event, 170, scroll, 180)">
  175. <dv-border-box8 :dur="5" class="dv_border_8" :style="`bottom: 20px; padding: 5px; height: ${scroll.y + 140}px`" >
  176. <!-- <div class="enter-detail" @click="goDetail()">
  177. <send-outlined class=""/>风机运行详情
  178. </div> -->
  179. <a-tabs class="tabs-box" v-model:activeKey="activeKey" @change="tabChange">
  180. <a-tab-pane key="1" tab="实时监测">
  181. <GroupMonitorTable v-if="activeKey === '1'" ref="MonitorDataTable" :dataSource="dataSource" :columnsType="`${selectData.deviceType}_monitor`" @selectRow="getSelectRow" :scroll="scroll"/>
  182. </a-tab-pane>
  183. <a-tab-pane key="2" tab="实时曲线图" force-render>
  184. <div class="tab-item" v-if="activeKey === '2'">
  185. <div class="vent-flex-row-between" style="height: 100%">
  186. <BarSingle
  187. :xAxisData="[
  188. { key: 'T1', valueKey: 'gas1' },
  189. { key: 'T2', valueKey: 'gas2' },
  190. { key: 'T3', valueKey: 'gas3' },
  191. { key: 'T4', valueKey: 'gas4' },
  192. ]"
  193. :dataSource="dataSource[selectRowIndex]"
  194. height="100%"
  195. :chartsColumns="chartsColumns"
  196. style="flex: 3"
  197. />
  198. <BarSingle
  199. :xAxisData="[
  200. { key: 'F1', valueKey: 'windQuantity1' },
  201. { key: 'F2', valueKey: 'windQuantity2' },
  202. ]"
  203. :dataSource="dataSource[selectRowIndex]"
  204. height="100%"
  205. :chartsColumns="chartsColumns1"
  206. style="flex: 2"
  207. />
  208. </div>
  209. </div>
  210. </a-tab-pane>
  211. <a-tab-pane key="3" tab="历史数据">
  212. <div class="tab-item" v-if="activeKey === '3'">
  213. <HistoryTable :columns-type="`${selectData.deviceType}`" :device-type="`${devicekide}`" :device-list-api="baseList" designScope="fanlocal-history" :scroll="scroll" />
  214. </div>
  215. </a-tab-pane>
  216. <a-tab-pane key="4" tab="报警历史">
  217. <div class="tab-item" v-if="activeKey === '4'">
  218. <AlarmHistoryTable columns-type="alarm" :device-type="`${devicekide}`" :device-list-api="baseList" designScope="alarm-history" :scroll="scroll" />
  219. </div>
  220. </a-tab-pane>
  221. <a-tab-pane key="5" tab="操作历史">
  222. <div class="tab-item" v-if="activeKey === '5'">
  223. <HandlerHistoryTable columns-type="operator_history" :device-type="`${devicekide}`" :device-list-api="baseList" designScope="alarm-history" :scroll="scroll" />
  224. </div>
  225. </a-tab-pane>
  226. </a-tabs>
  227. </dv-border-box8>
  228. </div>
  229. </div>
  230. <div style="z-index: -1; position: absolute; top: 50px; right: 10px; width: 300px; height: 280px; margin: auto" class="player1">
  231. <LivePlayer id="jb-player1" ref="player1" :videoUrl="flvURL1()" muted live loading controls />
  232. </div>
  233. <a-modal v-model:visible="modalIsShow" :title="modalTitle" @ok="handleOk">
  234. <div class="modal-container">
  235. <div class="vent-flex-row">
  236. <ExclamationCircleFilled style="color: #ffb700; font-size: 30px" />
  237. <div class="warning-text">您是否要进行{{ modalTitle }}操作?</div>
  238. </div>
  239. <div class="" v-if="modalType == 'startSmoke'">
  240. <div class="startSmoke-select">
  241. <div class="label">1#风机:</div>
  242. <a-radio-group v-model:value="mainWindIsShow1" @change="changeMotor" name="localWind1">
  243. <a-radio value="open">开启</a-radio>
  244. <a-radio value="stop">停止</a-radio>
  245. </a-radio-group>
  246. </div>
  247. <div class="startSmoke-select">
  248. <div class="label">2#风机:</div>
  249. <a-radio-group v-model:value="mainWindIsShow2" @change="changeMotor" name="localWind2">
  250. <a-radio value="open">开启</a-radio>
  251. <a-radio value="stop">停止</a-radio>
  252. </a-radio-group>
  253. </div>
  254. </div>
  255. <!-- 调频 -->
  256. <!-- <div class="vent-flex-row input-box" v-if="modalType == 'Fan1Frequency'">
  257. <div class="label">主风机运行频率(Hz):</div>
  258. <a-input-number size="small" v-model:value="fan1FrequencyVal" :min="30" :max="50" :step="0.1" />
  259. </div>
  260. <div class="vent-flex-row input-box" v-if="modalType == 'Fan2Frequency'">
  261. <div class="label">备风机运行频率(Hz):</div>
  262. <a-input-number size="small" v-model:value="fan2FrequencyVal" :min="30" :max="50" :step="0.1" />
  263. </div> -->
  264. <!-- <div class="vent-flex-row input-box" v-if="modalType == 'needAir'">
  265. <div class="label">需风量(单位):</div>
  266. <a-input-number size="small" v-model:value="frequencyVal" :min="30" :max="50" :step="0.1" />
  267. </div>
  268. <div class="vent-flex-row input-box" v-if="modalType == 'disAirAlarm'">
  269. <div class="label">漏风率(%):</div>
  270. <a-input-number size="small" v-model:value="frequencyVal" :min="0" :max="50" :step="0.1" />
  271. </div>
  272. <div class="vent-flex-row input-box" v-if="modalType == 'diameter'">
  273. <div class="label">风筒直径(m):</div>
  274. <a-input-number size="small" v-model:value="frequencyVal" :min="0" :max="50" :step="0.1" />
  275. </div>
  276. <div class="vent-flex-row input-box" v-if="modalType == 'len'">
  277. <div class="label">风筒长度(m):</div>
  278. <a-input-number size="small" v-model:value="frequencyVal" :min="0" :max="50" :step="0.1" />
  279. </div>
  280. <div class="vent-flex-row input-box" v-if="modalType == 'windPowerLimit'">
  281. <div class="label">风电闭锁限值(m³/min):</div>
  282. <a-input-number size="small" v-model:value="frequencyVal" :min="0" :max="50" :step="0.1" />
  283. </div>
  284. <div class="vent-flex-row input-box" v-if="modalType == 'gasPowerLimit'">
  285. <div class="label">瓦斯电闭锁限值(m³/min):</div>
  286. <a-input-number size="small" v-model:value="frequencyVal" :min="0" :max="50" :step="0.1" />
  287. </div>
  288. <div v-if="modalType == 'gasAlarm'">
  289. <div class="vent-flex-row input-box">
  290. <div class="label">传感器名称:</div>
  291. <a-select placeholder="传感器" @change="handleChangeSensor" :options="sensorList" v-model:value="modalSensor" />
  292. </div>
  293. <div class="vent-flex-row input-box">
  294. <div class="label">传感器值:</div>
  295. <a-input-number size="small" v-model:value="frequencyVal" :min="0" :max="50" :step="0.1" />
  296. </div>
  297. </div>
  298. <div v-if="modalType == 'airVolumeAlarm'">
  299. <div class="vent-flex-row input-box">
  300. <div class="label">风量上限(m³/min):</div>
  301. <a-input-number size="small" v-model:value="modalTypeArr.rightBtnArr[3].min" :min="0" :max="50" :step="0.1" />
  302. </div>
  303. <div class="vent-flex-row input-box">
  304. <div class="label">风量下限(m³/min):</div>
  305. <a-input-number size="small" v-model:value="modalTypeArr.rightBtnArr[3].max" :min="0" :max="50" :step="0.1" />
  306. </div>
  307. </div> -->
  308. <!-- 启动或停止 -->
  309. <div class="" v-if="modalType == 'startSmoke'"> </div>
  310. <div v-if="!globalConfig?.simulatedPassword" class="vent-flex-row input-box">
  311. <div class="label">操作密码:</div>
  312. <a-input size="small" type="password" v-model:value="passWord" />
  313. </div>
  314. </div>
  315. </a-modal>
  316. <!-- <DetailModal @register="registerModal"/> -->
  317. </template>
  318. <script setup lang="ts">
  319. import { ExclamationCircleFilled } from '@ant-design/icons-vue';
  320. import { onBeforeMount, ref, watch, onMounted, nextTick, toRaw, reactive, onUnmounted, inject } from 'vue';
  321. import BarSingle from '../../../../components/chart/BarSingle.vue';
  322. import GroupMonitorTable from '../comment/GroupMonitorTable.vue';
  323. import HistoryTable from '../comment/HistoryTable.vue';
  324. import AlarmHistoryTable from '../comment/AlarmHistoryTable.vue';
  325. import HandlerHistoryTable from '../comment/HandlerHistoryTable.vue';
  326. // import DetailModal from './components/DetailModal.vue';
  327. import { mountedThree, setModelType, destroy, addCssText, addText, playSmoke } from './fanLocal.three';
  328. import lodash from 'lodash';
  329. import { getTableList, list } from '/@/views/vent/monitorManager/fanLocalMonitor/fanLocal.api';
  330. import { list as baseList } from '../../deviceManager/fanTabel/fan.api';
  331. import { chartsColumns, chartsColumns1 } from './fanLocal.data';
  332. import { deviceControlApi } from '/@/api/vent/index';
  333. import LivePlayer from '@liveqing/liveplayer-v3';
  334. import { setDivHeight } from '/@/utils/event';
  335. import { BorderBox8 as DvBorderBox8 } from '@kjgl77/datav-vue3';
  336. import { getTableHeaderColumns } from '/@/hooks/web/useWebColumns';
  337. import { SvgIcon } from '/@/components/Icon';
  338. import { useRouter } from 'vue-router';
  339. import { useModal } from '/@/components/Modal';
  340. import type { BasicColumn } from '/@/components/Table/src/types/table';
  341. import JDictSelectTag from '/@/components/Form/src/jeecg/components/JDictSelectTag.vue';
  342. import { getPopupContainer } from '/@/utils';
  343. import { getDictItemsByCode } from '/@/utils/dict';
  344. import { message } from 'ant-design-vue';
  345. const globalConfig = inject('globalConfig');
  346. const [registerModal, { openModal, closeModal }] = useModal();
  347. const { currentRoute } = useRouter();
  348. const modalTypeArr = reactive({
  349. leftBtnArr: [
  350. {
  351. key: 'startSmoke',
  352. value: '启动风机',
  353. },
  354. {
  355. key: 'changeSmoke',
  356. value: '一键倒机',
  357. },
  358. // {
  359. // key: 'Fan1Frequency',
  360. // value: '主机调频',
  361. // },
  362. // {
  363. // key: 'Fan2Frequency',
  364. // value: '备机调频',
  365. // },
  366. // {
  367. // key: 'windPower',
  368. // value: '风电闭锁',
  369. // },
  370. // {
  371. // key: 'gasPower',
  372. // value: '瓦斯电闭锁',
  373. // },
  374. // {
  375. // key: 'needAir',
  376. // value: '需风量',
  377. // },
  378. // {
  379. // key: 'diameter',
  380. // value: '风筒直径',
  381. // },
  382. // {
  383. // key: 'len',
  384. // value: '风筒长度',
  385. // },
  386. ],
  387. rightBtnArr: [
  388. // {
  389. // key: 'fanlocal',
  390. // value: '变频风机',
  391. // },
  392. // {
  393. // key: 'fanlocaldp',
  394. // value: '定频风机',
  395. // },
  396. // {
  397. // key: 'frequency',
  398. // value: '调频',
  399. // },
  400. // {
  401. // key: 'windPowerLimit',
  402. // value: '风电闭锁限值',
  403. // },
  404. // {
  405. // key: 'gasPowerLimit',
  406. // value: '瓦斯电闭锁限值',
  407. // },
  408. // {
  409. // key: 'airVolumeAlarm',
  410. // value: '风量报警',
  411. // min: 0,
  412. // max: 100,
  413. // },
  414. // {
  415. // key: 'disAirAlarm',
  416. // value: '漏风率报警',
  417. // },
  418. // {
  419. // key: 'gasAlarm',
  420. // value: '瓦斯报警',
  421. // },
  422. ],
  423. });
  424. const sensorList = ref<any[]>([
  425. {
  426. value: '1',
  427. label: 'T1',
  428. },
  429. {
  430. value: '2',
  431. label: 'T2',
  432. },
  433. {
  434. value: '3',
  435. label: 'T3',
  436. },
  437. ]);
  438. const scroll = reactive({
  439. y: 180
  440. })
  441. const MonitorDataTable = ref()
  442. const modalSensor = ref(null);
  443. const loading = ref(false);
  444. const activeKey = ref('1');
  445. const player1 = ref();
  446. const modalIsShow = ref<boolean>(false); // 是否显示模态框
  447. const modalTitle = ref(''); // 模态框标题显示内容,根据设备操作类型决定
  448. const fan1FrequencyVal = ref(40); //主机频率
  449. const fan2FrequencyVal = ref(40); //备机频率
  450. const mainWindIsShow1 = ref('open'); // 1#风机默认启动leftColumns
  451. const mainWindIsShow2 = ref('stop'); // 2#风机默认不启动
  452. const passWord = ref('');
  453. // 默认初始是第一行
  454. const selectRowIndex = ref(-1);
  455. const dataMonitorRowIndex = ref(0);
  456. // 默认数据右边监测的是1#风机
  457. const warningMonitorRowIndex = ref(0);
  458. // 设备数据
  459. const controlType = ref(1);
  460. const modalType = ref('fm');
  461. // 监测数据
  462. const initData = {
  463. deviceID: '',
  464. deviceType: '',
  465. strname: '',
  466. dataDh: '-', //压差
  467. dataDtestq: '-', //测试风量
  468. sourcePressure: '-', //气源压力
  469. dataDequivalarea: '-',
  470. netStatus: '0', //通信状态
  471. warnLevel_str: '',
  472. stationname: ''
  473. };
  474. const dataSource = ref([]);
  475. // 监测数据
  476. let selectData = reactive(lodash.cloneDeep(initData));
  477. const rightColumns = ref<BasicColumn[]>([])
  478. const leftColumns = ref<BasicColumn[]>([])
  479. const devicekide = ref('fanlocal')
  480. const deviceType = ref(selectData.deviceType)
  481. watch(deviceType , (type) => {
  482. rightColumns.value = getTableHeaderColumns(type + '_monitor_right') as []
  483. if(rightColumns.value && rightColumns.value.length < 1){
  484. rightColumns.value = getTableHeaderColumns(type.split('_')[0] + '_monitor_right') as []
  485. }
  486. leftColumns.value = getTableHeaderColumns(type + '_monitor_left') as []
  487. if (leftColumns.value && leftColumns.value.length < 1) {
  488. leftColumns.value = getTableHeaderColumns(type.split('_')[0] + '_monitor_left') as []
  489. }
  490. })
  491. const flvURL1 = () => {
  492. // return `https://sf1-hscdn-tos.pstatp.com/obj/media-fe/xgplayer_doc_video/flv/xgplayer-demo-360p.flv`;
  493. return ''
  494. };
  495. const changeDeviceKind = (e) => {
  496. devicekide.value = e
  497. loading.value = true
  498. selectRowIndex.value = -1
  499. nextTick( () => {
  500. // selectRowIndex.value = 0
  501. selectData = lodash.cloneDeep(initData)
  502. loading.value = false
  503. if(selectData.deviceID)MonitorDataTable.value.setSelectedRowKeys([selectData.deviceID])
  504. })
  505. }
  506. const tabChange = (activeKeyVal) => {
  507. activeKey.value = activeKeyVal;
  508. // if (activeKeyVal == 1) {
  509. // nextTick(() => {
  510. // MonitorDataTable.value.setSelectedRowKeys([selectData.deviceID])
  511. // })
  512. // }
  513. };
  514. const selectDevice = (key, val) => {
  515. if (key === 'dataMonitorRowIndex') {
  516. dataMonitorRowIndex.value = val;
  517. } else {
  518. warningMonitorRowIndex.value = val;
  519. }
  520. };
  521. //详情
  522. function goDetail() {
  523. openModal()
  524. }
  525. //
  526. async function getDataSource(){
  527. const res = await list({ devicetype: devicekide.value, pagetype: 'normal' });
  528. if(res.msgTxt && res.msgTxt[0] && res.msgTxt[0].datalist && res.msgTxt[0].datalist.length > 0){
  529. const dataArr = res.msgTxt[0].datalist || [];
  530. dataSource.value = [];
  531. dataArr.forEach((data) => {
  532. const readData = data.readData;
  533. data = Object.assign(data, readData);
  534. dataSource.value.push(data);
  535. });
  536. if (MonitorDataTable.value && selectRowIndex.value == -1) {
  537. MonitorDataTable.value.setSelectedRowKeys([dataSource.value[0]['deviceID']])
  538. }
  539. const data: any = toRaw(dataSource.value[selectRowIndex.value]); //maxarea
  540. return data;
  541. }else{
  542. return dataSource.value = []
  543. }
  544. };
  545. // https获取监测数据
  546. let timer: null | NodeJS.Timeout = null;
  547. async function getMonitor(flag?) {
  548. if (Object.prototype.toString.call(timer) === '[object Null]') {
  549. timer = await setTimeout(async () => {
  550. await getDataSource();
  551. if (dataSource.value.length > 0 && selectRowIndex.value == -1 && MonitorDataTable.value) {
  552. // 初始打开页面
  553. if (currentRoute.value && currentRoute.value['query'] && currentRoute.value['query']['id']) {
  554. MonitorDataTable.value.setSelectedRowKeys(currentRoute.value['query']['id'])
  555. }else{
  556. MonitorDataTable.value.setSelectedRowKeys(dataSource.value[0]['deviceID'])
  557. }
  558. }
  559. selectData = Object.assign(deviceBaseList.value, lodash.cloneDeep(initData))
  560. if(dataSource.value.length > 0 && dataSource.value[selectRowIndex.value] ){
  561. Object.assign(selectData, dataSource.value[selectRowIndex.value])
  562. deviceType.value = selectData['deviceType']
  563. addText(selectData);
  564. // playAnimation(data, selectData.maxarea);
  565. playSmoke(selectData)
  566. }else{
  567. // lodash.omit(selectData, lodash.keys(selectData))
  568. Object.assign(selectData, initData)
  569. deviceType.value = ''
  570. }
  571. if (timer) {
  572. timer = null;
  573. }
  574. getMonitor();
  575. }, flag ? 0 :1000);
  576. }
  577. };
  578. // 获取设备基本信息列表
  579. const deviceBaseList = ref([]);
  580. function getDeviceBaseList() {
  581. getTableList({ pageSize: 1000 }).then((res) => {
  582. deviceBaseList.value = res.records;
  583. });
  584. };
  585. // 切换检测数据
  586. function getSelectRow(id) {
  587. console.log('选中的设备id------->', id)
  588. if (!id) return;
  589. loading.value = true;
  590. const baseDataIndex: any = dataSource.value.findIndex((baseData: any) => baseData.deviceID == id);
  591. selectRowIndex.value = baseDataIndex;
  592. modalType.value = baseDataIndex > 0 ? 'fm' : 'fc';
  593. nextTick(() => {
  594. setModelType(modalType.value).then(() => {
  595. loading.value = false;
  596. });
  597. const data = dataSource.value[baseDataIndex];
  598. if (data['Fan1StartStatus'] == 1) {
  599. mainWindIsShow1.value = 'open';
  600. mainWindIsShow2.value = 'stop';
  601. } else if (data['Fan2StartStatus'] == 1) {
  602. mainWindIsShow2.value = 'open';
  603. mainWindIsShow1.value = 'stop';
  604. }
  605. })
  606. // addCssText();
  607. return;
  608. };
  609. // 打开并设置modal的标题
  610. function showModal(obj) {
  611. modalType.value = obj.key;
  612. modalTitle.value = obj.value;
  613. passWord.value = '';
  614. modalIsShow.value = true;
  615. };
  616. function changeMotor(e){
  617. const target = e.target;
  618. if (target.name === 'localWind1') {
  619. if (target.value === 'open') {
  620. mainWindIsShow2.value = 'stop';
  621. }
  622. } else if (target.name === 'localWind2') {
  623. if (target.value === 'open') {
  624. mainWindIsShow1.value = 'stop';
  625. }
  626. }
  627. };
  628. function handleOk(e: MouseEvent) {
  629. const handType = modalType.value;
  630. const data = {
  631. deviceid: selectData.deviceID,
  632. devicetype: selectData.deviceType,
  633. paramcode: '',
  634. password: passWord.value,
  635. value: null,
  636. };
  637. if (handType === 'startSmoke') {
  638. // 启动风机
  639. if (mainWindIsShow1.value === 'open' && mainWindIsShow2.value === 'stop') {
  640. // playSmoke(handType, 'top', frequency, 'open');
  641. data.paramcode = 'CtrlFan1Start';
  642. deviceControlApi(data).then(() => {
  643. if (globalConfig.History_Type == 'remote') {
  644. message.success('指令已下发至生产管控平台成功!')
  645. } else {
  646. message.success('指令已下发成功!')
  647. }
  648. modalTitle.value = '';
  649. modalIsShow.value = false;
  650. }).catch((err) => {
  651. });
  652. } else if (mainWindIsShow2.value === 'open' && mainWindIsShow1.value === 'stop') {
  653. // playSmoke(handType, 'down', frequency, 'open');
  654. data.paramcode = 'CtrlFan2Start';
  655. deviceControlApi(data).then(() => {
  656. if (globalConfig.History_Type == 'remote') {
  657. message.success('指令已下发至生产管控平台成功!')
  658. } else {
  659. message.success('指令已下发成功!')
  660. }
  661. modalTitle.value = '';
  662. modalIsShow.value = false;
  663. }).catch((err) => {
  664. });
  665. } else if (mainWindIsShow1.value === 'stop' && mainWindIsShow2.value === 'stop') {
  666. // playSmoke(handType, '', frequency, 'stop');
  667. }
  668. } else if (handType === 'Fan1Frequency' || handType === 'Fan2Frequency') {
  669. // 调频
  670. if (handType === 'Fan1Frequency') {
  671. data.paramcode = 'Fan1FreqHz';
  672. data.value = fan1FrequencyVal.value;
  673. } else if (handType === 'Fan2Frequency') {
  674. data.paramcode = 'Fan2FreqHz';
  675. data.value = fan2FrequencyVal.value;
  676. }
  677. deviceControlApi(data).then((res) => {
  678. if (globalConfig.History_Type == 'remote') {
  679. message.success('指令已下发至生产管控平台成功!')
  680. } else {
  681. message.success('指令已下发成功!')
  682. }
  683. modalTitle.value = '';
  684. modalIsShow.value = false;
  685. }).catch((err) => {
  686. });
  687. } else if (handType === 'changeSmoke') {
  688. if(selectData['Fan1StartStatus'] == 0 || !selectData['Fan1StartStatus']){
  689. data.paramcode = 'CtrlFan1Start';
  690. deviceControlApi(data).then((res) => {
  691. if (globalConfig.History_Type == 'remote') {
  692. message.success('指令已下发至生产管控平台成功!')
  693. } else {
  694. message.success('指令已下发成功!')
  695. }
  696. modalTitle.value = '';
  697. modalIsShow.value = false;
  698. mainWindIsShow1.value = 'stop';
  699. mainWindIsShow2.value = 'open';
  700. }).catch((err) => {
  701. });
  702. }else if(selectData['Fan2StartStatus'] == 0 || !selectData['Fan2StartStatus']){
  703. data.paramcode = 'CtrlFan2Start';
  704. deviceControlApi(data).then((res) => {
  705. if (globalConfig.History_Type == 'remote') {
  706. message.success('指令已下发至生产管控平台成功!')
  707. } else {
  708. message.success('指令已下发成功!')
  709. }
  710. modalTitle.value = '';
  711. modalIsShow.value = false;
  712. mainWindIsShow1.value = 'open';
  713. mainWindIsShow2.value = 'stop';
  714. }).catch((err) => {
  715. });
  716. }
  717. // // 一键倒机
  718. // if (mainWindIsShow1.value === 'open' && mainWindIsShow2.value === 'stop') {
  719. // // playSmoke('startSmoke', 'down', frequency, 'open');
  720. // data.paramcode = 'CtrlFan2Start';
  721. // deviceControlApi(data).then((res) => {
  722. // console.log('设备操作结果', res);
  723. // modalTitle.value = '';
  724. // modalIsShow.value = false;
  725. // }).catch((err) => {
  726. // });
  727. // mainWindIsShow1.value = 'stop';
  728. // mainWindIsShow2.value = 'open';
  729. // } else if (mainWindIsShow2.value === 'open' && mainWindIsShow1.value === 'stop') {
  730. // // playSmoke('startSmoke', 'top', frequency, 'open');
  731. // data.paramcode = 'CtrlFan1Start';
  732. // deviceControlApi(data).then((res) => {
  733. // console.log('设备操作结果', res);
  734. // modalTitle.value = '';
  735. // modalIsShow.value = false;
  736. // }).catch((err) => {
  737. // });
  738. // mainWindIsShow1.value = 'open';
  739. // mainWindIsShow2.value = 'stop';
  740. // } else if (mainWindIsShow1.value === 'stop' && mainWindIsShow2.value === 'stop') {
  741. // // playSmoke(handType, '', frequency, 'stop');
  742. // }
  743. }
  744. };
  745. function handleChangeSensor(value: string){
  746. console.log(value);
  747. };
  748. function addPlayVideo() {
  749. if (player1.value && player1.value.play) {
  750. if (!player1.value.paused()) player1.value.play();
  751. document.body.removeEventListener('mousedown', addPlayVideo);
  752. }
  753. };
  754. onBeforeMount(() => {
  755. getDeviceBaseList();
  756. });
  757. onMounted(() => {
  758. mountedThree(player1.value).then(() => {
  759. nextTick(async() => {
  760. await getMonitor(true);
  761. addCssText();
  762. // playSmoke('startSmoke', 'top', frequencyVal.value, 'open');
  763. });
  764. });
  765. document.body.addEventListener('mousedown', addPlayVideo, false);
  766. });
  767. onUnmounted(() => {
  768. destroy();
  769. if (timer) {
  770. clearTimeout(timer);
  771. timer = undefined;
  772. }
  773. });
  774. </script>
  775. <style scoped lang="less">
  776. @import '/@/design/vent/modal.less';
  777. :deep(.@{ventSpace}-tabs-tabpane-active) {
  778. height: 100%;
  779. }
  780. .scene-box {
  781. .title-text{
  782. height: 32px;
  783. }
  784. .bottom-tabs-box{
  785. height: 280px;
  786. .tabs-box{
  787. position: relative !important;
  788. }
  789. }
  790. }
  791. .data-show-box {
  792. position: relative;
  793. display: flex;
  794. flex-direction: row;
  795. justify-content: space-between;
  796. padding: 10px 5px;
  797. color: #ffffff;
  798. z-index: 999;
  799. top: 60px;
  800. .data-item {
  801. pointer-events: auto;
  802. .item-header {
  803. width: 374px;
  804. background: url('/@/assets/images/vent/lr-top-bg.png') no-repeat;
  805. background-size: auto;
  806. height: 32px;
  807. text-align: center;
  808. line-height: 34px;
  809. font-size: 15px;
  810. font-weight: 600;
  811. color: #fafafa;
  812. }
  813. .item-container {
  814. width: 346px;
  815. margin: 0 14px;
  816. padding: 10px;
  817. background: #00377c33;
  818. backdrop-filter: blur(2px);
  819. .tab {
  820. width: 323px;
  821. background: url('/@/assets/images/vent/lr-tab-bg.png') no-repeat;
  822. display: flex;
  823. .tab-item {
  824. flex: 1;
  825. text-align: center;
  826. padding-top: 2px;
  827. color: #ffffff99;
  828. cursor: pointer;
  829. &:hover {
  830. color: #ffffff;
  831. }
  832. }
  833. .tab-item-active-l {
  834. color: #ffffff;
  835. background-image: url('/@/assets/images/vent/l-tab-active.png');
  836. background-repeat: no-repeat;
  837. background-size: auto;
  838. background-position: 6px 3px;
  839. }
  840. .tab-item-active-r {
  841. color: #ffffff;
  842. background-image: url('/@/assets/images/vent/r-tab-active.png');
  843. background-repeat: no-repeat;
  844. background-position: 0 3px;
  845. }
  846. }
  847. .container-group {
  848. width: 314px;
  849. margin: 0px 4px;
  850. padding: 10px 0;
  851. min-height: 432px;
  852. background: linear-gradient(to right, #00deff22, #2081ff05);
  853. max-height: 440px;
  854. overflow-y: auto;
  855. }
  856. .container-group-l {
  857. .container-item {
  858. width: 100%;
  859. height: 60px;
  860. display: flex;
  861. padding: 10px 0 0 20px;
  862. margin-bottom: 5px;
  863. position: relative;
  864. background: url('/@/assets/images/vent/plane-bottom.png') no-repeat;
  865. background-size: auto;
  866. background-position: bottom;
  867. &::before {
  868. content: '';
  869. display: block;
  870. width: 100%;
  871. height: 5px;
  872. position: absolute;
  873. top: 62px;
  874. left: 0;
  875. background-color: #73f4ff66;
  876. backdrop-filter: blur(5px);
  877. }
  878. .item-icon {
  879. width: 54px;
  880. height: 45px;
  881. background: url('/@/assets/images/vent/plane-icon-bg.png') no-repeat;
  882. background-size: cover;
  883. .icon-style {
  884. margin: 10px 0 0 18px;
  885. }
  886. }
  887. .item-name {
  888. width: 180px;
  889. line-height: 60px;
  890. }
  891. .item-value {
  892. position: relative;
  893. height: 26px;
  894. line-height: 24px;
  895. margin: 15px 0;
  896. text-align: center;
  897. width: 80px;
  898. border: 1px solid #00f5fe;
  899. border-radius: 13px;
  900. background: linear-gradient(to right, #00f5fe44, #0090ff44);
  901. &::before {
  902. width: 6px;
  903. height: 6px;
  904. content: '';
  905. position: absolute;
  906. left: -3px;
  907. top: 8px;
  908. background: #ffa500;
  909. border-radius: 3px;
  910. }
  911. }
  912. }
  913. }
  914. .warning-header {
  915. display: flex;
  916. font-size: 14px;
  917. .header-item {
  918. flex: 1;
  919. display: flex;
  920. flex-direction: column;
  921. margin: 5px 10px;
  922. justify-content: center;
  923. align-items: center;
  924. .header-title {
  925. color: #39e7fe;
  926. }
  927. .header-value {
  928. width: 133px;
  929. height: 56px;
  930. font-weight: 600;
  931. font-family: 'douyuFont';
  932. font-size: 16px;
  933. color: #ffa500;
  934. display: flex;
  935. justify-content: center;
  936. align-items: center;
  937. background: url('/@/assets/images/vent/count-header-bg.png') no-repeat;
  938. }
  939. }
  940. }
  941. .warning-group {
  942. padding: 0 10px;
  943. position: relative;
  944. &::before {
  945. content: '';
  946. display: block;
  947. width: 1px;
  948. height: 100%;
  949. position: absolute;
  950. left: 12px;
  951. background-color: #00f5fe;
  952. }
  953. .warning-item {
  954. display: flex;
  955. flex-direction: row;
  956. justify-content: space-between;
  957. align-items: center;
  958. height: 38px;
  959. .item-name {
  960. .icon {
  961. width: 6px;
  962. height: 6px;
  963. display: inline-block;
  964. background-color: #1cd5ff;
  965. border-radius: 3px;
  966. position: relative;
  967. margin-right: 5px;
  968. &::before {
  969. content: '';
  970. width: 10px;
  971. height: 10px;
  972. display: block;
  973. border: 1px solid #34edff99;
  974. border-radius: 5px;
  975. position: absolute;
  976. top: -2px;
  977. left: -2px;
  978. }
  979. }
  980. }
  981. }
  982. }
  983. }
  984. }
  985. }
  986. .input-box {
  987. display: flex;
  988. align-items: center;
  989. .input-title {
  990. color: #73e8fe;
  991. width: auto;
  992. }
  993. margin-right: 10px;
  994. }
  995. .label {
  996. max-width: 220px;
  997. }
  998. #fanLocalSelectDom{
  999. :deep(.@{ventSpace}-select-dropdown){
  1000. left: 0px !important;
  1001. top: 35px !important;
  1002. }
  1003. }
  1004. .@{ventSpace}-input {
  1005. width: 150px;
  1006. }
  1007. </style>