index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579
  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.id"
  13. :label="item.aname"
  14. :value="item.id"
  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.id"
  40. :label="item.name"
  41. :value="item.id"
  42. >
  43. </el-option>
  44. </el-select>
  45. </div>
  46. <div class="station">
  47. 时间:
  48. <div>
  49. <el-date-picker
  50. v-model="pickerTimer"
  51. type="date"
  52. value-format="YYYY-MM-DD"
  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 class="singleMachine_content">
  69. <div class="singleMachine_title clearfix">
  70. <div class="leftContent floatLeft"><span>单机性能分析</span></div>
  71. </div>
  72. <div class="economicTable1">
  73. <el-table
  74. :data="singleMachineData"
  75. size="mini"
  76. stripe
  77. ref="stand_table"
  78. height="calc(100% - 40px)"
  79. style="width: 100%"
  80. >
  81. <el-table-column
  82. align="center"
  83. prop="wtcode"
  84. :label="tabIndex == -1 ? '风机' : '逆变器'"
  85. show-overflow-tooltip
  86. width="80"
  87. sortable
  88. >
  89. <template #default="scope">
  90. <span
  91. @click="seachDetail(scope.row)"
  92. style="cursor: pointer; color: #05bb4c"
  93. >{{ scope.row.wtcode }}</span
  94. >
  95. </template>
  96. </el-table-column>
  97. <el-table-column
  98. v-for="(item, index) in tabIndex === -1
  99. ? tableHeader
  100. : tableHeaderGf"
  101. :key="index"
  102. sortable
  103. :prop="item.code"
  104. :label="item.title"
  105. show-overflow-tooltip
  106. header-align="center"
  107. :align="item.align ? item.align : 'center'"
  108. >
  109. <!-- <template #header="scope">
  110. <div v-if="scope.column.label.indexOf('(') > 0">
  111. <div>
  112. {{
  113. scope.column.label.slice(0, scope.column.label.indexOf("("))
  114. }}
  115. </div>
  116. <div>
  117. {{
  118. scope.column.label.slice(scope.column.label.indexOf("("))
  119. }}
  120. </div>
  121. </div>
  122. <div v-else>{{ scope.column.label }}</div>
  123. </template> -->
  124. <template #default="scope">
  125. <span>{{ scope.row[item.code] }}</span>
  126. </template>
  127. </el-table-column>
  128. <el-table-column label="操作" align="center" width="80">
  129. <template #default="scope">
  130. <span class="historyBtn" @click="seachHistoryData(scope.row)"
  131. >历史</span
  132. >
  133. </template>
  134. </el-table-column>
  135. </el-table>
  136. <el-pagination
  137. @current-change="handleCurrentChange"
  138. :current-page="page.currentPage"
  139. :page-size="page.pagesize"
  140. @size-change="handleSizeChange"
  141. :page-sizes="[23, 50, 100, 500]"
  142. layout="total, sizes, prev, pager, next, jumper"
  143. :total="page.total"
  144. >
  145. </el-pagination>
  146. </div>
  147. </div>
  148. <el-dialog
  149. v-model="dialogVisible"
  150. width="100%"
  151. custom-class="windhistoryDetailModel"
  152. destroy-on-close
  153. fullscreen
  154. >
  155. <template #title>
  156. <div class="dialog-title">
  157. <div class="title">{{ dialogTitle }}</div>
  158. </div>
  159. </template>
  160. <div style="height: 100%">
  161. <history-detail
  162. ref="windhistoryDetail"
  163. :tabIndex="tabIndex"
  164. :historyStationOptions="stationOptions"
  165. >
  166. </history-detail>
  167. </div>
  168. </el-dialog>
  169. <el-dialog
  170. v-model="dialogDetail"
  171. width="100%"
  172. custom-class="windDetailmodal"
  173. fullscreen
  174. destroy-on-close
  175. >
  176. <template #title>
  177. <div class="dialog-title">
  178. <div class="title">详情</div>
  179. </div>
  180. </template>
  181. <div style="height: 100%">
  182. <wind-detail-dialog
  183. ref="windDetail"
  184. :wpArray="stationOptions"
  185. ></wind-detail-dialog>
  186. </div>
  187. </el-dialog>
  188. </div>
  189. </template>
  190. <script>
  191. import {
  192. getApicompanyslist,
  193. getApiwpByCplistlist,
  194. getApiwindturbinegoodnesslist,
  195. getApiwindturbinegoodnesslistGf,
  196. } from "@/api/monthlyPerformanceAnalysis";
  197. import historyDetail from "./historyDetail.vue";
  198. import WindDetailDialog from "./windDetailDialog.vue";
  199. import utils from "@/utills/downXlsx";
  200. import dayjs from "dayjs";
  201. export default {
  202. name: "SingleWindAnasyle", //单机性能分析
  203. components: {
  204. historyDetail,
  205. WindDetailDialog,
  206. },
  207. data() {
  208. return {
  209. companyVal: "",
  210. companyOptions: [],
  211. stationVal: "",
  212. stationOptions: [],
  213. pickerTimer: "",
  214. singleMachineData: [],
  215. showBtn: true,
  216. dialogTitle: "",
  217. tabIndex: -1,
  218. tabOptions: [
  219. { id: -1, name: "风电" },
  220. { id: -2, name: "光伏" },
  221. ],
  222. tableHeader: [
  223. // { title: "风机", code: "wtcode" },
  224. { title: "型号", code: "modelId", align: "left" },
  225. { title: "日排行榜", code: "dayTop", align: "right" },
  226. { title: "日拟合优度(%)", code: "dayGoodness", align: "right" },
  227. { title: "日风速(m/s)", code: "daySpeed", align: "right" },
  228. { title: "月排行榜", code: "monthTop", align: "right" },
  229. { title: "月拟合优度(%)", code: "monthGoodness", align: "right" },
  230. { title: "月风速(m/s)", code: "monthSpeed", align: "right" },
  231. { title: "年排行榜", code: "yearTop", align: "right" },
  232. { title: "年拟合优度(%)", code: "yearGoodness", align: "right" },
  233. { title: "年风速(m/s)", code: "yearSpeed", align: "right" },
  234. // { title: "操作" },
  235. ],
  236. tableHeaderGf: [
  237. { title: "逆变器", code: "wtname" },
  238. { title: "型号", code: "model", align: "left" },
  239. { title: "日系统效率", code: "rxtxl", align: "right" },
  240. { title: "日离散率", code: "rlsl", align: "right" },
  241. { title: "日转换效率", code: "rzhxl", align: "right" },
  242. { title: "月系统效率", code: "yxtxl", align: "right" },
  243. { title: "月离散率", code: "ylsl", align: "right" },
  244. { title: "月转换效率", code: "yzhxl", align: "right" },
  245. { title: "年系统效率", code: "nxtxl", align: "right" },
  246. { title: "年离散率", code: "nlsl", align: "right" },
  247. { title: "年转换效率", code: "nzhxl", align: "right" },
  248. // { title: "操作" },
  249. ],
  250. page: {
  251. pagesize: 23,
  252. currentPage: 1,
  253. total: 0,
  254. },
  255. dialogVisible: false,
  256. dialogDetail: false,
  257. isFullScreen: false,
  258. };
  259. },
  260. created() {
  261. this.pickerTimer = dayjs().add(-1, "day").format("YYYY-MM-DD");
  262. this.getCompanyData();
  263. },
  264. mounted() {},
  265. computed: {
  266. pageHeight() {
  267. return {
  268. height: document.documentElement.clientHeight - 130 + "px",
  269. };
  270. },
  271. pagewindHeight() {
  272. return {
  273. height: document.documentElement.clientHeight - 100 + "px",
  274. };
  275. },
  276. },
  277. methods: {
  278. // 获取公司列表
  279. async getCompanyData() {
  280. this.companyOptions = [];
  281. const { data: datas } = await getApicompanyslist();
  282. this.companyOptions = datas.data;
  283. this.companyVal = datas.data[0]?.id;
  284. this.getStationData(this.companyVal);
  285. },
  286. changeBtn(id) {
  287. this.tabIndex = id;
  288. this.stationVal = "";
  289. this.singleMachineData = [];
  290. this.getStationData(this.companyVal);
  291. },
  292. changeCompan(val) {
  293. this.companyVal = val;
  294. this.stationVal = "";
  295. this.singleMachineData = [];
  296. this.getStationData(this.companyVal);
  297. },
  298. // 获取场站
  299. async getStationData(companyVal) {
  300. this.stationOptions = [];
  301. let params = {
  302. type: this.tabIndex,
  303. companyid: companyVal,
  304. };
  305. const { data } = await getApiwpByCplistlist(params);
  306. if (data.data.length) {
  307. this.stationOptions = data.data;
  308. this.stationVal = data.data[0].id || "";
  309. } else {
  310. this.stationOptions = [];
  311. this.stationVal = "";
  312. }
  313. this.getTableData();
  314. },
  315. changeStation(val) {
  316. this.stationVal = val;
  317. this.getTableData();
  318. },
  319. seachData() {
  320. this.getTableData();
  321. },
  322. async getTableData() {
  323. let params = {
  324. cmId: this.companyVal,
  325. type: this.tabIndex,
  326. pageNum: this.page.currentPage,
  327. pageSize: this.page.pagesize,
  328. recorddate: this.pickerTimer,
  329. wpId: this.stationVal,
  330. };
  331. let datas = {};
  332. if (this.tabIndex == -2) {
  333. datas = await getApiwindturbinegoodnesslistGf(params);
  334. this.singleMachineData = datas.data.data;
  335. this.page.total = datas.data.data.length;
  336. } else {
  337. datas = await getApiwindturbinegoodnesslist(params);
  338. this.singleMachineData = datas.data.data.records;
  339. this.page.total = datas.data.data.total;
  340. }
  341. },
  342. handleSizeChange(val) {
  343. this.page.currentPage = 1;
  344. this.page.pagesize = val;
  345. this.getTableData();
  346. },
  347. handleCurrentChange(val) {
  348. this.page.currentPage = val;
  349. this.getTableData();
  350. },
  351. //转换时间
  352. getchangeTime(date) {
  353. var y = date.getFullYear();
  354. var m = date.getMonth() + 1;
  355. m = m < 10 ? "0" + m : m;
  356. var d = date.getDate();
  357. d = d < 10 ? "0" + d : d;
  358. return y + "-" + m + "-" + d;
  359. },
  360. seachDetail(row) {
  361. this.dialogDetail = true;
  362. this.$nextTick(() => {
  363. this.$refs.windDetail.recorddate = this.pickerTimer;
  364. this.$refs.windDetail.init(row);
  365. });
  366. },
  367. seachHistoryData(row) {
  368. this.dialogVisible = true;
  369. this.dialogTitle = "历史数据查询";
  370. this.$nextTick(() => {
  371. let startT = dayjs(this.pickerTimer)
  372. .add(-10, "day")
  373. .format("YYYY-MM-DD");
  374. this.$refs.windhistoryDetail.pickerTimer = [startT, this.pickerTimer];
  375. if (this.tabIndex == -1) {
  376. this.$refs.windhistoryDetail.init(row);
  377. } else {
  378. row.windtpowerstationId = this.stationVal;
  379. this.$refs.windhistoryDetail.init(row);
  380. }
  381. // this.$refs.windhistoryDetail.getWindPowerStation();
  382. });
  383. },
  384. downXlsxFn() {
  385. let header = [];
  386. this.tableHeader.forEach((it) => {
  387. if (it.title !== "操作") {
  388. header.push(it.title);
  389. }
  390. });
  391. if (this.singleMachineData.length > 0) {
  392. utils.exportExcel(
  393. this.$refs["stand_table"].$el,
  394. header,
  395. "单机性能分析"
  396. );
  397. }
  398. },
  399. },
  400. };
  401. </script>
  402. <style lang="less">
  403. .singleMachine {
  404. height: 100%;
  405. padding: 0 20px;
  406. .singleMachine_title {
  407. .leftContent {
  408. width: 242px;
  409. height: 41px;
  410. line-height: 41px;
  411. background: url("~@/assets/imgs/title_left_bg1.png") no-repeat;
  412. span {
  413. font-size: 16px;
  414. font-family: Microsoft YaHei;
  415. font-weight: 400;
  416. color: #05bb4c;
  417. margin-left: 25px;
  418. }
  419. }
  420. .floatLeft {
  421. float: left;
  422. }
  423. .floatRight {
  424. float: right;
  425. }
  426. }
  427. .singleMachine_content {
  428. background: rgba(0, 0, 0, 0.45);
  429. height: calc(100% - 48px);
  430. }
  431. .clearfix::after {
  432. content: "";
  433. clear: both;
  434. height: 0;
  435. line-height: 0;
  436. visibility: hidden;
  437. display: block;
  438. }
  439. .clearfix {
  440. zoom: 1;
  441. }
  442. .singleMachine_top {
  443. display: flex;
  444. flex-direction: row;
  445. align-items: center;
  446. padding-top: 5px;
  447. padding-bottom: 5px;
  448. .station {
  449. display: flex;
  450. flex-direction: row;
  451. align-items: center;
  452. font-size: 14px;
  453. font-family: Microsoft YaHei;
  454. font-weight: 400;
  455. color: #b3b3b3;
  456. margin-right: 10px;
  457. margin-left: 10px;
  458. }
  459. .tabCut {
  460. display: inline-block;
  461. margin: 0 10px;
  462. div {
  463. display: inline-block;
  464. width: 60px;
  465. height: 27px;
  466. border: 1px solid #274934;
  467. text-align: center;
  468. line-height: 25px;
  469. cursor: pointer;
  470. }
  471. div:nth-child(1) {
  472. border-radius: 13px 0px 0px 13px;
  473. border-right-width: 0;
  474. }
  475. div:nth-child(2) {
  476. border-radius: 0px 13px 13px 0px;
  477. }
  478. .active {
  479. background-color: rgba(5, 187, 76, 0.9);
  480. color: #fff;
  481. }
  482. }
  483. .search-input {
  484. margin-left: 10px;
  485. .el-input__inner {
  486. width: 175px;
  487. }
  488. .el-input__suffix {
  489. right: -50px;
  490. }
  491. }
  492. .but {
  493. display: flex;
  494. flex-direction: row;
  495. align-content: center;
  496. margin-left: 20px;
  497. .buttons {
  498. background-color: rgba(5, 187, 76, 0.2);
  499. border: 1px solid #3b6c53;
  500. color: #b3b3b3;
  501. font-size: 14px;
  502. &:hover,
  503. &.active {
  504. background-color: rgba(5, 187, 76, 0.5);
  505. color: #ffffff;
  506. }
  507. }
  508. }
  509. }
  510. .economicTable1 {
  511. height: calc(100% - 42px);
  512. .el-table__fixed,
  513. .el-table__fixed-right {
  514. background: rgba(0, 0, 0, 1);
  515. border-left: 2px solid #000;
  516. height: calc(100% - 15px) !important;
  517. }
  518. .el-table__fixed::before {
  519. background-color: #2a2a2a;
  520. }
  521. .el-table__fixed-right::before {
  522. background-color: #2a2a2a;
  523. }
  524. .el-pagination {
  525. display: flex;
  526. justify-content: flex-end;
  527. padding-top: 5px;
  528. .el-pagination__total,
  529. .el-pagination__jump {
  530. color: #fff;
  531. }
  532. }
  533. .historyBtn {
  534. cursor: pointer;
  535. color: #05bb4c;
  536. margin-right: 15px;
  537. }
  538. }
  539. .el-overlay {
  540. .el-overlay-dialog {
  541. overflow-y: hidden !important;
  542. .windhistoryDetailModel,
  543. .windDetailmodal {
  544. margin-top: 0 !important;
  545. .el-dialog__body {
  546. padding: 10px 20px 20px 20px;
  547. height: calc(100% - 80px);
  548. }
  549. }
  550. }
  551. }
  552. }
  553. </style>