index.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723
  1. <template>
  2. <div class="data-search">
  3. <div class="leftContent">
  4. <span>{{ pageTitle }}</span>
  5. </div>
  6. <div class="data-content">
  7. <div class="card-left">
  8. <div class="cardleft">
  9. <div class="cardleft-search">
  10. <div class="cardleft_top">
  11. <el-select
  12. v-model="state.stationId"
  13. clearable
  14. size="mini"
  15. placeholder="全部场站"
  16. popper-class="select"
  17. @change="getWindturbineList()"
  18. >
  19. <el-option
  20. v-for="item in stationList"
  21. :key="item.id"
  22. :value="item.id"
  23. :label="item.name"
  24. ></el-option>
  25. </el-select>
  26. <el-select
  27. v-model="state.modeId"
  28. clearable
  29. size="mini"
  30. placeholder="全部机型"
  31. popper-class="select"
  32. >
  33. <el-option
  34. v-for="item in modelList"
  35. :key="item.id"
  36. :value="item.id"
  37. :label="item.aname"
  38. ></el-option>
  39. </el-select>
  40. </div>
  41. <el-input
  42. v-model="state.input"
  43. size="mini"
  44. placeholder="设备检索"
  45. clearable
  46. />
  47. </div>
  48. <div style="height: calc(100% - 74px)">
  49. <el-scrollbar height="100%">
  50. <div
  51. v-for="item in filterWTList"
  52. :key="item"
  53. :class="[
  54. 'scrollbar-demo-item',
  55. item.id == state.activeWT ? 'active' : '',
  56. ]"
  57. @click="clickWT(item)"
  58. >
  59. {{ item.aname }}
  60. </div>
  61. </el-scrollbar>
  62. </div>
  63. </div>
  64. <div class="cardright">
  65. <el-input
  66. v-model="state.inputAI"
  67. style="margin-bottom: 10px"
  68. size="mini"
  69. placeholder="点名检索"
  70. clearable
  71. />
  72. <el-tree
  73. :data="state.treedata"
  74. :props="state.defaultProps"
  75. node-key="id"
  76. highlight-current
  77. default-expand-all
  78. @node-click="handleNodeClick"
  79. />
  80. </div>
  81. </div>
  82. <div class="card-right">
  83. <el-table
  84. :data="filterAIList"
  85. style="width: 100%"
  86. height="calc(100% - 40px)"
  87. >
  88. <el-table-column
  89. prop="uniformCode"
  90. label="点名"
  91. align="center"
  92. width="120"
  93. />
  94. <el-table-column prop="name" label="名称" align="left" />
  95. <el-table-column prop="value" label="值" width="120" align="right">
  96. <template #default="scope">
  97. <el-tag
  98. size="small"
  99. color="#133122"
  100. style="border-color: #67c23a"
  101. type="success"
  102. >
  103. {{ scope.row.value }}
  104. </el-tag>
  105. </template>
  106. </el-table-column>
  107. <el-table-column label="时间" width="180" align="center">
  108. <template #default="scope">
  109. {{ dayjs(scope.row.time).format("YYYY-MM-DD HH:mm:ss") }}
  110. </template>
  111. </el-table-column>
  112. <el-table-column label="单位" width="100" align="center">
  113. <template #default="scope">
  114. {{ scope.row.unitName }}
  115. {{ scope.row.unitNameCn ? `(${scope.row.unitNameCn})` : "--" }}
  116. </template>
  117. </el-table-column>
  118. <el-table-column width="100" align="center">
  119. <template #default="scope">
  120. <el-button
  121. type="text"
  122. style="color: #05bb4c"
  123. size="mini"
  124. @click="handleSearch(scope.row)"
  125. >历史数据</el-button
  126. >
  127. </template>
  128. </el-table-column>
  129. </el-table>
  130. <div class="pagination">
  131. <el-pagination
  132. layout="total,sizes, prev, pager, next"
  133. :current-page="query.page"
  134. :page-size="query.limit"
  135. :page-sizes="[23, 50, 100, 500]"
  136. :total="query.pageTotal"
  137. @current-change="handlePageChange"
  138. @size-change="
  139. (value) => {
  140. query.page = 1;
  141. query.limit = value;
  142. getFetchAIPointListt();
  143. }
  144. "
  145. ></el-pagination>
  146. </div>
  147. </div>
  148. <el-dialog
  149. v-model="dialogVisible"
  150. width="80%"
  151. :before-close="dialogClose"
  152. >
  153. <template #title>
  154. <div class="dialog-title">
  155. <div class="title">
  156. {{
  157. `${state.handleSearch?.name || ""}_${state.activeAI}历史数据`
  158. }}
  159. </div>
  160. </div>
  161. </template>
  162. <div class="pickerFifter">
  163. <el-date-picker
  164. size="mini"
  165. v-model="state.pickerVal"
  166. type="datetimerange"
  167. range-separator="-"
  168. start-placeholder="开始时间"
  169. end-placeholder="结束时间"
  170. >
  171. </el-date-picker>
  172. <el-select
  173. v-model="state.selectDataVal"
  174. placeholder="数据源"
  175. style="margin: 0 10px"
  176. >
  177. <el-option
  178. v-for="item in state.selectData"
  179. :key="item.value"
  180. :label="item.label"
  181. :value="item.value"
  182. >
  183. </el-option>
  184. </el-select>
  185. <!-- <el-select
  186. v-model="state.selectTimeVal"
  187. v-show="state.selectDataVal !== 1"
  188. placeholder="时间间隔"
  189. >
  190. <el-option
  191. v-for="item in state.selectTime"
  192. :key="item.value"
  193. :label="item.label"
  194. :value="item.value"
  195. >
  196. </el-option>
  197. </el-select> -->
  198. <el-input
  199. size="mini"
  200. class="selectTimeVal"
  201. v-model="state.selectTimeVal"
  202. style="width: 200px"
  203. v-show="state.selectDataVal !== 1"
  204. placeholder="时间间隔"
  205. >
  206. <template #append>分钟</template>
  207. </el-input>
  208. <el-button size="mini" class="buttons" round @click="pickerSearch"
  209. >查询</el-button
  210. >
  211. <el-button size="mini" class="buttons" round @click="export2Excel"
  212. >导出</el-button
  213. >
  214. </div>
  215. <div
  216. class="dChart"
  217. style="height: 500px; width: 100%"
  218. :id="eChart"
  219. ></div>
  220. </el-dialog>
  221. </div>
  222. </div>
  223. </template>
  224. <script setup>
  225. import { getApiequipmentListByWp } from "@/api/monthlyPerformanceAnalysis.js";
  226. import {
  227. ref,
  228. onMounted,
  229. watch,
  230. reactive,
  231. computed,
  232. onUnmounted,
  233. nextTick,
  234. } from "vue";
  235. import {
  236. fetchWindturbineList,
  237. tree,
  238. fetchAIDIPointList,
  239. getAdapterLatest,
  240. getAdapterHistory,
  241. getAdapterHistorysnap,
  242. AdapterHistoryExport,
  243. getStationinfo,
  244. fetchModel,
  245. } from "@/api/zhbj/index.js";
  246. import dayjs from "dayjs";
  247. import * as echarts from "echarts";
  248. import { useStore } from "vuex";
  249. import axios from "axios";
  250. import { ElMessage } from "element-plus";
  251. import { outExportExcel } from "@/tools/excel/exportExcel.js"; //引入文件
  252. const pageTitle = "数据查询";
  253. const store = useStore();
  254. const isStation = computed(() => store.getters.isStation);
  255. const stationList = computed(() => store.state.stationListAll);
  256. const eChart = ref("eChart" + Date.now() + Math.random());
  257. watch(
  258. () => stationList,
  259. (val, old) => {
  260. val.value[0] &&
  261. nextTick(async () => {
  262. state.stationId = stationList.value[0]?.id;
  263. await getWindturbineList();
  264. getTreeData();
  265. });
  266. },
  267. {
  268. deep: true,
  269. immediate: true,
  270. }
  271. );
  272. onMounted(async () => {
  273. getequipmentmodel_list();
  274. getTreeData();
  275. });
  276. onUnmounted(() => {
  277. state.echarts?.clear(); //销毁
  278. clearInterval(state.Interval);
  279. state.Interval = null;
  280. });
  281. const dialogClose = (done) => {
  282. state.echarts?.clear(); //销毁
  283. done();
  284. };
  285. const query = reactive({
  286. page: 1,
  287. limit: 23,
  288. pageTotal: null,
  289. });
  290. const dialogVisible = ref(null);
  291. const state = reactive({
  292. handleSearch: null,
  293. input: "",
  294. inputAI: "",
  295. stationId: "",
  296. windturbineList: [],
  297. modelListAll: [],
  298. modeId: "",
  299. AIList: [],
  300. activeWT: "",
  301. activeAI: "",
  302. Interval: null,
  303. pickerVal: [new Date(new Date() - 1 * 60 * 60 * 1000), new Date()],
  304. echarts: null,
  305. treedata: [],
  306. defaultProps: {
  307. children: "children",
  308. label: (a) => {
  309. return a.node.name;
  310. },
  311. },
  312. structcode: "",
  313. selectDataVal: 1,
  314. selectData: [
  315. {
  316. label: "原始数据",
  317. value: 1,
  318. },
  319. {
  320. label: "数据快照",
  321. value: 2,
  322. },
  323. ],
  324. selectTimeVal: 1,
  325. downData: null,
  326. });
  327. // 机型
  328. const getequipmentmodel_list = async () => {
  329. const { data } = await fetchModel();
  330. state.modelListAll = data;
  331. };
  332. const modelList = computed(() => {
  333. // if (state.stationId == "") {
  334. // return [];
  335. // } else {
  336. // return state.modelListAll[state.stationId];
  337. // }
  338. return [
  339. {
  340. id: "SEC-W02B-1250kW",
  341. nemCode: "SEC-W02B-1250kW",
  342. name: "SEC-W02B-1250kW",
  343. aname: "SEC-W02B-1250kW",
  344. description: "DI",
  345. powerProduction: 1250,
  346. windturbineManufacturerId: "SHDQ_MF",
  347. photo: null,
  348. unit: null,
  349. cutinwindSpeed: 3,
  350. ratedwindSpeed: 11,
  351. cutoutwindSpeed: "25",
  352. sweptArea: 5800,
  353. equipmentCategory: "F",
  354. },
  355. ];
  356. });
  357. //get 风机机组
  358. const getWindturbineList = async () => {
  359. const { data } = await getApiequipmentListByWp({ wpid: state.stationId });
  360. state.windturbineList = data.data;
  361. state.activeWT = state.windturbineList[0]?.id;
  362. state.modeId = "";
  363. // await getequipmentmodel_list();
  364. await getFetchAIPointListt();
  365. };
  366. // 风机搜 索
  367. const filterWTList = computed(() =>
  368. state.windturbineList?.filter(
  369. (data) =>
  370. data.aname.includes(state.input) && data.modelId.includes(state.modeId)
  371. )
  372. );
  373. //AI点检索
  374. const filterAIList = computed(() =>
  375. // state.AIList?.filter(
  376. // (data) => !state.inputAI || data.name.includes(state.inputAI)
  377. // // !state.inputAI ||
  378. // // (data.uniformCode.toUpperCase().includes(state.inputAI.toUpperCase()) ||
  379. // // data.name.indexOf(state.inputAI))
  380. // )
  381. // );
  382. new Array(5).fill({
  383. unitNameCn: "",
  384. englishName: null,
  385. unitName: "",
  386. modelId: "SEC-W02B-1250kW",
  387. name: "01号风机风向",
  388. valueUnit: null,
  389. model: "SEC-W02B-1250kW",
  390. typeId: "F",
  391. id: "e222918e1e6a4feeba41007be3c85bf7",
  392. uniformCode: "AI067",
  393. value: "344.00",
  394. time: 1718176440000,
  395. })
  396. );
  397. // get 实时data
  398. const getLatest = async (stationId, AIlist) => {
  399. const data = await getAdapterLatest(stationId, AIlist, state.stationId);
  400. state.AIList?.forEach((e) => {
  401. if (e.uniformCode.indexOf("AI") !== -1) {
  402. e.value = Number(data[e.uniformCode]?.doubleValue).toFixed(2);
  403. }
  404. // else {
  405. // e.value = data[e.uniformCode]?.booleanValue - 0;
  406. // }
  407. e["time"] = data[e.uniformCode]?.ts;
  408. });
  409. };
  410. // get 历史数据
  411. const getHistory = async (stationId, AIpoint) => {
  412. let startTs = dayjs(state.pickerVal[0]).valueOf();
  413. let endTs = dayjs(state.pickerVal[1]).valueOf();
  414. const baseUrl = `http://10.81.3.162:801${
  415. /FDC/.test(state.stationId) ? "1" : "2"
  416. }/`;
  417. const data = await getAdapterHistory(
  418. stationId,
  419. AIpoint,
  420. startTs,
  421. endTs,
  422. baseUrl
  423. );
  424. state.downData = data;
  425. option.xAxis.data = [];
  426. option.series[0].data = [];
  427. let timeArr = [];
  428. data.forEach((e) => {
  429. option.series[0].data.push(e.doubleValue.toFixed(2));
  430. option.xAxis.data.push(dayjs(e.ts).format("YYYY-MM-DD HH:mm:ss"));
  431. });
  432. let chat = echarts.init(document.getElementById(eChart.value));
  433. state.echarts = await chat.setOption(option);
  434. };
  435. // get 数据快照
  436. const getHistorysnap = async (stationId, AIpoint, interval) => {
  437. let startTs = dayjs(state.pickerVal[0]).valueOf();
  438. let endTs = dayjs(state.pickerVal[1]).valueOf();
  439. const baseUrl = `http://10.81.3.162:801${
  440. /FDC/.test(state.stationId) ? "1" : "2"
  441. }/`;
  442. const data = await getAdapterHistorysnap(
  443. stationId,
  444. AIpoint,
  445. startTs,
  446. endTs,
  447. interval,
  448. baseUrl
  449. );
  450. state.downData = data;
  451. option.xAxis.data = [];
  452. option.series[0].data = [];
  453. let timeArr = [];
  454. data.forEach((e) => {
  455. option.series[0].data.push(e.doubleValue.toFixed(2));
  456. option.xAxis.data.push(dayjs(e.ts).format("YYYY-MM-DD HH:mm:ss"));
  457. });
  458. let chat = echarts.init(document.getElementById(eChart.value));
  459. state.echarts = await chat.setOption(option);
  460. };
  461. const option = {
  462. tooltip: {
  463. trigger: "axis",
  464. position: function (pt) {
  465. return [pt[0], "10%"];
  466. },
  467. },
  468. xAxis: {
  469. type: "category",
  470. boundaryGap: false,
  471. data: [],
  472. },
  473. yAxis: {
  474. type: "value",
  475. boundaryGap: [0, "100%"],
  476. },
  477. dataZoom: [
  478. {
  479. type: "inside",
  480. start: 0,
  481. end: 10,
  482. },
  483. {
  484. start: 0,
  485. end: 10,
  486. },
  487. ],
  488. series: [
  489. {
  490. data: [],
  491. type: "line",
  492. smooth: true,
  493. },
  494. ],
  495. };
  496. //get AI
  497. const getFetchAIPointListt = async () => {
  498. const res = await fetchAIDIPointList(
  499. "windturbine",
  500. state.structcode,
  501. state.stationId,
  502. state.activeWT,
  503. query.page,
  504. query.limit
  505. );
  506. state.AIList = res?.records;
  507. query.pageTotal = res?.total;
  508. let AIlist = [];
  509. res.records?.forEach((e) => {
  510. AIlist.push(e.uniformCode);
  511. });
  512. if (state.Interval) {
  513. clearInterval(state.Interval);
  514. }
  515. getLatest(state.activeWT, AIlist);
  516. state.Interval = setInterval(getLatest, 5000, state.activeWT, AIlist);
  517. };
  518. // TREE
  519. // tree click
  520. const handleNodeClick = (data) => {
  521. state.structcode = data.node.code;
  522. getFetchAIPointListt();
  523. };
  524. // gettreeData
  525. const getTreeData = async () => {
  526. // const res = await tree();
  527. // let data = res.children[1].children;
  528. let data = [];
  529. data.unshift({ node: { code: "", name: "全部" } });
  530. // data.unshift({ node: { code: "1010102", name: "全部" } });
  531. // data.forEach((e) => {
  532. // delete e.children;
  533. // });
  534. state.treedata = data;
  535. };
  536. //
  537. const handleSearch = (item) => {
  538. dialogVisible.value = true;
  539. state.handleSearch = item;
  540. state.activeAI = item.uniformCode;
  541. getHistory(state.activeWT, state.activeAI);
  542. };
  543. const clickWT = (item) => {
  544. state.activeWT = item.id;
  545. if (state.Interval) {
  546. clearInterval(state.Interval);
  547. }
  548. getFetchAIPointListt();
  549. };
  550. const pickerSearch = () => {
  551. if (state.selectDataVal == 1) {
  552. getHistory(state.activeWT, state.activeAI);
  553. } else {
  554. getHistorysnap(state.activeWT, state.activeAI, state.selectTimeVal * 60);
  555. }
  556. };
  557. // 分页导航
  558. const handlePageChange = (val) => {
  559. query.page = val;
  560. getFetchAIPointListt();
  561. };
  562. // 批量导出
  563. const export2Excel = async () => {
  564. let data = state.downData;
  565. data.forEach((e) => {
  566. e.doubleValue = e.doubleValue.toFixed(2);
  567. e.ts = dayjs(e.ts).format("YYYY-MM-DD HH:mm:ss");
  568. });
  569. ElMessage.success(`导出成功!`);
  570. const tableHeader = ["时间", "值"];
  571. const tableKey = ["ts", "doubleValue"];
  572. outExportExcel(
  573. tableHeader,
  574. tableKey,
  575. data,
  576. `${state.activeWT}_${state.activeAI}_${dayjs(state.pickerVal[0]).format(
  577. "YYYY-MM-DD HH:mm"
  578. )}-${dayjs(state.pickerVal[1]).format("YYYY-MM-DD HH:mm")}历史数据excel`
  579. );
  580. };
  581. </script>
  582. <style lang="less" scoped>
  583. .data-search {
  584. width: 100%;
  585. height: 100%;
  586. padding: 10px 20px;
  587. .leftContent {
  588. width: 242px;
  589. height: 41px;
  590. display: flex;
  591. align-items: center;
  592. // background: url("~@/assets/imgs/title_left_bg1.png") no-repeat;
  593. background: url("~@/assets/imgs/title_left_bg.png") no-repeat;
  594. span {
  595. font-size: 16px;
  596. font-family: Microsoft YaHei;
  597. font-weight: 400;
  598. color: #05bb4c;
  599. margin-left: 25px;
  600. }
  601. }
  602. .data-content {
  603. width: 100%;
  604. height: calc(100% - 40px);
  605. display: flex;
  606. justify-content: space-between;
  607. .card-left {
  608. width: 45%;
  609. height: 100%;
  610. display: flex;
  611. margin-right: 20px;
  612. .cardleft {
  613. width: 50%;
  614. height: 100%;
  615. margin-right: 20px;
  616. padding: 10px;
  617. background: #161f1e;
  618. .cardleft-search {
  619. padding-bottom: 10px;
  620. .cardleft_top {
  621. display: flex;
  622. padding-bottom: 10px;
  623. .el-select {
  624. flex: 1;
  625. &:nth-child(1) {
  626. margin-right: 10px;
  627. }
  628. }
  629. }
  630. }
  631. }
  632. .cardright {
  633. width: calc(50% - 20px);
  634. height: 100%;
  635. padding: 10px;
  636. background: #161f1e;
  637. }
  638. }
  639. .card-right {
  640. width: calc(55% - 20px);
  641. height: 100%;
  642. padding: 10px;
  643. background: #161f1e;
  644. .pagination {
  645. padding-top: 10px;
  646. }
  647. }
  648. }
  649. }
  650. .pickerFifter {
  651. width: 80%;
  652. margin: 0 auto;
  653. display: flex;
  654. align-items: center;
  655. margin-bottom: 10px;
  656. }
  657. .scrollbar-demo-item {
  658. display: flex;
  659. align-items: center;
  660. justify-content: center;
  661. cursor: pointer;
  662. height: 30px;
  663. margin: 8px 0;
  664. text-align: center;
  665. border-radius: 4px;
  666. color: #b3b3b3;
  667. background-color: rgba(7, 122, 52, 0.2);
  668. border: 1px solid #3b6c53;
  669. &.active,
  670. &:hover {
  671. background-color: rgba(5, 187, 76, 0.5);
  672. color: #ffffff;
  673. }
  674. &:nth-child(1) {
  675. margin-top: 0;
  676. }
  677. &:nth-last-child(1) {
  678. margin-bottom: 0;
  679. }
  680. }
  681. .buttons {
  682. background-color: rgba(5, 187, 76, 0.2);
  683. border: 1px solid #3b6c53;
  684. color: #b3b3b3;
  685. font-size: 14px;
  686. &:hover {
  687. background-color: rgba(5, 187, 76, 0.5);
  688. color: #ffffff;
  689. }
  690. }
  691. .selectTimeVal ::v-deep {
  692. margin-right: 10px;
  693. .el-input__inner {
  694. text-align: right;
  695. border-radius: 12.5px 0px 0px 12.5px !important;
  696. }
  697. .el-input-group__append {
  698. background-color: #172422 !important;
  699. color: #fff !important;
  700. border: 0px !important;
  701. border-radius: 12.5px !important;
  702. border-top-left-radius: 0 !important;
  703. border-bottom-left-radius: 0 !important;
  704. }
  705. }
  706. </style>