index.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707
  1. <template>
  2. <div class="singleMachine">
  3. <div class="singleMachine_top">
  4. <el-select
  5. size="mini"
  6. v-model="companyVal"
  7. placeholder="请选择"
  8. @change="changeCompan"
  9. >
  10. <el-option
  11. v-for="item in companyOptions"
  12. :key="item.value"
  13. :label="item.label"
  14. :value="item.value"
  15. >
  16. </el-option>
  17. </el-select>
  18. <div class="tabCut">
  19. <div
  20. @click="changeBtn(val.id)"
  21. :class="tabIndex === val.id ? 'active' : ''"
  22. v-for="val in tabOptions"
  23. :key="val.id"
  24. >
  25. <span>{{ val.name }}</span>
  26. </div>
  27. </div>
  28. <div class="station">
  29. 场站:
  30. <el-select
  31. size="mini"
  32. v-model="stationVal"
  33. placeholder="请选择"
  34. @change="changeStation"
  35. clearable
  36. >
  37. <el-option
  38. v-for="item in stationOptions"
  39. :key="item.value"
  40. :label="item.label"
  41. :value="item.value"
  42. >
  43. </el-option>
  44. </el-select>
  45. </div>
  46. <div class="station">
  47. 时间
  48. <div class="search-input">
  49. <el-date-picker
  50. v-model="pickerTimer"
  51. type="month"
  52. value-format="YYYY-MM"
  53. placeholder="选择日期"
  54. popper-class="date-select"
  55. >
  56. </el-date-picker>
  57. </div>
  58. </div>
  59. <div class="but">
  60. <el-button round size="mini" class="buttons" @click="seachData"
  61. >搜索</el-button
  62. >
  63. <el-button round size="mini" class="buttons" @click="downXlsxFn"
  64. >导出</el-button
  65. >
  66. </div>
  67. </div>
  68. <div style="background: rgba(0, 0, 0, 0.45)">
  69. <div class="singleMachine_title clearfix">
  70. <div class="leftContent floatLeft"><span>{{}}单机性能分析表</span></div>
  71. <!-- -->
  72. </div>
  73. <div class="singleMachine_Table">
  74. <el-table
  75. :data="stationDatas"
  76. ref="stand_table"
  77. style="width: 100%"
  78. empty-text="暂无数据"
  79. height="81vh"
  80. stripe
  81. :row-class-name="tableRowClassName"
  82. :header-cell-style="headerStyle"
  83. >
  84. <el-table-column
  85. v-for="(item, index) in stationData"
  86. :key="index"
  87. :prop="item.tagCode"
  88. align="center"
  89. :width="item.width"
  90. header-align="center"
  91. class-name="add"
  92. style="width: 100%"
  93. empty-text="暂无数据"
  94. height="81vh"
  95. >
  96. <template #header>
  97. <div>
  98. <div v-if="item.tagName.length > 6">
  99. {{ item.tagName }}
  100. <span v-if="item.valueUnit">({{ item.valueUnit }})</span>
  101. </div>
  102. <div v-else>
  103. <div
  104. style="
  105. white-space: nowrap;
  106. text-overflow: ellipsis;
  107. overflow: hidden;
  108. "
  109. >
  110. {{ item.tagName }}
  111. </div>
  112. <div v-if="item.valueUnit">({{ item.valueUnit }})</div>
  113. </div>
  114. </div>
  115. </template>
  116. </el-table-column>
  117. </el-table>
  118. </div>
  119. </div>
  120. </div>
  121. </template>
  122. <script>
  123. import utils from "@/utills/downXlsx";
  124. import {
  125. getEquipmentReportData,
  126. getApicompanyslist,
  127. getApiwpByCplistlist,
  128. } from "@api/reportForms";
  129. export default {
  130. name: "windMonthly", //单机月报
  131. components: {},
  132. data() {
  133. return {
  134. companyVal: "",
  135. companyOptions: [],
  136. stationVal: "",
  137. stationOptions: [],
  138. pickerTimer: "",
  139. singleMachineData: [],
  140. showBtn: true,
  141. dialogTitle: "",
  142. tabIndex: -1,
  143. stationDatas: [],
  144. stationData: [
  145. {
  146. tagName: "机组编号",
  147. tagCode: "tagNames",
  148. valueUnit: "",
  149. width: "120%",
  150. },
  151. {
  152. tagName: "scada月发电量",
  153. tagCode: "yfdl",
  154. valueUnit: "万kWh",
  155. width: "105%",
  156. },
  157. {
  158. tagName: "月理论发电量",
  159. tagCode: "yllfdl",
  160. valueUnit: "万kWh",
  161. width: "105%",
  162. },
  163. {
  164. tagName: "月平均风速",
  165. tagCode: "ypjfs",
  166. valueUnit: "m/s",
  167. width: "85%",
  168. },
  169. {
  170. tagName: "设备利用小时数",
  171. tagCode: "sblyxss",
  172. valueUnit: "小时",
  173. width: "105%",
  174. },
  175. {
  176. tagName: "设备可利用率",
  177. tagCode: "sbklylv",
  178. valueUnit: "%",
  179. width: "105%",
  180. },
  181. {
  182. tagName: "等效可用系数",
  183. tagCode: "ddxkyxs",
  184. valueUnit: "%",
  185. width: "97%",
  186. },
  187. {
  188. tagName: "有效风时数",
  189. tagCode: "yxfss",
  190. valueUnit: "小时",
  191. width: "105%",
  192. },
  193. {
  194. tagName: "有效风时利用率",
  195. tagCode: "yxfslylv",
  196. valueUnit: "%",
  197. width: "105%",
  198. },
  199. {
  200. tagName: "故障损失电量",
  201. tagCode: "gzssdl",
  202. valueUnit: "万kWh",
  203. width: "97%",
  204. },
  205. {
  206. tagName: " 调度限电损失电量",
  207. tagCode: "ddxdssdl",
  208. valueUnit: "万kWh",
  209. width: "111%",
  210. },
  211. {
  212. tagName: "计划检修损失电量",
  213. tagCode: "jhjxssdl",
  214. valueUnit: "万kWh",
  215. width: "105%",
  216. },
  217. {
  218. tagName: "性能未达标损失电量",
  219. tagCode: "xnwdbssdl",
  220. valueUnit: "万kWh",
  221. width: "113%",
  222. },
  223. {
  224. tagName: "受累损失电量",
  225. tagCode: "slssdl",
  226. valueUnit: "万kWh",
  227. width: "113%",
  228. },
  229. {
  230. tagName: "并网时间",
  231. tagCode: "bwsj",
  232. valueUnit: "小时",
  233. width: "80%",
  234. },
  235. {
  236. tagName: "待机时间",
  237. tagCode: "djsj",
  238. valueUnit: "小时",
  239. width: "80%",
  240. },
  241. {
  242. tagName: "检修停机时间",
  243. tagCode: "jxtjsj",
  244. valueUnit: "小时",
  245. width: "100%",
  246. },
  247. {
  248. tagName: "故障停机时间",
  249. tagCode: "gztjsj",
  250. valueUnit: "小时",
  251. width: "100%",
  252. },
  253. {
  254. tagName: "连续无故障时间",
  255. tagCode: "lxwgzsj",
  256. valueUnit: "小时",
  257. width: "100%",
  258. },
  259. {
  260. tagName: "通讯中断时间",
  261. tagCode: "txzdsj",
  262. valueUnit: "小时",
  263. width: "98%",
  264. },
  265. ],
  266. stationVals: "",
  267. tabOptions: [
  268. { id: -1, name: "风电" },
  269. { id: -2, name: "光伏" },
  270. ],
  271. pageSize: 25,
  272. stationDatasd: [],
  273. allStation: [],
  274. };
  275. },
  276. created() {
  277. this.$nextTick(() => this.addRollListener());
  278. this.getCompanyData();
  279. },
  280. mounted() {
  281. window.onresize = () => {
  282. this.getTableData();
  283. };
  284. },
  285. computed: {},
  286. methods: {
  287. // 获取公司列表
  288. async getCompanyData() {
  289. this.companyOptions = [];
  290. this.pickerTimer = this.getchangeTime(new Date());
  291. const { data } = await getApicompanyslist({ type: "0" });
  292. var arr = [];
  293. if (data && data.length > 0) {
  294. {
  295. data.forEach((item) => {
  296. if (item.preCode == "") {
  297. this.companyVal = item.code;
  298. }
  299. var obj = {};
  300. obj.value = item.code;
  301. obj.label = item.aname;
  302. arr.push(obj);
  303. });
  304. }
  305. }
  306. this.companyOptions = arr;
  307. this.getStationData();
  308. },
  309. headerStyle({ row, column, rowIndex, columnIndex }) {
  310. if (columnIndex < 0) {
  311. return { "backdrop-filter": "blur(10px)", padding: "6px 0" };
  312. } else {
  313. return { padding: "6px 0" };
  314. }
  315. },
  316. changeBtn(id) {
  317. this.tabIndex = id;
  318. this.stationVal = "";
  319. this.getStationData();
  320. },
  321. changeCompan(val) {
  322. this.companyVal = val;
  323. this.stationVal = "";
  324. this.getStationData();
  325. },
  326. // 获取场站
  327. async getStationData() {
  328. this.stationOptions = [];
  329. let params = {
  330. companyCode: this.companyVal,
  331. stationType: this.tabIndex,
  332. };
  333. const { data } = await getApiwpByCplistlist(params);
  334. var arr = [];
  335. if (data && data.length > 0) {
  336. {
  337. data.forEach((item) => {
  338. var obj = {};
  339. obj.value = item.CZMC;
  340. obj.CZBM = item.CZBM;
  341. arr.push(obj);
  342. });
  343. }
  344. }
  345. this.stationVals = data[0].CZBM;
  346. this.stationVal = data[0].CZMC;
  347. this.stationOptions = arr;
  348. this.getTableData();
  349. },
  350. changeStation(val) {
  351. let obj = {};
  352. if (this.stationOptions != "") {
  353. obj = this.stationOptions.find((item) => {
  354. return item.value == val;
  355. });
  356. this.stationVals = obj.CZBM;
  357. }
  358. this.stationVal = val;
  359. },
  360. seachData() {
  361. this.getTableData();
  362. },
  363. async getTableData() {
  364. let params = {
  365. date: this.pickerTimer,
  366. stationId: this.stationVals,
  367. };
  368. const { data } = await getEquipmentReportData(params);
  369. if (data.length >= 1) {
  370. data?.forEach((value) => {
  371. value.tagNames = value.powerstationName + value.deviceName;
  372. for (let key in value) {
  373. if (
  374. key == "yfdl" ||
  375. key == "gzssdl" ||
  376. key == "yllfdl" ||
  377. key == "ddxdssdl" ||
  378. key == "jhjxssdl" ||
  379. key == "xnwdbssdl" ||
  380. key == "slssdl"
  381. ) {
  382. value[key] = (value[key] / 10000).toFixed(4);
  383. } else if (typeof value[key] === "number") {
  384. value[key] = value[key].toFixed(2);
  385. }
  386. }
  387. });
  388. this.allStation = data;
  389. this.stationDatasd = this.deepClone(data);
  390. this.stationDatas = this.stationDatasd.splice(0, this.pageSize);
  391. } else {
  392. this.stationDatas = [];
  393. this.allStation = [];
  394. }
  395. },
  396. addRollListener() {
  397. const dom = document.getElementsByClassName("el-table__body-wrapper")[0];
  398. dom.addEventListener("scroll", () => {
  399. const scrollTop = dom.scrollTop;
  400. const offsetHeight = dom.offsetHeight;
  401. const clientHeight = dom.clientHeight;
  402. const scrollHeight = dom.scrollHeight;
  403. if (
  404. clientHeight + scrollTop + 1 > scrollHeight ||
  405. offsetHeight + scrollTop >= scrollHeight
  406. ) {
  407. this.pageSize++;
  408. this.getTableData();
  409. }
  410. });
  411. },
  412. tableRowClassName({ row, rowIndex }) {
  413. if (row.powerstationName == "") {
  414. return "success-row";
  415. } else if (rowIndex === "") {
  416. return "";
  417. }
  418. return "";
  419. },
  420. //转换时间
  421. getchangeTime(date) {
  422. var y = date.getFullYear();
  423. var m = date.getMonth() + 1;
  424. m = m < 10 ? "0" + m : m;
  425. return y + "-" + m;
  426. },
  427. deepClone(ob) {
  428. var obj = JSON.stringify(ob);
  429. var objClone = JSON.parse(obj);
  430. return objClone;
  431. },
  432. downXlsxFn() {
  433. this.stationDatas = this.allStation;
  434. let headers = [];
  435. this.stationData.forEach((it) => {
  436. headers.push(it.tagName + it.valueUnit);
  437. });
  438. setTimeout(() => {
  439. if (this.stationDatas.length > 0) {
  440. utils.exportExcel(
  441. this.$refs["stand_table"].$el,
  442. headers,
  443. this.pickerTimer + "单机月报"
  444. );
  445. }
  446. }, 1000);
  447. },
  448. },
  449. };
  450. </script>
  451. <style lang="less">
  452. .singleMachine {
  453. padding: 0 23px;
  454. .singleMachine_title {
  455. padding-left: 10px;
  456. .leftContent {
  457. width: 242px;
  458. height: 41px;
  459. line-height: 41px;
  460. background: url("~@/assets/imgs/title_left_bg1.png") no-repeat;
  461. span {
  462. font-size: 16px;
  463. font-family: Microsoft YaHei;
  464. font-weight: 400;
  465. color: #ffffff;
  466. margin-left: 25px;
  467. }
  468. }
  469. .floatLeft {
  470. float: left;
  471. }
  472. .floatRight {
  473. float: right;
  474. }
  475. .rightContent {
  476. width: 212px;
  477. height: 28px;
  478. margin-top: 13px;
  479. background: url("../../../../assets/imgs/title_right_bg.png");
  480. }
  481. }
  482. .clearfix::after {
  483. content: "";
  484. clear: both;
  485. height: 0;
  486. line-height: 0;
  487. visibility: hidden;
  488. display: block;
  489. }
  490. .clearfix {
  491. zoom: 1;
  492. }
  493. .singleMachine_top {
  494. display: flex;
  495. flex-direction: row;
  496. align-items: center;
  497. padding-top: 10px;
  498. margin-bottom: 10px;
  499. .station {
  500. display: flex;
  501. flex-direction: row;
  502. align-items: center;
  503. font-size: 14px;
  504. font-family: Microsoft YaHei;
  505. font-weight: 400;
  506. color: #b3b3b3;
  507. margin-right: 10px;
  508. margin-left: 10px;
  509. }
  510. .tabCut {
  511. display: inline-block;
  512. margin: 0 10px;
  513. div {
  514. display: inline-block;
  515. width: 60px;
  516. height: 27px;
  517. border: 1px solid #274934;
  518. text-align: center;
  519. line-height: 25px;
  520. cursor: pointer;
  521. }
  522. div:nth-child(1) {
  523. border-radius: 13px 0px 0px 13px;
  524. }
  525. div:nth-child(2) {
  526. border-radius: 0px 13px 13px 0px;
  527. }
  528. .active {
  529. background-color: rgba(5, 187, 76, 0.9);
  530. color: #fff;
  531. }
  532. }
  533. .search-input {
  534. margin-left: 10px;
  535. .el-input__inner {
  536. width: 175px;
  537. }
  538. .el-input__suffix {
  539. right: -50px;
  540. }
  541. }
  542. .but {
  543. display: flex;
  544. flex-direction: row;
  545. align-content: center;
  546. margin-left: 20px;
  547. .buttons:nth-child(1) {
  548. background: rgba(5, 187, 76, 0.6);
  549. border: 1px solid #3b6c53;
  550. border-radius: 13px;
  551. color: #fff;
  552. &:hover {
  553. background: rgba(5, 187, 76, 0.9);
  554. border-radius: 13px;
  555. color: #fff;
  556. }
  557. }
  558. .buttons:nth-child(2) {
  559. background: rgba(67, 81, 107, 0.3);
  560. border: 1px solid #3b6c53;
  561. border-radius: 13px;
  562. font-size: 14px;
  563. color: #b3b3b3;
  564. }
  565. }
  566. }
  567. .singleMachine_Table {
  568. margin: 0 5px 5px 5px;
  569. padding-bottom: 10px;
  570. .el-table--mini {
  571. margin: 5px;
  572. .el-table__header-wrapper {
  573. tr {
  574. background: rgba(83, 89, 104, 0.3) !important;
  575. th {
  576. vertical-align: top !important;
  577. }
  578. .cell {
  579. color: #b3b3b3;
  580. font-family: MicrosoftYaHei;
  581. font-size: 14px;
  582. }
  583. }
  584. }
  585. .el-table__body-wrapper {
  586. tr {
  587. &:nth-child(2n) {
  588. background: rgba(83, 89, 104, 0.05) !important;
  589. }
  590. .cell {
  591. color: #d8d8d9;
  592. font-family: ArialMT;
  593. font-size: 13px;
  594. height: 25px !important;
  595. }
  596. }
  597. }
  598. }
  599. .el-pagination {
  600. display: flex;
  601. justify-content: flex-end;
  602. margin-right: 40px;
  603. margin-top: 20px;
  604. .btn-prev,
  605. .btn-next,
  606. .btn-quickprev,
  607. .btn-quicknext,
  608. .el-pager,
  609. .number {
  610. background: rgba(58, 63, 75, 0.4);
  611. color: #fff;
  612. border-radius: 2px;
  613. }
  614. .el-pager .active {
  615. background: #083c94;
  616. color: #fff;
  617. border-radius: 2px;
  618. }
  619. .el-pagination__total,
  620. .el-pagination__jump {
  621. color: #fff;
  622. .el-input__inner {
  623. background: rgba(58, 63, 75, 0.2);
  624. border: 1px solid #3e4349;
  625. color: #fff;
  626. border-radius: 2px;
  627. }
  628. }
  629. }
  630. .historyBtn {
  631. cursor: pointer;
  632. color: #1c99ff;
  633. margin-right: 15px;
  634. }
  635. }
  636. .el-overlay {
  637. .el-overlay-dialog {
  638. overflow-y: hidden !important;
  639. .windhistoryDetailModel,
  640. .windDetailmodal {
  641. margin-top: 0 !important;
  642. .el-dialog__body {
  643. padding: 10px 20px 20px 20px;
  644. }
  645. }
  646. }
  647. }
  648. .el-picker__popper .el-date-range-picker__header .el-picker-panel__icon-btn {
  649. color: #fff;
  650. }
  651. .el-picker__popper .el-date-table .in-range div {
  652. background: #43516b;
  653. }
  654. }
  655. </style>