index.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721
  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. });
  275. onUnmounted(() => {
  276. state.echarts?.clear(); //销毁
  277. clearInterval(state.Interval);
  278. state.Interval = null;
  279. });
  280. const dialogClose = (done) => {
  281. state.echarts?.clear(); //销毁
  282. done();
  283. };
  284. const query = reactive({
  285. page: 1,
  286. limit: 23,
  287. pageTotal: null,
  288. });
  289. const dialogVisible = ref(null);
  290. const state = reactive({
  291. handleSearch: null,
  292. input: "",
  293. inputAI: "",
  294. stationId: "",
  295. windturbineList: [],
  296. modelListAll: [],
  297. modeId: "",
  298. AIList: [],
  299. activeWT: "",
  300. activeAI: "",
  301. Interval: null,
  302. pickerVal: [new Date(new Date() - 1 * 60 * 60 * 1000), new Date()],
  303. echarts: null,
  304. treedata: [],
  305. defaultProps: {
  306. children: "children",
  307. label: (a) => {
  308. return a.node.name;
  309. },
  310. },
  311. structcode: "",
  312. selectDataVal: 1,
  313. selectData: [
  314. {
  315. label: "原始数据",
  316. value: 1,
  317. },
  318. {
  319. label: "数据快照",
  320. value: 2,
  321. },
  322. ],
  323. selectTimeVal: 1,
  324. downData: null,
  325. });
  326. // 机型
  327. const getequipmentmodel_list = async () => {
  328. const { data } = await fetchModel();
  329. state.modelListAll = data;
  330. };
  331. const modelList = computed(() => {
  332. // if (state.stationId == "") {
  333. // return [];
  334. // } else {
  335. // return state.modelListAll[state.stationId];
  336. // }
  337. return [
  338. {
  339. id: "SEC-W02B-1250kW",
  340. nemCode: "SEC-W02B-1250kW",
  341. name: "SEC-W02B-1250kW",
  342. aname: "SEC-W02B-1250kW",
  343. description: "DI",
  344. powerProduction: 1250,
  345. windturbineManufacturerId: "SHDQ_MF",
  346. photo: null,
  347. unit: null,
  348. cutinwindSpeed: 3,
  349. ratedwindSpeed: 11,
  350. cutoutwindSpeed: "25",
  351. sweptArea: 5800,
  352. equipmentCategory: "F",
  353. },
  354. ];
  355. });
  356. //get 风机机组
  357. const getWindturbineList = async () => {
  358. const { data } = await getApiequipmentListByWp({ wpid: state.stationId });
  359. state.windturbineList = data.data;
  360. state.activeWT = state.windturbineList[0]?.id;
  361. state.modeId = "";
  362. // await getequipmentmodel_list();
  363. await getFetchAIPointListt();
  364. };
  365. // 风机搜 索
  366. const filterWTList = computed(() =>
  367. state.windturbineList?.filter(
  368. (data) =>
  369. data.aname.includes(state.input) && data.modelId.includes(state.modeId)
  370. )
  371. );
  372. //AI点检索
  373. const filterAIList = computed(() =>
  374. // state.AIList?.filter(
  375. // (data) => !state.inputAI || data.name.includes(state.inputAI)
  376. // // !state.inputAI ||
  377. // // (data.uniformCode.toUpperCase().includes(state.inputAI.toUpperCase()) ||
  378. // // data.name.indexOf(state.inputAI))
  379. // )
  380. // );
  381. new Array(5).fill({
  382. unitNameCn: "",
  383. englishName: null,
  384. unitName: "",
  385. modelId: "SEC-W02B-1250kW",
  386. name: "01号风机风向",
  387. valueUnit: null,
  388. model: "SEC-W02B-1250kW",
  389. typeId: "F",
  390. id: "e222918e1e6a4feeba41007be3c85bf7",
  391. uniformCode: "AI067",
  392. value: "344.00",
  393. time: 1718176440000,
  394. })
  395. );
  396. // get 实时data
  397. const getLatest = async (stationId, AIlist) => {
  398. // const data = await getAdapterLatest(stationId, AIlist, state.stationId);
  399. // state.AIList?.forEach((e) => {
  400. // if (e.uniformCode.indexOf("AI") !== -1) {
  401. // e.value = Number(data[e.uniformCode]?.doubleValue).toFixed(2);
  402. // }
  403. // // else {
  404. // // e.value = data[e.uniformCode]?.booleanValue - 0;
  405. // // }
  406. // e["time"] = data[e.uniformCode]?.ts;
  407. // });
  408. };
  409. // get 历史数据
  410. const getHistory = async (stationId, AIpoint) => {
  411. let startTs = dayjs(state.pickerVal[0]).valueOf();
  412. let endTs = dayjs(state.pickerVal[1]).valueOf();
  413. const baseUrl = `http://10.81.3.162:801${
  414. /FDC/.test(state.stationId) ? "1" : "2"
  415. }/`;
  416. const data = await getAdapterHistory(
  417. stationId,
  418. AIpoint,
  419. startTs,
  420. endTs,
  421. baseUrl
  422. );
  423. state.downData = data;
  424. option.xAxis.data = [];
  425. option.series[0].data = [];
  426. let timeArr = [];
  427. data.forEach((e) => {
  428. option.series[0].data.push(e.doubleValue.toFixed(2));
  429. option.xAxis.data.push(dayjs(e.ts).format("YYYY-MM-DD HH:mm:ss"));
  430. });
  431. let chat = echarts.init(document.getElementById(eChart.value));
  432. state.echarts = await chat.setOption(option);
  433. };
  434. // get 数据快照
  435. const getHistorysnap = async (stationId, AIpoint, interval) => {
  436. let startTs = dayjs(state.pickerVal[0]).valueOf();
  437. let endTs = dayjs(state.pickerVal[1]).valueOf();
  438. const baseUrl = `http://10.81.3.162:801${
  439. /FDC/.test(state.stationId) ? "1" : "2"
  440. }/`;
  441. const data = await getAdapterHistorysnap(
  442. stationId,
  443. AIpoint,
  444. startTs,
  445. endTs,
  446. interval,
  447. baseUrl
  448. );
  449. state.downData = data;
  450. option.xAxis.data = [];
  451. option.series[0].data = [];
  452. let timeArr = [];
  453. data.forEach((e) => {
  454. option.series[0].data.push(e.doubleValue.toFixed(2));
  455. option.xAxis.data.push(dayjs(e.ts).format("YYYY-MM-DD HH:mm:ss"));
  456. });
  457. let chat = echarts.init(document.getElementById(eChart.value));
  458. state.echarts = await chat.setOption(option);
  459. };
  460. const option = {
  461. tooltip: {
  462. trigger: "axis",
  463. position: function (pt) {
  464. return [pt[0], "10%"];
  465. },
  466. },
  467. xAxis: {
  468. type: "category",
  469. boundaryGap: false,
  470. data: [],
  471. },
  472. yAxis: {
  473. type: "value",
  474. boundaryGap: [0, "100%"],
  475. },
  476. dataZoom: [
  477. {
  478. type: "inside",
  479. start: 0,
  480. end: 10,
  481. },
  482. {
  483. start: 0,
  484. end: 10,
  485. },
  486. ],
  487. series: [
  488. {
  489. data: [],
  490. type: "line",
  491. smooth: true,
  492. },
  493. ],
  494. };
  495. //get AI
  496. const getFetchAIPointListt = async () => {
  497. // const res = await fetchAIDIPointList(
  498. // "windturbine",
  499. // state.structcode,
  500. // state.stationId,
  501. // state.activeWT,
  502. // query.page,
  503. // query.limit
  504. // );
  505. // state.AIList = res?.records;
  506. // query.pageTotal = res?.total;
  507. // let AIlist = [];
  508. // res.records?.forEach((e) => {
  509. // AIlist.push(e.uniformCode);
  510. // });
  511. // if (state.Interval) {
  512. // clearInterval(state.Interval);
  513. // }
  514. getLatest(state.activeWT, AIlist);
  515. state.Interval = setInterval(getLatest, 5000, state.activeWT, AIlist);
  516. };
  517. // TREE
  518. // tree click
  519. const handleNodeClick = (data) => {
  520. state.structcode = data.node.code;
  521. getFetchAIPointListt();
  522. };
  523. // gettreeData
  524. const getTreeData = async () => {
  525. // const res = await tree();
  526. // let data = res.children[1].children;
  527. let data = [];
  528. data.unshift({ node: { code: "", name: "全部" } });
  529. // data.unshift({ node: { code: "1010102", name: "全部" } });
  530. // data.forEach((e) => {
  531. // delete e.children;
  532. // });
  533. state.treedata = data;
  534. };
  535. //
  536. const handleSearch = (item) => {
  537. dialogVisible.value = true;
  538. state.handleSearch = item;
  539. state.activeAI = item.uniformCode;
  540. getHistory(state.activeWT, state.activeAI);
  541. };
  542. const clickWT = (item) => {
  543. state.activeWT = item.id;
  544. if (state.Interval) {
  545. clearInterval(state.Interval);
  546. }
  547. getFetchAIPointListt();
  548. };
  549. const pickerSearch = () => {
  550. if (state.selectDataVal == 1) {
  551. getHistory(state.activeWT, state.activeAI);
  552. } else {
  553. getHistorysnap(state.activeWT, state.activeAI, state.selectTimeVal * 60);
  554. }
  555. };
  556. // 分页导航
  557. const handlePageChange = (val) => {
  558. query.page = val;
  559. getFetchAIPointListt();
  560. };
  561. // 批量导出
  562. const export2Excel = async () => {
  563. let data = state.downData;
  564. data.forEach((e) => {
  565. e.doubleValue = e.doubleValue.toFixed(2);
  566. e.ts = dayjs(e.ts).format("YYYY-MM-DD HH:mm:ss");
  567. });
  568. ElMessage.success(`导出成功!`);
  569. const tableHeader = ["时间", "值"];
  570. const tableKey = ["ts", "doubleValue"];
  571. outExportExcel(
  572. tableHeader,
  573. tableKey,
  574. data,
  575. `${state.activeWT}_${state.activeAI}_${dayjs(state.pickerVal[0]).format(
  576. "YYYY-MM-DD HH:mm"
  577. )}-${dayjs(state.pickerVal[1]).format("YYYY-MM-DD HH:mm")}历史数据excel`
  578. );
  579. };
  580. </script>
  581. <style lang="less" scoped>
  582. .data-search {
  583. width: 100%;
  584. height: 100%;
  585. padding: 10px 20px;
  586. .leftContent {
  587. width: 242px;
  588. height: 41px;
  589. display: flex;
  590. align-items: center;
  591. background: url("~@/assets/imgs/title_left_bg1.png") no-repeat;
  592. span {
  593. font-size: 16px;
  594. font-family: Microsoft YaHei;
  595. font-weight: 400;
  596. color: #05bb4c;
  597. margin-left: 25px;
  598. }
  599. }
  600. .data-content {
  601. width: 100%;
  602. height: calc(100% - 40px);
  603. display: flex;
  604. justify-content: space-between;
  605. .card-left {
  606. width: 45%;
  607. height: 100%;
  608. display: flex;
  609. margin-right: 20px;
  610. .cardleft {
  611. width: 50%;
  612. height: 100%;
  613. margin-right: 20px;
  614. padding: 10px;
  615. background: #161f1e;
  616. .cardleft-search {
  617. padding-bottom: 10px;
  618. .cardleft_top {
  619. display: flex;
  620. padding-bottom: 10px;
  621. .el-select {
  622. flex: 1;
  623. &:nth-child(1) {
  624. margin-right: 10px;
  625. }
  626. }
  627. }
  628. }
  629. }
  630. .cardright {
  631. width: calc(50% - 20px);
  632. height: 100%;
  633. padding: 10px;
  634. background: #161f1e;
  635. }
  636. }
  637. .card-right {
  638. width: calc(55% - 20px);
  639. height: 100%;
  640. padding: 10px;
  641. background: #161f1e;
  642. .pagination {
  643. padding-top: 10px;
  644. }
  645. }
  646. }
  647. }
  648. .pickerFifter {
  649. width: 80%;
  650. margin: 0 auto;
  651. display: flex;
  652. align-items: center;
  653. margin-bottom: 10px;
  654. }
  655. .scrollbar-demo-item {
  656. display: flex;
  657. align-items: center;
  658. justify-content: center;
  659. cursor: pointer;
  660. height: 30px;
  661. margin: 8px 0;
  662. text-align: center;
  663. border-radius: 4px;
  664. color: #b3b3b3;
  665. background-color: rgba(7, 122, 52, 0.2);
  666. border: 1px solid #3b6c53;
  667. &.active,
  668. &:hover {
  669. background-color: rgba(5, 187, 76, 0.5);
  670. color: #ffffff;
  671. }
  672. &:nth-child(1) {
  673. margin-top: 0;
  674. }
  675. &:nth-last-child(1) {
  676. margin-bottom: 0;
  677. }
  678. }
  679. .buttons {
  680. background-color: rgba(5, 187, 76, 0.2);
  681. border: 1px solid #3b6c53;
  682. color: #b3b3b3;
  683. font-size: 14px;
  684. &:hover {
  685. background-color: rgba(5, 187, 76, 0.5);
  686. color: #ffffff;
  687. }
  688. }
  689. .selectTimeVal ::v-deep {
  690. margin-right: 10px;
  691. .el-input__inner {
  692. text-align: right;
  693. border-radius: 12.5px 0px 0px 12.5px !important;
  694. }
  695. .el-input-group__append {
  696. background-color: #172422 !important;
  697. color: #fff !important;
  698. border: 0px !important;
  699. border-radius: 12.5px !important;
  700. border-top-left-radius: 0 !important;
  701. border-bottom-left-radius: 0 !important;
  702. }
  703. }
  704. </style>