modeControl.vue 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741
  1. <template>
  2. <div class="body">
  3. <img class="logo" src="../../assets/img/logo.png" alt="" />
  4. <div class="control">
  5. <div :class="current === 0 ? 'smart_on' : 'smart'" @click="ChangeBar(0)">智能</div>
  6. <div :class="current === 1 ? 'recommend_on' : 'recommend'" @click="ChangeBar(1)" >推荐</div>
  7. <div :class="current === 2 ? 'manual_on' : 'manual'" @click="ChangeBar(2)" >手动</div>
  8. </div>
  9. <div style="display: flex; flex-direction: row; z-index: 2">
  10. <div class="showData">
  11. <div class="dataBox" @dblclick="dbClicks(
  12. $store.state.currentStation
  13. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  14. ?.healthIndex
  15. : showDate.healthIndex,
  16. '健康指数'
  17. )">
  18. <div class="dataTitle">健康指数</div>
  19. <div class="datas">
  20. {{$store.state.currentStation ? healthIndex.toFixed(0) : allHealthIndex.toFixed(0)}}%
  21. </div>
  22. </div>
  23. <div class="dataBox" @dblclick="dbClicks(
  24. $store.state.currentStation
  25. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  26. ?.resourceIndex
  27. : showDate.resourceIndex,
  28. '资源指数'
  29. )">
  30. <div class="dataTitle">资源指数</div>
  31. <div class="datas">
  32. {{
  33. $store.state.currentStation
  34. ? resourceIndex.toFixed(0)
  35. : allResourceIndex.toFixed(0)
  36. }}%
  37. </div>
  38. </div>
  39. </div>
  40. <div>
  41. <div id="mainEcharts" @dblclick="dbClick()" class="echarts"></div>
  42. <div class="powerContent">
  43. <div class="power" @dblclick="dbClicks(
  44. $store.state.currentStation
  45. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  46. ?.realTimePower
  47. : showDate.realTimePower,
  48. '实际功率'
  49. )">
  50. <div class="powerTitle">实际功率</div>
  51. <div class="powerDatas" style="background-color: rgba(75, 85, 174, 1)" >
  52. {{
  53. $store.state.currentStation
  54. ? showDate?.stationOverviewInfos[
  55. $store.state.currentStation
  56. ]?.realTimePower?.value.toFixed(2)
  57. : showDate.realTimePower?.value.toFixed(2)
  58. }}
  59. </div>
  60. </div>
  61. <div class="power1" @dblclick="
  62. dbClicks(
  63. $store.state.currentStation
  64. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  65. ?.theoreticalPower
  66. : showDate.theoreticalPower,
  67. '理论功率'
  68. ) "
  69. >
  70. <div class="powerTitle">理论功率</div>
  71. <div
  72. class="powerDatas"
  73. style="background-color: rgba(05, 187, 76, 1)"
  74. >
  75. {{
  76. $store.state.currentStation
  77. ? showDate?.stationOverviewInfos[
  78. $store.state.currentStation
  79. ]?.theoreticalPower?.value.toFixed(2)
  80. : showDate.theoreticalPower?.value.toFixed(2)
  81. }}
  82. </div>
  83. </div>
  84. <div
  85. class="power2"
  86. @dblclick="
  87. dbClicks(
  88. $store.state.currentStation
  89. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  90. ?.agcPowerSet
  91. : showDate.agcPowerSet,
  92. 'AGC有功设定'
  93. )
  94. "
  95. >
  96. <div class="powerTitle">AGC有功设定</div>
  97. <div
  98. class="powerDatas"
  99. style="background-color: rgba(186, 50, 55, 1)"
  100. >
  101. {{
  102. $store.state.currentStation
  103. ? showDate?.stationOverviewInfos[
  104. $store.state.currentStation
  105. ]?.agcPowerSet?.value.toFixed(2)
  106. : showDate.agcPowerSet?.value.toFixed(2)
  107. }}
  108. </div>
  109. </div>
  110. </div>
  111. </div>
  112. <div class="showData">
  113. <div
  114. class="dataBox-right"
  115. @dblclick="
  116. dbClicks(
  117. $store.state.currentStation
  118. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  119. ?.windEnergyRate
  120. : showDate.windEnergyRate,
  121. '风能利用率'
  122. )
  123. "
  124. >
  125. <div class="dataTitle">风能利用率</div>
  126. <div class="datas" v-if="showDate.windEnergyRate?.value <= 100">
  127. {{
  128. $store.state.currentStation
  129. ? showDate?.stationOverviewInfos[
  130. $store.state.currentStation
  131. ]?.windEnergyRate?.value.toFixed(2)
  132. : showDate.windEnergyRate?.value.toFixed(2)
  133. }}%
  134. </div>
  135. </div>
  136. <div
  137. class="dataBox-right"
  138. @dblclick="
  139. dbClicks(
  140. $store.state.currentStation
  141. ? showDate?.stationOverviewInfos[$store.state.currentStation]
  142. ?.curveFollowingRate
  143. : showDate.curveFollowingRate,
  144. '曲线跟随率'
  145. )
  146. "
  147. >
  148. <div class="dataTitle">曲线跟随率</div>
  149. <div class="datas" v-if="showDate.curveFollowingRate?.value <= 100">
  150. {{
  151. $store.state.currentStation
  152. ? showDate?.stationOverviewInfos[
  153. $store.state.currentStation
  154. ]?.curveFollowingRate?.value.toFixed(2)
  155. : showDate.curveFollowingRate?.value.toFixed(2)
  156. }}%
  157. </div>
  158. </div>
  159. </div>
  160. </div>
  161. </div>
  162. <DetailPages ref="detials" @closed="closed()" v-model="dialogVisible">
  163. </DetailPages>
  164. <Details
  165. @closed="closeds()"
  166. v-model="display"
  167. :partsName="partsName"
  168. echartsId="modelEcharts"
  169. :datas="modelDetails"
  170. @search-data="search"
  171. @original-data="originalData"
  172. ></Details>
  173. </template>
  174. <script>
  175. import BackgroundData from "utils/BackgroundData";
  176. import api from "api/index";
  177. import MessageBridge from "utils/MessageBridge";
  178. import * as echarts from "echarts";
  179. import DetailPages from "./detailPages.vue";
  180. import Details from "../basicDataDetails.vue";
  181. export default {
  182. props: {
  183. // current: {
  184. // type: Number,
  185. // },
  186. },
  187. components: {
  188. DetailPages,
  189. Details,
  190. },
  191. data() {
  192. return {
  193. current: 1,
  194. datas: {},
  195. list: {},
  196. showDate: {},
  197. winFlag: 0,
  198. curveFlag: 0,
  199. fieldFlag: 0,
  200. dialogVisible: false,
  201. display: false,
  202. modelDetails: [],
  203. partsName: "",
  204. modelData: {},
  205. interval: 60,
  206. allHealthIndex: 0,
  207. allResourceIndex: 0,
  208. healthIndex: 0,
  209. resourceIndex: 0,
  210. intervals: null,
  211. };
  212. },
  213. created() {
  214. this.initData();
  215. this.getEchartss();
  216. this.intervals = setInterval(this.getEchartss, 10000);
  217. },
  218. computed: {},
  219. mounted() {
  220. this.$nextTick(() => {
  221. if (document.getElementById("mainEcharts"), 'dark') {
  222. this.getEcharts();
  223. }
  224. });
  225. },
  226. methods: {
  227. initData: function () {
  228. let mb = MessageBridge.getInstance();
  229. let vss = [
  230. { key: "/topic/voice-control", action: this.windturbineMessage },
  231. ];
  232. mb.register(vss);
  233. },
  234. windturbineMessage(msg) {
  235. if (msg === "CLOSE") {
  236. this.dialogVisible = false;
  237. }
  238. if(msg === 'OPEN_PAGE_MANUAL'){
  239. this.ChangeBar(2)
  240. }
  241. if(msg === 'OPEN_PAGE_AUTOMATIC'){
  242. this.ChangeBar(0)
  243. }
  244. if(msg === 'OPEN_PAGE_RECOMMENDATION'){
  245. this.ChangeBar(1)
  246. }
  247. if(msg === 'OPEN_AGC-GS'){
  248. this.dbClick()
  249. }
  250. },
  251. getEchartss() {
  252. api.stationOverview().then((res) => {
  253. this.showDate = res.data;
  254. this.getEcharts();
  255. this.allHealthIndex =
  256. (this.showDate.healthIndex?.value / this.showDate.healthIndexMax) *
  257. 100;
  258. this.allResourceIndex =
  259. (this.showDate.resourceIndex?.value /
  260. this.showDate.resourceIndexMax) *
  261. 100;
  262. this.healthIndex = this.$store.state.currentStation
  263. ? (this.showDate.stationOverviewInfos[
  264. this.$store.state.currentStation
  265. ].healthIndex?.value /
  266. this.showDate.stationOverviewInfos[
  267. this.$store.state.currentStation
  268. ].healthIndexMax) *
  269. 100
  270. : 0;
  271. this.resourceIndex = this.$store.state.currentStation
  272. ? (this.showDate.stationOverviewInfos[
  273. this.$store.state.currentStation
  274. ].resourceIndex?.value /
  275. this.showDate.stationOverviewInfos[
  276. this.$store.state.currentStation
  277. ].resourceIndexMax) *
  278. 100
  279. : 0;
  280. });
  281. },
  282. getEcharts() {
  283. let chartDom = document.getElementById("mainEcharts");
  284. let myChart = echarts.init(chartDom);
  285. let option;
  286. option = {
  287. series: [
  288. {
  289. type: "gauge",
  290. max: 600,
  291. splitNumber: 8,
  292. radius: '70%',
  293. anchor: {
  294. show: true,
  295. showAbove: true,
  296. size: 9,
  297. width: 5,
  298. itemStyle: {
  299. color: "#FAC858",
  300. },
  301. },
  302. pointer: {
  303. icon: "",
  304. width: 3,
  305. length: "80%",
  306. offsetCenter: [0, "5%"],
  307. },
  308. progress: {
  309. show: true,
  310. overlap: true,
  311. roundCap: true,
  312. },
  313. axisLine: {
  314. lineStyle: {
  315. //仪表盘轴线相关配置。
  316. width: 2,
  317. color: [[1, "rgba(83, 92, 93, 0.5)"]],
  318. // color: [[1, "#fff"]],
  319. },
  320. },
  321. axisLabel: {
  322. textStyle: {//数字刻度样式
  323. color: '#fff',
  324. // fontSize: 12,
  325. }
  326. },
  327. splitLine: {
  328. //分隔线样式相关
  329. length: 0, //分割线的长度
  330. lineStyle: {
  331. width: 1,
  332. color: "#fff",
  333. },
  334. },
  335. data: [
  336. {
  337. value: 125.85,
  338. // name: '实际功率',
  339. itemStyle: {
  340. color: "rgba(75, 85, 174, 1)",
  341. },
  342. },
  343. {
  344. value: 137.63,
  345. // name: '理论功率',
  346. itemStyle: {
  347. color: "rgba(05, 187, 76, 1)",
  348. },
  349. },
  350. {
  351. value: 132.04,
  352. // name: 'AGC有功设定',
  353. itemStyle: {
  354. color: "rgba(186, 50, 55, 1)",
  355. },
  356. },
  357. ],
  358. title: false,
  359. detail: false,
  360. // title: {
  361. // fontSize: 12
  362. // },
  363. // detail: {
  364. // width: 20,
  365. // height: 7,
  366. // fontSize: 12,
  367. // color: '#fff',
  368. // backgroundColor: 'auto',
  369. // borderRadius: 3,
  370. // formatter: ''
  371. // }
  372. },
  373. ],
  374. };
  375. option.series[0].data[0].value = this.$store.state.currentStation
  376. ? Number(
  377. this.showDate?.stationOverviewInfos[
  378. this.$store.state.currentStation
  379. ]?.realTimePower?.value
  380. )
  381. : Number(this.showDate.realTimePower?.value);
  382. option.series[0].data[1].value = this.$store.state.currentStation
  383. ? Number(
  384. this.showDate?.stationOverviewInfos[
  385. this.$store.state.currentStation
  386. ]?.theoreticalPower?.value
  387. )
  388. : Number(this.showDate.theoreticalPower?.value);
  389. option.series[0].data[2].value = this.$store.state.currentStation
  390. ? Number(
  391. this.showDate?.stationOverviewInfos[
  392. this.$store.state.currentStation
  393. ]?.agcPowerSet?.value
  394. )
  395. : Number(this.showDate.agcPowerSet?.value);
  396. myChart.setOption(option, true);
  397. },
  398. ChangeBar(values) {
  399. let bd = BackgroundData.getInstance();
  400. this.$router.push(`/?current=${values}`);
  401. if (!bd.LoginUser) {
  402. this.$notify({
  403. title: "请登录",
  404. message: "切换模式需要先登录!",
  405. type: "warning",
  406. position: "bottom-right",
  407. offset: 60,
  408. duration: 3000,
  409. });
  410. return;
  411. }
  412. if (this.current !== values) {
  413. this.$store.commit("current", Number(values));
  414. if (values === 2) {
  415. this.current = values;
  416. this.$router.push(`/ManualPage?current=${values}`);
  417. } else if (values === 1) {
  418. this.$store.commit("currentStation", "");
  419. this.$store.commit("current", values);
  420. if (this.current === 0) {
  421. this.current = values;
  422. } else {
  423. this.current = values;
  424. this.$router.push(`/?current=${values}`);
  425. }
  426. } else if (values === 0) {
  427. this.$store.commit("currentStation", "");
  428. this.$store.commit("current", values);
  429. if (this.current === 2) {
  430. this.current = values;
  431. this.$router.push(`/?current=${values}`);
  432. } else {
  433. this.current = values;
  434. }
  435. }
  436. } else if (values === 2) {
  437. this.$router.push(`/ManualPage?current=${values}`);
  438. }
  439. },
  440. dbClick() {
  441. this.dialogVisible = true;
  442. this.$refs.detials.getDate();
  443. },
  444. dbClicks(data, partsName, timeValues) {
  445. this.modelData = data;
  446. let date = new Date();
  447. let endTs = timeValues
  448. ? timeValues[1] > date.getTime()
  449. ? date.getTime()
  450. : timeValues[1]
  451. : date.getTime();
  452. let startTs = timeValues ? timeValues[0] : endTs - 28800000;
  453. data.tag &&
  454. api
  455. .getPower({
  456. tagName: data.tag,
  457. startTs: startTs,
  458. endTs: endTs,
  459. interval: this.interval,
  460. })
  461. .then((res) => {
  462. if (res.data.length > 0) {
  463. this.partsName = partsName;
  464. this.display = true;
  465. this.modelDetails = res.data;
  466. } else {
  467. this.modelDetails = [];
  468. }
  469. });
  470. },
  471. original(data, partsName, timeValues) {
  472. this.modelData = data;
  473. let date = new Date();
  474. let endTs = timeValues
  475. ? timeValues[1] > date.getTime()
  476. ? date.getTime()
  477. : timeValues[1]
  478. : date.getTime();
  479. let startTs = timeValues ? timeValues[0] : endTs - 28800000;
  480. api
  481. .getOriginalPower({
  482. tagName: data.tag,
  483. startTs: startTs,
  484. endTs: endTs,
  485. })
  486. .then((res) => {
  487. if (res.data.length > 0) {
  488. this.partsName = partsName;
  489. this.display = true;
  490. this.modelDetails = res.data;
  491. } else {
  492. this.modelDetails = [];
  493. }
  494. });
  495. },
  496. search(values, interval) {
  497. this.interval = interval;
  498. this.dbClicks(this.modelData, this.partsName, values);
  499. },
  500. originalData(values) {
  501. this.original(this.modelData, this.partsName, values);
  502. },
  503. closed() {
  504. this.dialogVisible = false;
  505. },
  506. closeds() {
  507. this.display = false;
  508. this.interval = 60;
  509. },
  510. },
  511. unmounted() {
  512. clearInterval(this.intervals);
  513. this.intervals = null;
  514. },
  515. watch: {
  516. "$store.getters.current": {
  517. handler: function (json) {
  518. this.current = json;
  519. },
  520. },
  521. },
  522. };
  523. </script>
  524. <style scoped>
  525. .body {
  526. width: 100%;
  527. height: 28.5vh;
  528. /* background-color: #ffffff; */
  529. margin-left: 15px;
  530. margin-top: 20px;
  531. border: 1px solid #373737;
  532. /* border-left: 1px solid #373737;
  533. border-right: 1px solid #373737;
  534. border-bottom: 1px solid #373737; */
  535. /* background-image: url('../../assets/img/type/background.png'); */
  536. background-repeat: no-repeat;
  537. background-position: center;
  538. background-size: cover;
  539. }
  540. .logo {
  541. position: absolute;
  542. top: 8px;
  543. left: 12px;
  544. }
  545. .control {
  546. display: flex;
  547. flex-direction: row-reverse;
  548. align-items: center;
  549. font-size: 14px;
  550. color: #ffffff;
  551. /* margin-right: 5px; */
  552. position: absolute;
  553. top: 30px;
  554. right: 10px;
  555. z-index: 99;
  556. }
  557. .manual {
  558. display: flex;
  559. align-items: center;
  560. justify-content: center;
  561. height: 29px;
  562. width: 86px;
  563. border-left: 1px solid rgba(51, 51, 51, 1);
  564. border-top: 1px solid rgba(51, 51, 51, 1);
  565. border-bottom: 1px solid rgba(51, 51, 51, 1);
  566. border-top-left-radius: 15px;
  567. border-bottom-left-radius: 15px;
  568. }
  569. .manual_on {
  570. display: flex;
  571. align-items: center;
  572. justify-content: center;
  573. height: 29px;
  574. width: 86px;
  575. border-left: 1px solid rgba(37, 116, 219, 1);
  576. border-top: 1px solid rgba(37, 116, 219, 1);
  577. border-bottom: 1px solid rgba(37, 116, 219, 1);
  578. border-top-left-radius: 15px;
  579. border-bottom-left-radius: 15px;
  580. background-color: rgba(37, 116, 219, 1);
  581. }
  582. .recommend {
  583. display: flex;
  584. align-items: center;
  585. justify-content: center;
  586. height: 29px;
  587. width: 86px;
  588. border-top: 1px solid rgba(51, 51, 51, 1);
  589. border-bottom: 1px solid rgba(51, 51, 51, 1);
  590. }
  591. .recommend_on {
  592. display: flex;
  593. align-items: center;
  594. justify-content: center;
  595. height: 29px;
  596. width: 86px;
  597. border-top: 1px solid rgba(37, 116, 219, 1);
  598. border-bottom: 1px solid rgba(37, 116, 219, 1);
  599. background-color: rgba(37, 116, 219, 1);
  600. }
  601. .smart {
  602. display: flex;
  603. align-items: center;
  604. justify-content: center;
  605. height: 29px;
  606. width: 86px;
  607. border-right: 1px solid rgba(51, 51, 51, 1);
  608. border-top: 1px solid rgba(51, 51, 51, 1);
  609. border-bottom: 1px solid rgba(51, 51, 51, 1);
  610. border-top-right-radius: 15px;
  611. border-bottom-right-radius: 15px;
  612. }
  613. .smart_on {
  614. display: flex;
  615. align-items: center;
  616. justify-content: center;
  617. height: 29px;
  618. width: 86px;
  619. border-right: 1px solid rgba(37, 116, 219, 1);
  620. border-top: 1px solid rgba(37, 116, 219, 1);
  621. border-bottom: 1px solid rgba(37, 116, 219, 1);
  622. border-top-right-radius: 15px;
  623. border-bottom-right-radius: 15px;
  624. background-color: rgba(37, 116, 219, 1);
  625. }
  626. .echarts {
  627. width: 300px;
  628. height: 280px;
  629. margin-top: 10px;
  630. /* background-color: #000000; */
  631. margin-left: 10px;
  632. }
  633. .showData {
  634. display: flex;
  635. flex-direction: column;
  636. align-items: center;
  637. justify-content: center;
  638. }
  639. .dataBox {
  640. display: flex;
  641. flex-direction: column;
  642. width: 120px;
  643. height: 60px;
  644. border: 1px solid rgba(83, 92, 93, 1);
  645. margin-bottom: 15px;
  646. margin-top: 15px;
  647. margin-left: 30px;
  648. }
  649. .dataBox-right {
  650. display: flex;
  651. flex-direction: column;
  652. width: 120px;
  653. height: 60px;
  654. border: 1px solid rgba(83, 92, 93, 1);
  655. margin-bottom: 15px;
  656. margin-top: 15px;
  657. margin-right: 30px;
  658. }
  659. .dataTitle {
  660. height: 50%;
  661. display: flex;
  662. flex-direction: row;
  663. align-items: center;
  664. font-size: 12px;
  665. color: rgba(153, 162, 163, 1);
  666. margin-left: 5px;
  667. }
  668. .datas {
  669. height: 50%;
  670. display: flex;
  671. flex-direction: row-reverse;
  672. align-items: center;
  673. font-size: 16px;
  674. color: rgba(05, 187, 76, 1);
  675. margin-right: 5px;
  676. }
  677. .powerContent {
  678. display: flex;
  679. flex-direction: row;
  680. align-items: center;
  681. color: #ffffff;
  682. position: relative;
  683. bottom: 15%;
  684. left: 11%;
  685. }
  686. .power {
  687. display: flex;
  688. flex-direction: column;
  689. align-items: center;
  690. justify-content: center;
  691. font-size: 14px;
  692. padding: 5px 10px;
  693. }
  694. .power1 {
  695. display: flex;
  696. flex-direction: column;
  697. align-items: center;
  698. justify-content: center;
  699. font-size: 14px;
  700. padding: 5px 10px 5px 10px;
  701. }
  702. .power2 {
  703. display: flex;
  704. flex-direction: column;
  705. align-items: center;
  706. justify-content: center;
  707. font-size: 14px;
  708. padding: 5px 3px;
  709. }
  710. .powerTitle {
  711. margin-bottom: 3px;
  712. }
  713. .powerDatas {
  714. border-radius: 2px;
  715. display: flex;
  716. align-items: center;
  717. justify-content: center;
  718. font-size: 14px;
  719. font-weight: 600;
  720. padding: 2px 5px;
  721. }
  722. </style>