index.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693
  1. <script setup name="prepare">
  2. import excelCop from '@/components/excel.vue'
  3. import treeCop from '@/components/tree.vue'
  4. import barChartCop from './components/barChart.vue'
  5. import SubmitBtn from '@/components/SubmitBtn.vue'
  6. import { ref, nextTick, onActivated, onMounted, reactive } from 'vue'
  7. import request from '@/api/axios.js'
  8. import { ElMessage } from 'element-plus'
  9. import util from "@tools/util";
  10. import CurrentScatterChart from './components/current-scatter-chart.vue'
  11. // import dotRes from '@/data/dot.json'
  12. // import tableRes from '@/data/table.json'
  13. // import areaDataRes from '@/data/areaData.json'
  14. /**配置参数 */
  15. const treeHeight = ref(window.innerHeight - 160 + 'px') //tree高度
  16. const excelHeight = ref(window.innerHeight - 160 + 'px') //excel高度
  17. const tableHeight = ref(window.innerHeight - 160 + 'px')
  18. /**excel 开始 */
  19. const excelCheckboxShow = ref(false)
  20. const excelCheckIds = ref([])
  21. const excelList = ref([])
  22. const funExcelChange = async (obj) => { //点击excel项时
  23. return false
  24. }
  25. const funExcelCheckChange = ({ checkArr, data }) => { //bug
  26. excelCheckIds.value = checkArr
  27. funSubmit()
  28. }
  29. /**prepare tree 开始 */
  30. const treeData = ref([])
  31. const treeCopRef = ref() //treeCop ref
  32. const actTreeNode = ref(null) //当前激活的treeNode
  33. const funRepeatMap = (arr, type='fitting') => {
  34. return arr.map(o => {
  35. if (o.children) {
  36. const findIndex = o.children.findIndex(p => !!p.type)
  37. if (findIndex !== -1) {
  38. o.childs = o.children
  39. o.children = []
  40. if(!actTreeNode.value && type === 'fitting'){ //判断当且仅有process获取tree时 赋值
  41. actTreeNode.value = o
  42. }
  43. }
  44. }
  45. return {
  46. ...o,
  47. children: o.children ? funRepeatMap(o.children, type) : []
  48. }
  49. })
  50. }
  51. const funGetTree = async () => {
  52. actTreeNode.value = null
  53. const res = await request.get("/power/fitting/tree")
  54. treeData.value = funRepeatMap(res.data)
  55. excelList.value = []
  56. if(actTreeNode.value){
  57. funCurrentChange({current: actTreeNode.value, currentNode: null})
  58. if(treeCopRef.value){
  59. treeCopRef.value.setCheckedKeys([actTreeNode.value.id])
  60. excelCheckIds.value = actTreeNode.value.childs.map(o => o.id)
  61. funSubmit()
  62. }
  63. }
  64. }
  65. const funCurrentChange = ({ current, currentNode }) => {
  66. excelCheckboxShow.value = true
  67. if (current.childs) {
  68. excelList.value = current.childs.map(o => {
  69. return {
  70. id: o.id,
  71. interval: o.interval,
  72. path: o.path,
  73. prepareid: o.prepareid,
  74. station: o.station,
  75. time: o.time,
  76. type: o.type,
  77. windturbine: o.windturbine,
  78. name: o.path.substring(o.path.indexOf(o.station + '_') + (o.station + '_').length)
  79. }
  80. })
  81. } else {
  82. excelList.value = []
  83. }
  84. }
  85. const funTreeCheckChange = ({ current, checkedNodes, checkedKeys, halfCheckedNodes, halfCheckedKeys }) => { //tree change -> excel change
  86. funCurrentChange({ current, currentNode: '' })
  87. const checkIds = []
  88. if (checkedNodes.length) {
  89. for (const node of checkedNodes) {
  90. if (node.childs && node.childs.length) {
  91. for (const child of node.childs) {
  92. checkIds.push(child.id)
  93. }
  94. }
  95. }
  96. }
  97. excelCheckIds.value = checkIds
  98. funSubmit()
  99. }
  100. /**search 开始 */
  101. const funSubmit = async () => {
  102. if (!excelCheckIds.value.length) {
  103. ElMessage.error('请勾选要执行的项')
  104. return false
  105. }
  106. wtData.value = []
  107. const params = {
  108. ids: excelCheckIds.value.join(',')
  109. }
  110. const res = await request.get('/power/fitting/line', { params: params })
  111. if (res.code === 200) {
  112. seriesData.value = []
  113. if(res.data.bzgl){
  114. seriesData.value.push(
  115. {
  116. name: "保证功率",
  117. type: "line",
  118. symbol: "line", //设定为实心点
  119. symbolSize: 0, //设定实心点的大小
  120. smooth: true, //这个是把线变成曲线
  121. data: res.data.bzgl || [],
  122. xAxisIndex: 0,
  123. },
  124. )
  125. }
  126. if(res.data.sjgl?.length){
  127. for(const wtObj of res.data.sjgl){
  128. seriesData.value.push(
  129. {
  130. name: wtObj.obj.windturbineId + "\n实际功率",
  131. type: "line",
  132. symbol: "line", //设定为实心点
  133. symbolSize: 0, //设定实心点的大小
  134. smooth: true, //这个是把线变成曲线
  135. data: wtObj.sjgl || [],
  136. xAxisIndex: 0,
  137. },
  138. )
  139. wtData.value.push(wtObj.obj)
  140. }
  141. }
  142. }
  143. }
  144. /**chart Data */
  145. const avgObj = reactive({ //平均cpz等
  146. cpavg: '',
  147. frequency: '',
  148. pcratio: ''
  149. })
  150. const xAxisData = ref([])
  151. const chartRef = ref() //chart 的ref
  152. const seriesData = ref([])
  153. const isChartArea = ref(false) // 用来控制图表是否区域划分
  154. const dataSet = ref('')
  155. const funChartSelect = async (batch) => {
  156. return false
  157. }
  158. const funChartArea = () => {
  159. if (seriesData.value?.length) {
  160. // 获取数据后 展示dialog table 数据
  161. wtDialog.value = true
  162. if (!isChartArea.value) {
  163. // 请求一下
  164. seriesData.value[0] = {
  165. ...seriesData.value[0],
  166. markLine: {
  167. symbol: 'none',
  168. label: {
  169. show: false
  170. },
  171. lineStyle: {
  172. color: 'rgba(96,174,255, 1)'
  173. },
  174. data: [
  175. {
  176. xAxis: 3,
  177. valueIndex: 0,
  178. },
  179. {
  180. xAxis: 5,
  181. valueIndex: 0
  182. },
  183. {
  184. xAxis: 10,
  185. valueIndex: 0
  186. },
  187. {
  188. xAxis: 12,
  189. valueIndex: 0
  190. },
  191. {
  192. xAxis: 25,
  193. valueIndex: 0
  194. },
  195. ]
  196. },
  197. markArea: {
  198. label: {
  199. fontSize: util.vh(12),
  200. },
  201. itemStyle: {
  202. color: 'rgba(236,245,255, 0)'
  203. },
  204. emphasis: {
  205. itemStyle: {
  206. color: 'rgba(96,174,255, 0.5)'
  207. }
  208. },
  209. data: [
  210. [
  211. {
  212. name: `3~5m`,
  213. xAxis: 3,
  214. },
  215. {
  216. xAxis: 5,
  217. }
  218. ],
  219. [
  220. {
  221. name: `5~10m`,
  222. xAxis: 5,
  223. },
  224. {
  225. xAxis: 10,
  226. }
  227. ],
  228. [
  229. {
  230. name: `10~12m`,
  231. xAxis: 10,
  232. },
  233. {
  234. xAxis: 12,
  235. }
  236. ],
  237. [
  238. {
  239. name: `12~25m`,
  240. xAxis: 12,
  241. },
  242. {
  243. xAxis: 25,
  244. }
  245. ],
  246. ]
  247. },
  248. }
  249. isChartArea.value = true
  250. } else {
  251. seriesData.value[0] = {
  252. ...seriesData.value[0],
  253. markLine: null,
  254. markArea: null,
  255. }
  256. isChartArea.value = false
  257. }
  258. }
  259. }
  260. /**tmdialog 数据 */
  261. const tmDialog = ref(false)
  262. const barxAxis = reactive({
  263. type: 'category',
  264. data: [],
  265. splitLine: {
  266. show: false
  267. },
  268. axisTick: {
  269. show: true
  270. }
  271. })
  272. const baryAxis = ref({
  273. type: 'value',
  274. name: '小时',
  275. splitLine: {
  276. show: false
  277. },
  278. axisTick: {
  279. show: true
  280. }
  281. })
  282. const barSeries = ref([{
  283. name: "不运行",
  284. type: "bar",
  285. stack: 'a',
  286. data: [],
  287. },{
  288. name: "3~5m风速",
  289. type: "bar",
  290. stack: 'a',
  291. data: [],
  292. },{
  293. name: "5~10m风速",
  294. type: "bar",
  295. stack: 'a',
  296. data: [],
  297. },{
  298. name: "10~12m风速",
  299. type: "bar",
  300. stack: 'a',
  301. data: [],
  302. },{
  303. name: "12~25m风速",
  304. type: "bar",
  305. stack: 'a',
  306. data: [],
  307. }])
  308. const barUnWorkSeries = ref([{
  309. name: "3~5m风速",
  310. type: "bar",
  311. stack: 'a',
  312. data: [],
  313. },{
  314. name: "5~10m风速",
  315. type: "bar",
  316. stack: 'a',
  317. data: [],
  318. },{
  319. name: "10~12m风速",
  320. type: "bar",
  321. stack: 'a',
  322. data: [],
  323. },{
  324. name: "12~25m风速",
  325. type: "bar",
  326. stack: 'a',
  327. data: [],
  328. }])
  329. const funTimeArea = async () => {
  330. if(seriesData.value?.length){
  331. activeTab.value = '1'
  332. //获取数据
  333. const res = await request.get('/power/fitting/time',{params: {ids: excelCheckIds.value.join(',')}})
  334. if(res.code===200){
  335. barxAxis.data = []
  336. barSeries.value = [{
  337. name: "不运行",
  338. type: "bar",
  339. stack: 'a',
  340. data: [],
  341. },{
  342. name: "3~5m风速",
  343. type: "bar",
  344. stack: 'a',
  345. data: [],
  346. },{
  347. name: "5~10m风速",
  348. type: "bar",
  349. stack: 'a',
  350. data: [],
  351. },{
  352. name: "10~12m风速",
  353. type: "bar",
  354. stack: 'a',
  355. data: [],
  356. },{
  357. name: "12~25m风速",
  358. type: "bar",
  359. stack: 'a',
  360. data: [],
  361. }]
  362. barUnWorkSeries.value = [{
  363. name: "3~5m风速",
  364. type: "bar",
  365. stack: 'a',
  366. data: [],
  367. },{
  368. name: "5~10m风速",
  369. type: "bar",
  370. stack: 'a',
  371. data: [],
  372. },{
  373. name: "10~12m风速",
  374. type: "bar",
  375. stack: 'a',
  376. data: [],
  377. },{
  378. name: "12~25m风速",
  379. type: "bar",
  380. stack: 'a',
  381. data: [],
  382. }]
  383. for(const wtObj of res.data){
  384. barxAxis.data.push(wtObj.wtId)
  385. for(const timeKey in wtObj.time1){
  386. barSeries.value[timeKey].data.push((wtObj.time1[timeKey]/60).toFixed(0))
  387. }
  388. for(const time2Key in wtObj.time2){
  389. barUnWorkSeries.value[time2Key].data.push((wtObj.time2[time2Key]/60).toFixed(0))
  390. }
  391. }
  392. tmDialog.value = true
  393. }
  394. }
  395. }
  396. /**额定功率 */
  397. const powerDialog = ref(false)
  398. const powerxAxis = reactive({
  399. type: 'category',
  400. data: [],
  401. splitLine: {
  402. show: false
  403. },
  404. axisTick: {
  405. show: true
  406. }
  407. })
  408. const poweryAxis = ref({
  409. type: 'value',
  410. name: 'kW',
  411. max: null,
  412. splitLine: {
  413. show: false
  414. },
  415. axisTick: {
  416. show: true
  417. }
  418. })
  419. const countyAxis = ref({
  420. type: 'value',
  421. name: '次数',
  422. splitLine: {
  423. show: false
  424. },
  425. axisTick: {
  426. show: true
  427. }
  428. })
  429. const powerSeries = ref([{
  430. name: "平均全功率",
  431. type: "bar",
  432. data: [],
  433. markLine: {}
  434. }])
  435. const countSeries = ref([{
  436. name: '次数',
  437. type: 'bar',
  438. data: [],
  439. markLine: {}
  440. }])
  441. const funPower = async () => {
  442. if(seriesData.value?.length){
  443. //获取数据
  444. const res = await request.get('/rated/power',{params: {ids: excelCheckIds.value.join(',')}})
  445. if(res.code===200){
  446. powerxAxis.data = []
  447. powerSeries.value = [
  448. {
  449. name: "平均全功率",
  450. type: "bar",
  451. data: [],
  452. markLine: {
  453. symbol: 'none',
  454. label: {
  455. show: true
  456. },
  457. lineStyle: {
  458. color: 'rgba(96,174,255, 1)'
  459. },
  460. data: [],
  461. }
  462. }
  463. ]
  464. countSeries.value = [
  465. {
  466. name: "次数",
  467. type: "bar",
  468. data: [],
  469. markLine: {}
  470. }
  471. ]
  472. if(res.data.data){
  473. const xAxis = [], seriesArr = [], countArr = []
  474. for(const val of res.data.data){
  475. xAxis.push(val.wt)
  476. seriesArr.push(val.avg)
  477. countArr.push(val.count)
  478. }
  479. powerxAxis.data = xAxis
  480. countSeries.value[0].data = countArr.map(o => {
  481. return {
  482. value: o,
  483. // itemStyle: {
  484. // color: o < res.data.avg ? 'rgb(197,78,82)' : 'rgb(50,93,171)'
  485. // }
  486. }
  487. })
  488. powerSeries.value[0].data = seriesArr.map(o => {
  489. return {
  490. value: o,
  491. itemStyle: {
  492. color: o < res.data.avg ? 'rgb(197,78,82)' : 'rgb(50,93,171)'
  493. }
  494. }
  495. })
  496. let maxValue = Math.max(...seriesArr)
  497. if(res.data.power > maxValue){
  498. maxValue = res.data.power
  499. }else{
  500. maxValue = null
  501. }
  502. poweryAxis.value.max = maxValue
  503. powerSeries.value[0].markLine.data = [{
  504. yAxis: res.data.avg,
  505. name: `平均额定功率${res.data.avg}`,
  506. lineStyle: {
  507. color: 'green'
  508. }
  509. },{
  510. yAxis: res.data.power,
  511. name: `额定功率${res.data.power}`,
  512. lineStyle: {
  513. color: 'red'
  514. }
  515. }]
  516. }
  517. powerDialog.value = true
  518. }
  519. }
  520. }
  521. /**dialog 数据 */
  522. const wtDialog = ref(false)
  523. const wtData = ref([])
  524. const activeTab = ref('1')
  525. /**created */
  526. // funGetTree()
  527. // funGetProcessTree()
  528. /**mounted */
  529. onMounted(() => {
  530. tableHeight.value = window.innerHeight - 160 + 'px'
  531. excelHeight.value = window.innerHeight - 160 + 'px'
  532. treeHeight.value = window.innerHeight - 160 + 'px'
  533. window.addEventListener('resize', () => {
  534. tableHeight.value = window.innerHeight - 160 + 'px'
  535. excelHeight.value = window.innerHeight - 160 + 'px'
  536. treeHeight.value = window.innerHeight - 160 + 'px'
  537. })
  538. /**test */
  539. // funExcelChange({
  540. // id: 1,
  541. // name: 'excel',
  542. // type: 'fitting',
  543. // })
  544. })
  545. /**activated */
  546. onActivated(() => {
  547. funGetTree()
  548. })
  549. </script>
  550. <template>
  551. <div class="bg-white py-[10px] px-[10px] s-dialog-body">
  552. <el-dialog draggable width="1000px" v-model="tmDialog" title="时间占比">
  553. <el-tabs v-model="activeTab">
  554. <el-tab-pane label="运行时间" name="1">
  555. </el-tab-pane>
  556. <el-tab-pane label="不运行时间" name="2"></el-tab-pane>
  557. <div v-if="activeTab === '1'">
  558. <bar-chart-cop height="600px" width="100%" :xAxis="barxAxis" :yAxis="baryAxis" :series="barSeries"></bar-chart-cop>
  559. </div>
  560. <div v-if="activeTab === '2'">
  561. <bar-chart-cop height="600px" :colors='["rgb(222,132,82)","rgb(105,188,80)","rgb(197,78,82)","rgb(129,114,181)"]' width="100%" :xAxis="barxAxis" :yAxis="baryAxis" :series="barUnWorkSeries"></bar-chart-cop>
  562. </div>
  563. </el-tabs>
  564. </el-dialog>
  565. <el-dialog draggable width="1000px" v-model="powerDialog" title="额定功率">
  566. <bar-chart-cop height="350px" :colors='[]' width="100%" :xAxis="powerxAxis" :yAxis="countyAxis" :series="countSeries"></bar-chart-cop>
  567. <bar-chart-cop height="350px" :colors='[]' width="100%" :xAxis="powerxAxis" :yAxis="poweryAxis" :series="powerSeries"></bar-chart-cop>
  568. </el-dialog>
  569. <el-dialog draggable v-model="wtDialog" title="曲线偏差率">
  570. <el-table :data="wtData" row-key="id" :max-height="600">
  571. <el-table-column property="windturbine" align="center" label="风机" />
  572. <el-table-column
  573. property="pc5ratio"
  574. sortable
  575. align="center"
  576. label="3~5m"
  577. />
  578. <el-table-column
  579. property="pc10ratio"
  580. sortable
  581. align="center"
  582. label="5~10m"
  583. />
  584. <el-table-column
  585. property="pc12ratio"
  586. sortable
  587. align="center"
  588. label="10~12m"
  589. />
  590. <el-table-column
  591. property="pc25ratio"
  592. sortable
  593. align="center"
  594. label="12~25m"
  595. />
  596. <el-table-column
  597. property="pcratio"
  598. sortable
  599. align="center"
  600. label="3~25m"
  601. />
  602. </el-table>
  603. </el-dialog>
  604. <div
  605. class="
  606. relative
  607. shadow
  608. rounded-[6px]
  609. shadow-blue-500
  610. px-[10px]
  611. pt-[20px]
  612. pb-[10px]
  613. "
  614. >
  615. <div class="text-[14px] absolute top-[-7px] text-[#838383] left-[20px]">
  616. 数据展示
  617. </div>
  618. <el-row :gutter="10">
  619. <el-col :span="5">
  620. <tree-cop
  621. ref="treeCopRef"
  622. :data="treeData"
  623. @checkChange="funTreeCheckChange"
  624. :show-checkbox="true"
  625. :height="treeHeight"
  626. @currentChange="funCurrentChange"
  627. @refresh="funGetTree"
  628. ></tree-cop>
  629. </el-col>
  630. <el-col :span="3">
  631. <excel-cop
  632. :checkIds="excelCheckIds"
  633. :showCheckbox="excelCheckboxShow"
  634. :data="excelList"
  635. :height="excelHeight"
  636. @excelChange="funExcelChange"
  637. @checkChange="funExcelCheckChange"
  638. ></excel-cop>
  639. </el-col>
  640. <el-col :span="16">
  641. <div class="px-[10px] shadow rounded-[6px] shadow-blue-500">
  642. <SubmitBtn
  643. class="absolute right-[196px] top-[6px] z-10"
  644. desc="区域划分"
  645. @click="funChartArea"
  646. ></SubmitBtn>
  647. <SubmitBtn
  648. class="absolute right-[106px] top-[6px] z-10"
  649. desc="时间占比"
  650. @click="funTimeArea"
  651. ></SubmitBtn>
  652. <SubmitBtn
  653. class="absolute right-[16px] top-[6px] z-10"
  654. desc="额定功率"
  655. @click="funPower"
  656. ></SubmitBtn>
  657. <div
  658. :style="{
  659. height:
  660. typeof tableHeight === 'string'
  661. ? tableHeight
  662. : tableHeight + 'px',
  663. overflow: 'hidden'
  664. }"
  665. class="p-[10px]"
  666. >
  667. <CurrentScatterChart
  668. ref="chartRef"
  669. width="100%"
  670. :height="`calc( ${tableHeight} - 40px )`"
  671. chartTitle=""
  672. :xAxisData="xAxisData"
  673. :yAxisData="{ splitLine: { show: false } }"
  674. :seriesData="seriesData"
  675. :showLegend="true"
  676. :brushSelected="!isChartArea"
  677. :dataSet="dataSet"
  678. @getSelected="funChartSelect"
  679. />
  680. </div>
  681. </div>
  682. </el-col>
  683. </el-row>
  684. </div>
  685. </div>
  686. </template>
  687. <style scoped>
  688. .s-dialog-body /deep/ .el-dialog__body{
  689. padding: 0px 20px;
  690. }
  691. </style>