index.vue 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960
  1. <template>
  2. <div
  3. class="alarmBox"
  4. :class="getAlarmBoxClass()"
  5. :style="`width: ${getAlarmBoxWidth()}px;`"
  6. >
  7. <div
  8. class="columnItem"
  9. v-for="columnNumber in $store.state.columnNumber"
  10. :key="columnNumber"
  11. v-show="alarmList[(columnNumber - 1) * $store.state.alarmShowNumber]"
  12. >
  13. <div
  14. :class="`${
  15. index >= (columnNumber - 1) * $store.state.alarmShowNumber &&
  16. index < columnNumber * $store.state.alarmShowNumber
  17. ? item.class + ' alarmItem animate__animated'
  18. : ''
  19. }`"
  20. v-for="(item, index) in alarmList"
  21. :key="index"
  22. >
  23. <template
  24. v-if="
  25. index >= (columnNumber - 1) * $store.state.alarmShowNumber &&
  26. index < columnNumber * $store.state.alarmShowNumber
  27. "
  28. >
  29. <div class="alarmTitle">{{ item.wpName }}&nbsp;{{ item.code }}</div>
  30. <div class="alarmContent">
  31. <div class="contentItem" @click="goToAlertDescPage(item)">
  32. 报警描述:
  33. <span class="alertDescCursor">{{ item.description }}</span>
  34. </div>
  35. <div class="contentItem">报警时间:{{ item.tsName }}</div>
  36. </div>
  37. <div class="btnBox" :class="`lv${item.lv}BdColor`">
  38. <div class="btnItem" :class="`lv${item.lv}BdColor lv${item.lv}`">
  39. <el-button
  40. class="comfirmBtn"
  41. size="small"
  42. type="text"
  43. @click="comfirm(item)"
  44. >确认本条</el-button
  45. >
  46. </div>
  47. <div class="btnItem" :class="`lv${item.lv}`">
  48. <el-button
  49. class="comfirmBtn"
  50. size="small"
  51. type="text"
  52. @click="comfirmAll"
  53. >全部确认</el-button
  54. >
  55. </div>
  56. </div>
  57. </template>
  58. </div>
  59. </div>
  60. <div
  61. class="collapseBtn"
  62. v-if="alarmList?.[0]?.id && $store.state.alarmShowNumber > 0"
  63. @click="
  64. () => {
  65. isCollapse = !isCollapse;
  66. }
  67. "
  68. >
  69. <el-link type="primary" href="javascript:;">
  70. <!-- <el-icon color="#909399"><CaretBottom /></el-icon> -->
  71. <span style="margin-left: 6px" v-if="!isCollapse">点此折叠报警</span>
  72. <span style="margin-left: 6px; color: var(--el-color-danger)" v-else
  73. >报警已折叠,点此恢复</span
  74. >
  75. </el-link>
  76. </div>
  77. </div>
  78. </template>
  79. <script>
  80. import { confirmAlart, alarm_history } from "@api/api.js";
  81. import { ElNotification } from "element-plus";
  82. import dayJs from "dayjs";
  83. export default {
  84. data() {
  85. return {
  86. alarmConfigArray: [],
  87. alarmList: [],
  88. seriousWarning: false,
  89. audioElement: null,
  90. ws: null,
  91. timeConnect: 0,
  92. limitConnect: 5,
  93. columnNumber: 2,
  94. showSocketLog: false,
  95. isCollapse: false,
  96. requestAlarmHistoryParams: [
  97. {
  98. alarmType: "booststation",
  99. deviceType: "",
  100. },
  101. {
  102. alarmType: "inverter",
  103. deviceType: "",
  104. },
  105. {
  106. alarmType: "windturbine",
  107. deviceType: "",
  108. },
  109. {
  110. alarmType: "custom",
  111. deviceType: "booststation",
  112. },
  113. {
  114. alarmType: "custom",
  115. deviceType: "inverter",
  116. },
  117. {
  118. alarmType: "custom",
  119. deviceType: "windturbine",
  120. },
  121. ],
  122. // websocket相关
  123. socketObj: "", // websocket实例对象
  124. //心跳检测
  125. heartCheck: {
  126. vueThis: this, // vue实例
  127. timeout: 30000, // 超时时间
  128. timeoutObj: null, // 计时器对象——向后端发送心跳检测
  129. serverTimeoutObj: null, // 计时器对象——等待后端心跳检测的回复
  130. // 心跳检测重置
  131. reset: function () {
  132. clearTimeout(this.timeoutObj);
  133. clearTimeout(this.serverTimeoutObj);
  134. return this;
  135. },
  136. // 心跳检测启动
  137. start: function () {
  138. this.timeoutObj && clearTimeout(this.timeoutObj);
  139. this.serverTimeoutObj && clearTimeout(this.serverTimeoutObj);
  140. this.timeoutObj = setTimeout(() => {
  141. // 这里向后端发送一个心跳检测,后端收到后,会返回一个心跳回复
  142. this.vueThis.socketObj.send("HeartBeat");
  143. this.showSocketLog && console.log("发送心跳检测");
  144. this.serverTimeoutObj = setTimeout(() => {
  145. // 如果超过一定时间还没重置计时器,说明websocket与后端断开了
  146. this.showSocketLog && console.log("未收到心跳检测回复");
  147. // 关闭WebSocket
  148. this.vuethis.socketObj && this.socketObj.close();
  149. }, this.timeout);
  150. }, this.timeout);
  151. },
  152. },
  153. socketReconnectTimer: null, // 计时器对象——重连
  154. socketReconnectLock: false, // WebSocket重连的锁
  155. socketLeaveFlag: false, // 离开标记(解决 退出登录再登录 时出现的 多次相同推送 问题,出现的本质是多次建立了WebSocket连接)
  156. };
  157. },
  158. created() {
  159. this.getAlarmConfig();
  160. let requestResult = [];
  161. this.requestAlarmHistoryParams.forEach(({ alarmType, deviceType }) => {
  162. requestResult.push(this.getAlarmHistory(alarmType, deviceType));
  163. });
  164. Promise.all(requestResult)
  165. .then((promiseResult) => {
  166. promiseResult.forEach(({ data }) => {
  167. data?.ls?.forEach((ele) => {
  168. this.pushALarmItem(ele);
  169. });
  170. });
  171. // this.webSocketInit(
  172. // `ws://10.81.3.154:6014/websocket/${this.$store.state.user.userId}_${this.$store.state.user.authToken}`
  173. // );
  174. this.createWebSocket();
  175. })
  176. .catch(() => {
  177. requestResult.forEach((ele, index) => {
  178. ele
  179. .then(({ data }) => {
  180. data?.ls?.forEach((ele) => {
  181. this.pushALarmItem(ele);
  182. });
  183. })
  184. .catch((error) => {
  185. ElNotification({
  186. type: "error",
  187. title: "查询历史未处理报警请求出错!",
  188. dangerouslyUseHTMLString: true,
  189. message: `<div class="currentRequestErrorNotification">
  190. <p><span>主要参数:</p>
  191. <p style="color:var(--el-color-primary)"><span class="errorTitle">alarmType:</span><span class="errorDesc">"${this.requestAlarmHistoryParams[index].alarmType}"</span></p>
  192. <p style="color:var(--el-color-primary)"><span class="errorTitle">deviceType:</span><span class="errorDesc">"${this.requestAlarmHistoryParams[index].deviceType}"</span></p>
  193. <p style="color:var(--el-color-danger)"><span class="errorTitle">错误正文:</span><span class="errorDesc">${error}</span></p>
  194. </div>`,
  195. });
  196. });
  197. });
  198. });
  199. },
  200. unmounted() {
  201. this.socketLeaveFlag = true;
  202. this.socketObj && this.socketObj.close();
  203. },
  204. methods: {
  205. getAlarmName(alarmItem) {
  206. let alarmName = "";
  207. if (alarmItem.deviceType === "booststation") {
  208. alarmName = "升压站报警";
  209. } else if (alarmItem.deviceType === "inverter") {
  210. alarmName = "光伏报警";
  211. } else if (alarmItem.deviceType === "windturbine") {
  212. alarmName = "设备报警";
  213. } else if (alarmItem.deviceType === "station") {
  214. alarmName = "场站";
  215. }
  216. if (alarmItem.alarmType === "custom") {
  217. alarmName = "自定义报警";
  218. }
  219. return alarmName;
  220. },
  221. getLvName(alarmItem) {
  222. if (alarmItem.rank === 1) {
  223. return "低级";
  224. } else if (alarmItem.rank === 2) {
  225. return "低中级";
  226. } else if (alarmItem.rank === 3) {
  227. return "中级";
  228. } else if (alarmItem.rank === 4) {
  229. return "中高级";
  230. } else if (alarmItem.rank === 5) {
  231. return "高级";
  232. }
  233. },
  234. comfirm(item) {
  235. this.$confirm("您确定要执行此操作吗?", "提示", {
  236. confirmButtonText: "确定",
  237. cancelButtonText: "取消",
  238. type: "warning",
  239. }).then(() => {
  240. item.class = `animate__bounceOutRight lv${item.lv}`;
  241. item.confirm = true;
  242. setTimeout(() => {
  243. item.class = `animate__bounceOutRight hidden lv${item.lv}`;
  244. confirmAlart([item]).then((res) => {
  245. if (res.code === 200) {
  246. this.BASE.showMsg({
  247. type: "success",
  248. msg: "确认成功",
  249. });
  250. this.$store.commit("removeWarning", item);
  251. this.playAudioEffect();
  252. }
  253. });
  254. }, 500);
  255. });
  256. },
  257. comfirmAll() {
  258. this.$confirm("您确定要执行此操作吗?", "提示", {
  259. confirmButtonText: "确定",
  260. cancelButtonText: "取消",
  261. type: "warning",
  262. }).then(() => {
  263. for (let i = 0; i < this.alarmList.length; i++) {
  264. if (!this.alarmList[i].comfirm) {
  265. this.alarmList[
  266. i
  267. ].class = `animate__bounceOutRight lv${this.alarmList[i].lv}`;
  268. this.alarmList[i].confirm = true;
  269. setTimeout(() => {
  270. this.alarmList[i] &&
  271. (this.alarmList[i].class = `animate__bounceOutRight hidden lv${
  272. this.alarmList[i]?.lv || 3
  273. }`);
  274. }, 500);
  275. }
  276. }
  277. confirmAlart(this.alarmList).then((res) => {
  278. if (res.code === 200) {
  279. this.BASE.showMsg({
  280. type: "success",
  281. msg: "全部确认成功",
  282. });
  283. this.alarmList = [];
  284. this.$store.commit("emptyWarning");
  285. this.playAudioEffect();
  286. }
  287. });
  288. this.playAudioEffect();
  289. });
  290. },
  291. playAudioEffect() {
  292. const lv1Config = this.getConfigItem(1);
  293. let lv1Play = false;
  294. if (lv1Config.isAlarmSound) {
  295. lv1Play = this.alarmList.some((ele) => {
  296. return ele.lv === 1 && !ele.confirm;
  297. });
  298. }
  299. const lv2Config = this.getConfigItem(2);
  300. let lv2Play = false;
  301. if (lv2Config.isAlarmSound) {
  302. lv2Play = this.alarmList.some((ele) => {
  303. return ele.lv === 2 && !ele.confirm;
  304. });
  305. }
  306. const lv3Config = this.getConfigItem(3);
  307. let lv3Play = false;
  308. if (lv3Config.isAlarmSound) {
  309. lv3Play = this.alarmList.some((ele) => {
  310. return ele.lv === 3 && !ele.confirm;
  311. });
  312. }
  313. const lv4Config = this.getConfigItem(4);
  314. let lv4Play = false;
  315. if (lv4Config.isAlarmSound) {
  316. lv4Play = this.alarmList.some((ele) => {
  317. return ele.lv === 4 && !ele.confirm;
  318. });
  319. }
  320. const lv5Config = this.getConfigItem(5);
  321. let lv5Play = false;
  322. if (lv5Config.isAlarmSound) {
  323. lv5Play = this.alarmList.some((ele) => {
  324. return ele.lv === 5 && !ele.confirm;
  325. });
  326. }
  327. if (lv5Play && !this.seriousWarning) {
  328. this.seriousWarning = true;
  329. this.audioElement = new Audio();
  330. this.audioElement.src = "./static/sound/lv5.mp3";
  331. this.audioElement.loop = true;
  332. this.audioElement?.play();
  333. } else if (
  334. (lv1Play || lv2Play || lv3Play || lv4Play) &&
  335. !this.seriousWarning
  336. ) {
  337. this.audioElement = new Audio();
  338. this.audioElement.src = "./static/sound/lv4.mp3";
  339. this.audioElement.addEventListener("ended", () => {
  340. this.audioElement?.removeEventListener(
  341. "ended",
  342. this.stopPlayAudioEffect
  343. );
  344. });
  345. this.audioElement?.play();
  346. } else {
  347. if (!this.seriousWarning) {
  348. this.stopPlayAudioEffect();
  349. }
  350. }
  351. },
  352. stopPlayAudioEffect() {
  353. this.seriousWarning = false;
  354. if (this.audioElement) {
  355. this.audioElement.pause();
  356. this.audioElement.currentTime = 0;
  357. this.audioElement.loop = false;
  358. }
  359. this.audioElement = null;
  360. },
  361. getAlarmHistory(alarmType, deviceType) {
  362. let params = {
  363. pageNum: 1,
  364. pageSize: 10,
  365. alarmId: "",
  366. alarmType,
  367. deviceType,
  368. stationid: "",
  369. deviceid: "",
  370. modelId: "",
  371. components: "",
  372. description: "",
  373. begin: dayJs().add(-1, "hour").format("YYYY-MM-DD HH:mm:ss"),
  374. end: dayJs().format("YYYY-MM-DD HH:mm:ss"),
  375. isclose: false,
  376. };
  377. if (
  378. params.alarmType == "windturbine" ||
  379. (params.alarmType == "custom" && params.deviceType == "windturbine")
  380. ) {
  381. params.stationid = "SXJ_KGDL_DJY_FDC_STA";
  382. } else if (
  383. params.alarmType == "inverter" ||
  384. (params.alarmType == "custom" && params.deviceType == "inverter")
  385. ) {
  386. params.stationid = "SXJ_KGDL_JR_GDC_STA";
  387. }
  388. return alarm_history(params, 12000);
  389. },
  390. // websocket启动
  391. createWebSocket() {
  392. let webSocketLink = `ws://10.81.3.154:6014/websocket/${this.$store.state.user.userId}_${this.$store.state.user.authToken}`; // webSocket地址
  393. try {
  394. if ("WebSocket" in window) {
  395. this.socketObj = new WebSocket(webSocketLink);
  396. }
  397. // websocket事件绑定
  398. this.socketEventBind();
  399. } catch (e) {
  400. this.showSocketLog && console.log("catch" + e);
  401. // websocket重连
  402. this.socketReconnect();
  403. }
  404. },
  405. // websocket事件绑定
  406. socketEventBind() {
  407. // 连接成功建立的回调
  408. this.socketObj.onopen = this.onopenCallback;
  409. // 连接发生错误的回调
  410. this.socketObj.onerror = this.onerrorCallback;
  411. // 连接关闭的回调
  412. this.socketObj.onclose = this.oncloseCallback;
  413. // 向后端发送数据的回调
  414. this.socketObj.onsend = this.onsendCallback;
  415. // 接收到消息的回调
  416. this.socketObj.onmessage = this.getMessageCallback;
  417. //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
  418. window.onbeforeunload = () => {
  419. this.socketObj.close();
  420. };
  421. },
  422. // websocket重连
  423. socketReconnect() {
  424. if (this.socketReconnectLock) {
  425. return;
  426. }
  427. this.socketReconnectLock = true;
  428. this.socketReconnectTimer && clearTimeout(this.socketReconnectTimer);
  429. this.socketReconnectTimer = setTimeout(() => {
  430. this.showSocketLog && console.log("WebSocket:重连中...");
  431. this.socketReconnectLock = false;
  432. // websocket启动
  433. this.createWebSocket();
  434. }, 4000);
  435. },
  436. // 连接成功建立的回调
  437. onopenCallback(event) {
  438. this.showSocketLog && console.log("WebSocket:已连接");
  439. // 心跳检测重置
  440. this.heartCheck.reset().start();
  441. },
  442. // 连接发生错误的回调
  443. onerrorCallback(event) {
  444. this.showSocketLog && console.log("WebSocket:发生错误");
  445. // websocket重连
  446. this.socketReconnect();
  447. },
  448. // 连接关闭的回调
  449. oncloseCallback(event) {
  450. this.showSocketLog && console.log("WebSocket:已关闭");
  451. // 心跳检测重置
  452. this.heartCheck.reset();
  453. if (!this.socketLeaveFlag) {
  454. // 没有离开——重连
  455. // websocket重连
  456. this.socketReconnect();
  457. } else {
  458. this.socketObj && this.socketObj.close();
  459. }
  460. },
  461. // 向后端发送数据的回调
  462. onsendCallback() {
  463. this.showSocketLog && console.log("WebSocket:发送信息给后端");
  464. },
  465. // 接收到消息的回调
  466. getMessageCallback(msg) {
  467. // console.log(msg);
  468. if (Object.keys(msg) && msg.data == "ok") {
  469. // 心跳回复——心跳检测重置
  470. // 收到心跳检测回复就说明连接正常
  471. this.showSocketLog && console.log("收到心跳检测回复");
  472. // 心跳检测重置
  473. this.heartCheck.reset().start();
  474. } else {
  475. // 普通推送——正常处理
  476. this.showSocketLog && console.log("收到推送消息");
  477. // 相关处理
  478. let alarmItem = JSON.parse(msg.data);
  479. if (alarmItem) {
  480. this.pushALarmItem(alarmItem);
  481. }
  482. }
  483. },
  484. // webSocketInit(serveIP) {
  485. // if ("WebSocket" in window) {
  486. // this.ws = new WebSocket(serveIP);
  487. // this.ws.onmessage = (res) => {
  488. // let alarmItem = JSON.parse(res.data);
  489. // if (alarmItem) {
  490. // this.pushALarmItem(alarmItem);
  491. // }
  492. // };
  493. // this.ws.onclose = () => {
  494. // this.ws = null;
  495. // };
  496. // this.ws.onopen = () => {
  497. // this.timeConnect = 0;
  498. // this.showSocketLog && console.log("WebSocket 服务已建立");
  499. // };
  500. // this.ws.onerror = () => {
  501. // this.reconnect(serveIP);
  502. // };
  503. // } else {
  504. // this.BASE.showMsg({
  505. // msg: "当前浏览器不支持 WebSocket ,请更换浏览器后重试",
  506. // });
  507. // }
  508. // },
  509. pushALarmItem(alarmItem) {
  510. const configItem = this.getConfigItem(alarmItem.rank);
  511. const alarmOption = {
  512. id: alarmItem.id,
  513. lv: alarmItem.rank,
  514. lvName: this.getLvName(alarmItem),
  515. rank: alarmItem.rank,
  516. class: `animate__bounceInRight lv${alarmItem.rank}`,
  517. isClose: alarmItem.closeTime ? true : alarmItem.endts ? true : false,
  518. isCloseName: alarmItem.isClose ? "已解除" : "未解除",
  519. alarmId: alarmItem.alarmId,
  520. alarmType: alarmItem.alarmType,
  521. alarmName: this.getAlarmName(alarmItem),
  522. description: alarmItem.description,
  523. deviceType: alarmItem.deviceType,
  524. oval: alarmItem.oval,
  525. triggerType: alarmItem.triggerType,
  526. ts: alarmItem.ts,
  527. tsName: new Date(alarmItem.ts).formatDate("MM-dd hh:mm:ss"),
  528. fullTsName: new Date(alarmItem.ts).formatDate("yyyy-MM-dd hh:mm:ss"),
  529. endts: alarmItem.endts
  530. ? dayjs(alarmItem.endts).format("YYYY-MM-DD HH:mm:ss")
  531. : alarmItem.closeTime
  532. ? dayjs(alarmItem.closeTime).format("YYYY-MM-DD HH:mm:ss")
  533. : null,
  534. // endtsName:
  535. // alarmItem.endts > 0
  536. // ? new Date(alarmItem.endts).formatDate("yyyy-MM-dd hh:mm:ss")
  537. // : "",
  538. endtsName:
  539. alarmItem.endts > 0
  540. ? new Date(alarmItem.endts).formatDate("yyyy-MM-dd hh:mm:ss")
  541. : "",
  542. deviceId: alarmItem.deviceId,
  543. faultCause: alarmItem.faultCause,
  544. resolvent: alarmItem.resolvent,
  545. characteristic: alarmItem.characteristic,
  546. code: alarmItem.code,
  547. wpName: alarmItem.wpName,
  548. stationId: alarmItem.stationid,
  549. };
  550. if (
  551. configItem.isAlarmSound ||
  552. configItem.isAlart ||
  553. configItem.isContinuousAlarm
  554. ) {
  555. this.alarmList.push(alarmOption);
  556. }
  557. this.$store.commit("setWarning", alarmOption);
  558. this.alarmList.sort((a, b) => {
  559. return b.lv - a.lv;
  560. });
  561. this.playAudioEffect();
  562. },
  563. reconnect(serveIP) {
  564. if (this.timeConnect < this.limitConnect) {
  565. console.log(`webSocket 连接失败,第 ${++this.timeConnect} 次重连`);
  566. setTimeout(() => {
  567. this.webSocketInit(serveIP);
  568. }, 2000);
  569. } else {
  570. console.log("webSocket 连接已超时");
  571. this.BASE.showMsg({
  572. showClose: true,
  573. duration: 0,
  574. msg: `webSocket 连接超时,实时报警获取失败`,
  575. });
  576. this.ws?.close();
  577. }
  578. },
  579. goToAlertDescPage({
  580. deviceId,
  581. alarmId,
  582. deviceType,
  583. alarmType,
  584. ts,
  585. stationId,
  586. }) {
  587. if (alarmType == "custom") {
  588. this.$router.push(
  589. `/safe/customWarning/${deviceId}/${alarmId}/${deviceType}`
  590. );
  591. } else if (alarmType == "booststation") {
  592. this.$router.push(
  593. `/safe/historywaring/${stationId}/${alarmId}/${deviceType}/${ts}`
  594. );
  595. } else {
  596. this.$router.push(
  597. `/safe/historywaring/${deviceId}/${alarmId}/${deviceType}`
  598. );
  599. }
  600. },
  601. getConfigItem(lv) {
  602. return (
  603. this.alarmConfigArray.find((ele) => {
  604. return ele.alarmLevel === lv;
  605. }) || {}
  606. );
  607. },
  608. getAlarmBoxClass() {
  609. let classList = [];
  610. if (this.alarmList?.length) classList.push("notEmpty");
  611. if (this.isCollapse) classList.push("collapseAlarmBox");
  612. return classList.join(" ");
  613. },
  614. getAlarmBoxWidth() {
  615. const baseWIdth = 240;
  616. let widthStep = 0;
  617. for (let i = 0; i < this.$store.state.columnNumber; i++) {
  618. if (this.alarmList?.[i * this.$store.state.columnNumber]) {
  619. widthStep++;
  620. }
  621. }
  622. return widthStep * baseWIdth;
  623. },
  624. getAlarmConfig() {
  625. if (localStorage.getItem("alarmConfigArray")) {
  626. this.alarmConfigArray = JSON.parse(
  627. localStorage.getItem("alarmConfigArray")
  628. );
  629. } else {
  630. this.alarmConfigArray = [
  631. {
  632. id: "1",
  633. alarmLevel: 1,
  634. isAlart: false,
  635. isAlarmSound: false,
  636. isContinuousAlarm: false,
  637. },
  638. {
  639. id: "2",
  640. alarmLevel: 2,
  641. isAlart: false,
  642. isAlarmSound: false,
  643. isContinuousAlarm: false,
  644. },
  645. {
  646. id: "3",
  647. alarmLevel: 3,
  648. isAlart: true,
  649. isAlarmSound: false,
  650. isContinuousAlarm: false,
  651. },
  652. {
  653. id: "4",
  654. alarmLevel: 4,
  655. isAlart: true,
  656. isAlarmSound: true,
  657. isContinuousAlarm: false,
  658. },
  659. {
  660. id: "5",
  661. alarmLevel: 5,
  662. isAlart: true,
  663. isAlarmSound: true,
  664. isContinuousAlarm: true,
  665. },
  666. ];
  667. localStorage.setItem(
  668. "alarmConfigArray",
  669. JSON.stringify(this.alarmConfigArray)
  670. );
  671. }
  672. },
  673. },
  674. watch: {
  675. "$store.state.alarmResetFlg"() {
  676. // getAlartConfig()
  677. // .then((res) => {
  678. // this.alarmConfigArray = res.data;
  679. // })
  680. // .catch(() => {
  681. // this.BASE.showMsg({
  682. // msg: "报警配置获取失败,请重试",
  683. // });
  684. // });
  685. this.getAlarmConfig();
  686. },
  687. },
  688. };
  689. </script>
  690. <style lang="scss" scoped>
  691. .alarmBox {
  692. height: calc(100% - 180px);
  693. padding: 0 12px 15px 30px;
  694. position: absolute;
  695. right: 0;
  696. bottom: 0;
  697. z-index: 1000;
  698. display: flex;
  699. justify-content: flex-end;
  700. align-items: flex-end;
  701. pointer-events: none;
  702. overflow: hidden;
  703. .columnItem {
  704. width: 240px;
  705. height: 100%;
  706. display: flex;
  707. flex-direction: column-reverse;
  708. align-items: center;
  709. font-size: 12px;
  710. overflow-y: scroll;
  711. border-radius: 8px;
  712. margin-left: 4px;
  713. transition: 0.2s;
  714. .alarmItem {
  715. width: 100%;
  716. box-sizing: border-box;
  717. border-radius: 8px;
  718. border: 1px solid #ebeef5;
  719. background: #1890ff;
  720. margin-bottom: 4px;
  721. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
  722. color: #fff;
  723. pointer-events: auto;
  724. cursor: pointer;
  725. .alarmTitle {
  726. display: flex;
  727. justify-content: flex-start;
  728. align-content: center;
  729. width: calc(100% - 16px);
  730. overflow: hidden;
  731. text-overflow: ellipsis;
  732. white-space: nowrap;
  733. padding: 0 8px;
  734. margin-top: 8px;
  735. }
  736. .alarmContent {
  737. width: calc(100% - 16px);
  738. display: flex;
  739. justify-content: flex-start;
  740. align-items: flex-start;
  741. flex-wrap: wrap;
  742. margin-top: 4px;
  743. padding: 0 8px;
  744. .contentItem {
  745. width: 100%;
  746. display: flex;
  747. justify-content: flex-start;
  748. align-items: flex-start;
  749. margin-bottom: 2px;
  750. .alertDescCursor {
  751. width: 160px;
  752. cursor: pointer;
  753. transition: 0.2s;
  754. &:hover {
  755. color: var(--el-color-primary);
  756. text-decoration: underline;
  757. transition: 0.2s;
  758. }
  759. }
  760. &:last-child {
  761. margin-bottom: 0;
  762. }
  763. }
  764. }
  765. .btnBox {
  766. display: flex;
  767. width: 100%;
  768. justify-content: center;
  769. align-items: center;
  770. margin-top: 4px;
  771. border-top: 1px solid red;
  772. .btnItem {
  773. width: 50%;
  774. display: flex;
  775. justify-content: center;
  776. align-items: center;
  777. padding: 4px 0;
  778. .el-button {
  779. padding: 0;
  780. width: 45%;
  781. height: 20px;
  782. min-height: 20px;
  783. }
  784. &.lv1 .el-button,
  785. &.lv2 .el-button,
  786. &.lv3 .el-button {
  787. color: var(--el-color-info) !important;
  788. }
  789. &.lv4 .el-button {
  790. color: rgb(50, 65, 87) !important;
  791. }
  792. &.lv5 .el-button {
  793. color: #242f42;
  794. }
  795. &:first-child {
  796. border-right: 1px solid red;
  797. }
  798. }
  799. }
  800. &.lv5 {
  801. background: #fef0f0;
  802. border: 1px solid #242f42;
  803. color: #242f42;
  804. }
  805. &.lv4 {
  806. background: #f0f9eb;
  807. border: 1px solid rgb(50, 65, 87);
  808. color: rgb(50, 65, 87);
  809. }
  810. &.lv1,
  811. &.lv2,
  812. &.lv3 {
  813. background: #fdf6ec;
  814. border: 1px solid var(--el-color-info);
  815. color: var(--el-color-info);
  816. }
  817. .lv1BdColor,
  818. .lv2BdColor,
  819. .lv3BdColor {
  820. border-color: var(--el-color-info) !important;
  821. }
  822. .lv4BdColor {
  823. border-color: rgb(50, 65, 87) !important;
  824. }
  825. .lv5BdColor {
  826. border-color: #242f42 !important;
  827. }
  828. &.hidden {
  829. height: 0;
  830. padding: 0;
  831. margin-bottom: 0;
  832. border: 0;
  833. transition: 0.2s;
  834. overflow: hidden;
  835. }
  836. }
  837. &::-webkit-scrollbar {
  838. width: 0; /* 隐藏Webkit浏览器的滚动条宽度 */
  839. height: 0; /* 隐藏Webkit浏览器的滚动条高度 */
  840. }
  841. }
  842. .collapseBtn {
  843. right: 12px;
  844. bottom: 0;
  845. font-size: 14px;
  846. color: #909399;
  847. display: flex;
  848. justify-content: center;
  849. align-items: center;
  850. white-space: nowrap;
  851. position: absolute;
  852. pointer-events: auto;
  853. user-select: none;
  854. cursor: pointer;
  855. transition: 0.2s;
  856. }
  857. &.notEmpty:hover {
  858. // background: rgba(0, 0, 0, 0.12);
  859. // box-shadow: 0 0 12px rgba(0, 0, 0, 0.12);
  860. transition: 0.2s;
  861. }
  862. &.collapseAlarmBox {
  863. width: 0;
  864. height: 0;
  865. overflow: hidden;
  866. }
  867. }
  868. </style>
  869. <style lang="scss">
  870. .currentRequestErrorNotification {
  871. width: 100%;
  872. display: flex;
  873. flex-direction: column;
  874. justify-content: flex-start;
  875. align-items: flex-start;
  876. p {
  877. width: 100%;
  878. display: flex;
  879. justify-content: flex-start;
  880. align-items: center;
  881. flex-wrap: wrap;
  882. .errorTitle {
  883. width: 85px;
  884. }
  885. .errorDesc {
  886. width: calc(100% - 85px);
  887. }
  888. }
  889. }
  890. </style>