index.vue 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717
  1. <template>
  2. <div
  3. draggable="true"
  4. @dragstart="startDrag"
  5. @dragend="stopDrag"
  6. :style="{ right: x + 'px', bottom: y + 'px' }"
  7. style="cursor: pointer; position: absolute; z-index: 999"
  8. >
  9. <el-badge
  10. :value="warnLength"
  11. :class="{ active: warnLength && !displayAlarms ? false : true }"
  12. >
  13. <div class="alarmDeligo" @click="displayAlarm" v-if="warnLength">
  14. <img src="@assets/imgs/ygj.png" class="trans" />
  15. <img src="@assets/imgs/ygj1.png" />
  16. </div>
  17. <div
  18. class="alarmDeligo"
  19. @dblclick="setting"
  20. v-if="!warnLength"
  21. title="双击打开配置"
  22. >
  23. <img src="@assets/imgs/wgj.png" class="trans" />
  24. <img src="@assets/imgs/wgj1.png" />
  25. </div>
  26. </el-badge>
  27. <deligo-alarm-list
  28. :dialogList="warnList.slice(0, 12)"
  29. @setConfig="displaySetting = true"
  30. @confirmed="handleConfirm"
  31. ref="deligoChild"
  32. @close="displayClose"
  33. v-if="displayAlarms == true"
  34. ></deligo-alarm-list>
  35. <alarmSetting
  36. v-if="displaySetting"
  37. @saveConfig="getAlarmConfig"
  38. @cancleConfig="displaySetting = false"
  39. :alarmConfigArray="alarmConfigArray"
  40. />
  41. </div>
  42. </template>
  43. <script>
  44. import deligoAlarmList from "@/components/deligoAlarmList";
  45. import alarmSetting from "@/components/alarm-badge/alarm-setting.vue";
  46. import { ElNotification } from "element-plus";
  47. import { alarm_history } from "@/api/zhbj/index.js";
  48. import dayjs from "dayjs";
  49. export default {
  50. name: "alarmBadge",
  51. components: {
  52. deligoAlarmList,
  53. alarmSetting,
  54. },
  55. data() {
  56. return {
  57. //实时报警弹窗ws
  58. socketObj1: "", // websocket实例对象
  59. //心跳检测
  60. heartCheck1: {
  61. vueThis1: this, // vue实例
  62. timeout1: 30000, // 超时时间
  63. timeoutObj1: null, // 计时器对象——向后端发送心跳检测
  64. serverTimeoutObj1: null, // 计时器对象——等待后端心跳检测的回复
  65. // 心跳检测重置
  66. reset: function () {
  67. clearTimeout(this.timeoutObj1);
  68. clearTimeout(this.serverTimeoutObj1);
  69. return this;
  70. },
  71. // 心跳检测启动
  72. start: function () {
  73. this.timeoutObj1 && clearTimeout(this.timeoutObj1);
  74. this.serverTimeoutObj1 && clearTimeout(this.serverTimeoutObj1);
  75. this.timeoutObj1 = setTimeout(() => {
  76. // 这里向后端发送一个心跳检测,后端收到后,会返回一个心跳回复
  77. this.vueThis1.socketObj1.send("HeartBeat");
  78. console.log("发送心跳检测1");
  79. this.serverTimeoutObj1 = setTimeout(() => {
  80. // 如果超过一定时间还没重置计时器,说明websocket与后端断开了
  81. console.log("未收到心跳检测回复1");
  82. // 关闭WebSocket
  83. this.vueThis1.socketObj1.close();
  84. }, this.timeout1);
  85. }, this.timeout1);
  86. },
  87. },
  88. socketReconnectTimer1: null, // 计时器对象——重连
  89. socketReconnectLock1: false, // WebSocket重连的锁
  90. socketLeaveFlag1: false, // 离开标记(解决 退出登录再登录 时出现的 多次相同推送 问题,出现的本质是多次建立了WebSocket连接)
  91. alarmList: [],
  92. dialogList: [],
  93. realList: [],
  94. //实时报警图标相关
  95. x: null,
  96. y: null,
  97. currentX: 0,
  98. currentY: 0,
  99. displayAlarms: false,
  100. indexNum: 0,
  101. data1: 0,
  102. dragging: false,
  103. initialX: 0,
  104. initialY: 0,
  105. timer: null,
  106. firstTime: null,
  107. lastTime: null,
  108. key: false,
  109. //请求参数
  110. requestAlarmHistoryParams: [
  111. {
  112. alarmType: "booststation",
  113. deviceType: "",
  114. },
  115. {
  116. alarmType: "inverter",
  117. deviceType: "",
  118. },
  119. {
  120. alarmType: "windturbine",
  121. deviceType: "",
  122. },
  123. {
  124. alarmType: "custom",
  125. deviceType: "inverter",
  126. },
  127. {
  128. alarmType: "custom",
  129. deviceType: "windturbine",
  130. },
  131. ],
  132. alarmConfigArray: [],
  133. displaySetting: false,
  134. seriousWarning: false,
  135. audioElement: null,
  136. };
  137. },
  138. computed: {
  139. warnLength() {
  140. return this.$store.state.warnList.length;
  141. },
  142. warnList() {
  143. return this.$store.state.warnList;
  144. },
  145. isLogined() {
  146. return this.$store.state.user?.loginState;
  147. },
  148. },
  149. created() {
  150. this.getAlarmConfig();
  151. this.x = 80;
  152. this.y = 80;
  153. let requestResult = [];
  154. this.requestAlarmHistoryParams.forEach(({ alarmType, deviceType }) => {
  155. requestResult.push(this.getAlarmHistory(alarmType, deviceType));
  156. });
  157. Promise.all(requestResult)
  158. .then((promiseResult) => {
  159. promiseResult.forEach(({ data }) => {
  160. data?.ls?.forEach((ele) => {
  161. this.pushALarmItem(ele);
  162. });
  163. });
  164. this.dialogList.sort((a, b) => {
  165. return b.ts - a.ts;
  166. });
  167. this.realList.sort((a, b) => {
  168. return b.ts - a.ts;
  169. });
  170. this.$store.commit("changeAlarmlist", this.alarmList);
  171. this.$store.commit("setWarning", this.dialogList);
  172. this.$store.commit("setWarningList", this.realList);
  173. if (!this.socketLeaveFlag1) {
  174. // 没有离开——重连
  175. // websocket重连
  176. this.socketReconnect1();
  177. }
  178. })
  179. .catch(() => {
  180. requestResult.forEach((ele, index) => {
  181. ele
  182. .then(({ data }) => {
  183. data?.ls?.forEach((ele) => {
  184. this.pushALarmItem(ele);
  185. });
  186. })
  187. .catch((error) => {
  188. ElNotification({
  189. type: "error",
  190. title: "查询历史未处理报警请求出错!",
  191. dangerouslyUseHTMLString: true,
  192. message: `<div class="currentRequestErrorNotification">
  193. <p><span>主要参数:</p>
  194. <p style="color:var(--el-color-primary)"><span class="errorTitle">alarmType:</span><span class="errorDesc">"${this.requestAlarmHistoryParams[index].alarmType}"</span></p>
  195. <p style="color:var(--el-color-primary)"><span class="errorTitle">deviceType:</span><span class="errorDesc">"${this.requestAlarmHistoryParams[index].deviceType}"</span></p>
  196. <p style="color:var(--el-color-danger)"><span class="errorTitle">错误正文:</span><span class="errorDesc">${error}</span></p>
  197. </div>`,
  198. });
  199. });
  200. });
  201. });
  202. },
  203. mounted() {},
  204. unmounted() {
  205. console.log("离开标记", this.socketLeaveFlag);
  206. },
  207. methods: {
  208. setting() {
  209. this.displaySetting = true;
  210. },
  211. handleConfirm(flag) {
  212. this.playAudioEffect();
  213. if (flag) {
  214. this.dialogList = [];
  215. this.$store.commit("setWarning", this.dialogList);
  216. setTimeout(() => {
  217. this.$refs.deligoChild && this.$refs.deligoChild.init(this.warnList);
  218. }, 0);
  219. }
  220. },
  221. playAudioEffect() {
  222. const lv1Config = this.getConfigItem(1);
  223. let lv1Play = false;
  224. if (lv1Config.isAlarmSound) {
  225. lv1Play = this.dialogList.some((ele) => {
  226. return ele.lv === 1 && !ele.confirm;
  227. });
  228. }
  229. const lv2Config = this.getConfigItem(2);
  230. let lv2Play = false;
  231. if (lv2Config.isAlarmSound) {
  232. lv2Play = this.dialogList.some((ele) => {
  233. return ele.lv === 2 && !ele.confirm;
  234. });
  235. }
  236. const lv3Config = this.getConfigItem(3);
  237. let lv3Play = false;
  238. if (lv3Config.isAlarmSound) {
  239. lv3Play = this.dialogList.some((ele) => {
  240. return ele.lv === 3 && !ele.confirm;
  241. });
  242. }
  243. const lv4Config = this.getConfigItem(4);
  244. let lv4Play = false;
  245. if (lv4Config.isAlarmSound) {
  246. lv4Play = this.dialogList.some((ele) => {
  247. return ele.lv === 4 && !ele.confirm;
  248. });
  249. }
  250. const lv5Config = this.getConfigItem(5);
  251. let lv5Play = false;
  252. if (lv5Config.isAlarmSound) {
  253. lv5Play = this.dialogList.some((ele) => {
  254. return ele.lv === 5 && !ele.confirm;
  255. });
  256. }
  257. // console.log(lv1Play, lv2Play, lv3Play, lv4Play, lv5Play);
  258. if (lv5Play && !this.seriousWarning) {
  259. this.seriousWarning = true;
  260. this.audioElement = new Audio();
  261. this.audioElement.src = "./static/sound/lv5.mp3";
  262. this.audioElement.loop = true;
  263. this.audioElement?.play();
  264. } else if (
  265. (lv1Play || lv2Play || lv3Play || lv4Play) &&
  266. !this.seriousWarning
  267. ) {
  268. this.audioElement = new Audio();
  269. this.audioElement.src = "./static/sound/lv4.mp3";
  270. this.audioElement.addEventListener("ended", () => {
  271. this.audioElement?.removeEventListener(
  272. "ended",
  273. this.stopPlayAudioEffect
  274. );
  275. });
  276. this.audioElement?.play();
  277. } else {
  278. if (!this.seriousWarning) {
  279. this.stopPlayAudioEffect();
  280. }
  281. }
  282. },
  283. stopPlayAudioEffect() {
  284. this.seriousWarning = false;
  285. if (this.audioElement) {
  286. this.audioElement.pause();
  287. this.audioElement.currentTime = 0;
  288. this.audioElement.loop = false;
  289. }
  290. this.audioElement = null;
  291. },
  292. //获取报警配置
  293. getAlarmConfig() {
  294. if (localStorage.getItem("alarmConfigArray")) {
  295. this.alarmConfigArray = JSON.parse(
  296. localStorage.getItem("alarmConfigArray")
  297. );
  298. } else {
  299. this.alarmConfigArray = [
  300. {
  301. id: "1",
  302. alarmLevel: 1,
  303. isAlart: false,
  304. isAlarmSound: false,
  305. isContinuousAlarm: false,
  306. },
  307. {
  308. id: "2",
  309. alarmLevel: 2,
  310. isAlart: false,
  311. isAlarmSound: false,
  312. isContinuousAlarm: false,
  313. },
  314. {
  315. id: "3",
  316. alarmLevel: 3,
  317. isAlart: false,
  318. isAlarmSound: false,
  319. isContinuousAlarm: false,
  320. },
  321. {
  322. id: "4",
  323. alarmLevel: 4,
  324. isAlart: true,
  325. isAlarmSound: true,
  326. isContinuousAlarm: false,
  327. },
  328. {
  329. id: "5",
  330. alarmLevel: 5,
  331. isAlart: true,
  332. isAlarmSound: true,
  333. isContinuousAlarm: true,
  334. },
  335. ];
  336. localStorage.setItem(
  337. "alarmConfigArray",
  338. JSON.stringify(this.alarmConfigArray)
  339. );
  340. }
  341. },
  342. getAlarmName(alarmItem) {
  343. let alarmName = "";
  344. if (alarmItem.deviceType === "booststation") {
  345. alarmName = "升压站报警";
  346. } else if (alarmItem.deviceType === "inverter") {
  347. alarmName = "光伏报警";
  348. } else if (alarmItem.deviceType === "windturbine") {
  349. alarmName = "设备报警";
  350. } else if (alarmItem.deviceType === "station") {
  351. alarmName = "场站";
  352. }
  353. if (alarmItem.alarmType === "custom") {
  354. alarmName = "自定义报警";
  355. }
  356. return alarmName;
  357. },
  358. getLvName(alarmItem) {
  359. if (alarmItem.rank === 1) {
  360. return "低级";
  361. } else if (alarmItem.rank === 2) {
  362. return "低中级";
  363. } else if (alarmItem.rank === 3) {
  364. return "中级";
  365. } else if (alarmItem.rank === 4) {
  366. return "中高级";
  367. } else if (alarmItem.rank === 5) {
  368. return "高级";
  369. }
  370. },
  371. //查历史报警
  372. getAlarmHistory(alarmType, deviceType) {
  373. let params = {
  374. pageNum: 1,
  375. pageSize: 50,
  376. alarmId: "",
  377. alarmType,
  378. deviceType,
  379. stationid: "",
  380. deviceid: "",
  381. modelId: "",
  382. components: "",
  383. description: "",
  384. isclose: false,
  385. begin: dayjs().add(-1, "hour").format("YYYY-MM-DD HH:mm:ss"),
  386. end: dayjs().format("YYYY-MM-DD HH:mm:ss"),
  387. };
  388. if (params.alarmType == "windturbine") {
  389. // params.stationid = "SXJ_KGDL_DJY_FDC_STA";
  390. params.stationid = ""
  391. } else if (params.alarmType == "inverter") {
  392. params.stationid = "SXJ_KGDL_JR_GDC_STA";
  393. }
  394. return alarm_history(params, 12000);
  395. },
  396. pushALarmItem(alarmItem, type) {
  397. const configItem = this.getConfigItem(alarmItem.rank);
  398. const alarmOption = {
  399. id: alarmItem.id ? alarmItem.id : alarmItem.tbname,
  400. lv: alarmItem.rank,
  401. modelId: alarmItem.modelId,
  402. lvName: this.getLvName(alarmItem),
  403. rank: alarmItem.rank,
  404. confirmed: alarmItem.confirmed,
  405. class: `animate__bounceInRight lv${alarmItem.rank}`,
  406. deviceId: alarmItem.deviceId,
  407. faultCause: alarmItem.faultCause,
  408. resolvent: alarmItem.resolvent,
  409. characteristic: alarmItem.characteristic,
  410. code: alarmItem.code,
  411. wpName: alarmItem.stationName
  412. ? alarmItem.stationName
  413. : alarmItem.wpName,
  414. stationId: alarmItem.stationId ? alarmItem.stationId : alarmItem.wpId,
  415. isClose: alarmItem.closeTime ? true : alarmItem.endts ? true : false,
  416. isCloseName: alarmItem.closeTime
  417. ? "已解除"
  418. : alarmItem.endts
  419. ? "已解除"
  420. : "未解除",
  421. alarmId: alarmItem.alarmId,
  422. alarmType: alarmItem.alarmType,
  423. alarmName: this.getAlarmName(alarmItem),
  424. description: alarmItem.description,
  425. deviceType: alarmItem.deviceType,
  426. oval: alarmItem.oval,
  427. triggerType: alarmItem.triggerType,
  428. ts: alarmItem.ts
  429. ? dayjs(alarmItem.ts).valueOf()
  430. : dayjs(alarmItem.updateTime).valueOf(),
  431. endts: alarmItem.endts
  432. ? dayjs(alarmItem.endts).format("YYYY-MM-DD HH:mm:ss")
  433. : null,
  434. closeTime: alarmItem.closeTime
  435. ? dayjs(alarmItem.closeTime).format("YYYY-MM-DD HH:mm:ss")
  436. : null,
  437. deviceName: alarmItem.deviceName
  438. ? alarmItem.deviceName
  439. : alarmItem.code,
  440. tsName: alarmItem.ts
  441. ? new Date(alarmItem.ts).formatDate("MM-dd hh:mm:ss")
  442. : new Date(alarmItem.updateTime).formatDate("MM-dd hh:mm:ss"),
  443. fullTsName: alarmItem.ts
  444. ? new Date(alarmItem.ts).formatDate("yyyy-MM-dd hh:mm:ss")
  445. : new Date(alarmItem.updateTime).formatDate("yyyy-MM-dd hh:mm:ss"),
  446. };
  447. if (
  448. alarmOption.alarmType == "booststation" &&
  449. alarmOption.deviceType != "custom"
  450. ) {
  451. if (
  452. configItem.isAlarmSound ||
  453. configItem.isAlart ||
  454. configItem.isContinuousAlarm
  455. ) {
  456. let a = {};
  457. a[`${alarmOption.stationId}`] =
  458. alarmOption.closeTime || alarmOption.confirmed ? false : true;
  459. this.alarmList.push(a);
  460. this.alarmList = [
  461. ...new Set(this.alarmList.map((t) => JSON.stringify(t))),
  462. ].map((s) => JSON.parse(s));
  463. }
  464. }
  465. if (
  466. configItem.isAlarmSound ||
  467. configItem.isAlart ||
  468. configItem.isContinuousAlarm
  469. ) {
  470. if (type && type == "ws") {
  471. this.dialogList.unshift(alarmOption);
  472. } else {
  473. this.dialogList.push(alarmOption);
  474. }
  475. }
  476. if (type && type == "ws") {
  477. this.realList.unshift(alarmOption);
  478. } else {
  479. this.realList.push(alarmOption);
  480. }
  481. // && alarmOption.deviceType != "custom"
  482. this.playAudioEffect();
  483. },
  484. getConfigItem(lv) {
  485. return (
  486. this.alarmConfigArray.find((ele) => {
  487. return ele.alarmLevel === lv;
  488. }) || {}
  489. );
  490. },
  491. // 拖拽相关
  492. startDrag(event) {
  493. this.currentX = event.clientX;
  494. this.currentY = event.clientY;
  495. },
  496. stopDrag(event) {
  497. let x = event.clientX - this.currentX;
  498. let y = event.clientY - this.currentY;
  499. if (event.clientX <= 80 || event.clientX >= window.innerWidth - 40) {
  500. this.x = this.x;
  501. } else {
  502. this.x -= x;
  503. }
  504. if (event.clientY <= 70 || event.clientY >= window.innerHeight - 60) {
  505. this.y = this.y;
  506. } else {
  507. this.y -= y;
  508. }
  509. },
  510. //开关列表
  511. displayAlarm(val) {
  512. this.displayAlarms = !this.displayAlarms;
  513. if (this.displayAlarms == true) {
  514. setTimeout(() => {
  515. this.$refs.deligoChild && this.$refs.deligoChild.init(this.warnList);
  516. }, 0);
  517. }
  518. },
  519. displayClose() {
  520. this.displayAlarms = false;
  521. },
  522. // websocket启动
  523. createWebSocket1() {
  524. // let webSocketLink = `ws://192.168.1.102:6014/websocket/${this.$store.state.user.userId}_${this.$store.state.user.authToken}`;
  525. let webSocketLink1 = `${process.env.VUE_APP_WS_URL}/websocket/nxf_${this.$store.state.user.authToken}`; // webSocket地址
  526. try {
  527. if ("WebSocket" in window) {
  528. this.socketObj1 = new WebSocket(webSocketLink1);
  529. }
  530. // websocket事件绑定
  531. this.socketEventBind1();
  532. } catch (e) {
  533. console.log("catch" + e);
  534. // websocket重连
  535. this.socketReconnect1();
  536. }
  537. },
  538. // websocket事件绑定
  539. socketEventBind1() {
  540. // 连接成功建立的回调
  541. this.socketObj1.onopen = this.onopenCallback1;
  542. // 连接发生错误的回调
  543. this.socketObj1.onerror = this.onerrorCallback1;
  544. // 连接关闭的回调
  545. this.socketObj1.onclose = this.oncloseCallback1;
  546. // 向后端发送数据的回调
  547. this.socketObj1.onsend = this.onsendCallback1;
  548. // 接收到消息的回调
  549. this.socketObj1.onmessage = this.getMessageCallback1;
  550. //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
  551. window.onbeforeunload = () => {
  552. this.socketObj1.close();
  553. };
  554. },
  555. // websocket重连
  556. socketReconnect1() {
  557. if (this.socketReconnectLock1) {
  558. return;
  559. }
  560. this.socketReconnectLock1 = true;
  561. this.socketReconnectTimer1 && clearTimeout(this.socketReconnectTimer1);
  562. this.socketReconnectTimer1 = setTimeout(() => {
  563. console.log("WebSocket1:重连中...");
  564. this.socketReconnectLock1 = false;
  565. // websocket启动
  566. this.createWebSocket1();
  567. }, 4000);
  568. },
  569. // 连接成功建立的回调
  570. onopenCallback1: function (event) {
  571. console.log("WebSocket1:已连接");
  572. // 心跳检测重置
  573. this.heartCheck1.reset().start();
  574. },
  575. // 连接发生错误的回调
  576. onerrorCallback1: function (event) {
  577. console.log("WebSocket1:发生错误");
  578. // websocket重连
  579. this.socketReconnect1();
  580. },
  581. // 连接关闭的回调
  582. oncloseCallback1: function (event) {
  583. console.log("WebSocket1:已关闭");
  584. // 心跳检测重置
  585. this.heartCheck1.reset();
  586. if (!this.socketLeaveFlag1) {
  587. // 没有离开——重连
  588. // websocket重连
  589. this.socketReconnect1();
  590. }
  591. },
  592. // 向后端发送数据的回调
  593. onsendCallback1: function () {
  594. console.log("WebSocket1:发送信息给后端");
  595. },
  596. // 接收到消息的回调
  597. getMessageCallback1: function (msg) {
  598. console.log("msg11=====>>>", msg);
  599. if (Object.keys(msg) && msg.data == "ok") {
  600. // 心跳回复——心跳检测重置
  601. // 收到心跳检测回复就说明连接正常
  602. console.log("收到心跳检测回复1");
  603. // 心跳检测重置
  604. this.heartCheck1.reset().start();
  605. } else {
  606. // 普通推送——正常处理
  607. let data = JSON.parse(msg.data);
  608. // 相关处理
  609. if (data) {
  610. // console.log("收到推送消息1", data);
  611. this.pushALarmItem(data, "ws");
  612. // this.dialogList.sort((a, b) => {
  613. // return b.lv - a.lv;
  614. // });
  615. this.$store.commit("changeAlarmlist", this.alarmList);
  616. this.$store.commit("setWarning", this.dialogList);
  617. this.$store.commit("setWarningList", this.realList);
  618. }
  619. }
  620. },
  621. },
  622. watch: {
  623. isLogined: {
  624. handler(res) {
  625. if (!res && this.socketObj1) {
  626. // 离开标记
  627. this.socketLeaveFlag1 = true;
  628. // 关闭WebSocket
  629. this.socketObj1.close();
  630. }
  631. },
  632. immediate: true,
  633. },
  634. },
  635. };
  636. </script>
  637. <style lang="less" scoped>
  638. .alarmDeligo {
  639. z-index: 2003;
  640. position: relative;
  641. img {
  642. width: 43px;
  643. max-width: 43px;
  644. height: 43px;
  645. z-index: 3;
  646. }
  647. img:first-child {
  648. width: 53px;
  649. height: 57px;
  650. max-width: 53px;
  651. top: -7px;
  652. left: -5px;
  653. position: absolute;
  654. z-index: 3;
  655. }
  656. }
  657. .el-badge {
  658. &.active {
  659. .el-badge__content--danger {
  660. opacity: 0 !important;
  661. }
  662. }
  663. }
  664. .el-badge__content--danger {
  665. background: #ff4e00 !important;
  666. border: none !important;
  667. }
  668. .el-badge__content.is-fixed {
  669. top: 7px !important;
  670. right: 14px !important;
  671. z-index: 2003 !important;
  672. }
  673. .trans {
  674. animation: circleProgess 3s linear infinite;
  675. }
  676. @keyframes circleProgess {
  677. 0% {
  678. transform: rotateZ(360deg);
  679. }
  680. 25% {
  681. transform: rotateZ(270deg);
  682. }
  683. 50% {
  684. transform: rotateZ(180deg);
  685. }
  686. 75% {
  687. transform: rotateZ(90deg);
  688. }
  689. 100% {
  690. transform: rotateZ(0deg);
  691. }
  692. }
  693. </style>