index.vue 17 KB

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