dlyc.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. <template>
  2. <div class="dlTable">
  3. <div class="query mg-b-8">
  4. <div class="query-items">
  5. <div class="query-item">
  6. <div class="lable">选择日期:</div>
  7. <div class="search-input">
  8. <el-date-picker v-model="month" type="month" value-format="YYYY-MM" placeholder="选择日期" popper-class="date-select"> </el-date-picker>
  9. </div>
  10. </div>
  11. <div class="query-actions">
  12. <button class="btn" @click="exportExcel()">导出</button>
  13. </div>
  14. </div>
  15. </div>
  16. <panel :title="monthTitle + '发电量预测'" :showLine="false">
  17. <el-table :data="tableData" height="85vh" style="width: 100%;" class="" show-summary :summary-method="getSummaries" :span-method="yhmxdbMethod">
  18. <el-table-column fixed prop="region" label="名称" width="120" />
  19. <el-table-column fixed prop="speed" label="" width="50" />
  20. <el-table-column label="日期">
  21. <el-table-column v-for="(item, index) in dateS" :key="index" :prop="item.prop" :label="item.label"
  22. width="70" align="center" />
  23. </el-table-column>
  24. <el-table-column fixed="right" prop="hj" label="平均风速/合计" width="100" align="center" />
  25. <el-table-column fixed="right" prop="dqhj" label="地区合并" width="100" align="center" />
  26. </el-table>
  27. <el-table :data="tableDataExport" height="85vh" style="width: 100%;display: none;" id="out-table" :span-method="yhmxdbMethod">
  28. <el-table-column prop="region" label="名称" width="120" />
  29. <el-table-column prop="speed" label="" width="50" />
  30. <el-table-column label="日期">
  31. <el-table-column v-for="(item, index) in dateS" :key="index" :prop="item.prop" :label="item.label" width="70" align="center" />
  32. </el-table-column>
  33. <el-table-column prop="hj" label="平均风速/合计" width="100" align="center" />
  34. <el-table-column prop="dqhj" label="地区合并" width="100" align="center" />
  35. </el-table>
  36. </panel>
  37. </div>
  38. </template>
  39. <script>
  40. import $ from "jquery";
  41. import Panel from "@/components/coms/panel/panel.vue";
  42. import FileSaver from "file-saver";
  43. import XLSX from "xlsx";
  44. export default {
  45. components: {
  46. Panel,
  47. FileSaver,
  48. XLSX
  49. },
  50. data() {
  51. return {
  52. month: new Date().formatDate("yyyy-MM"),
  53. monthTitle: "",
  54. spanArr: [],
  55. pos: 0,
  56. spanArrHj: [],
  57. SUM:[],
  58. dateS: [{
  59. prop: "01",
  60. label: 1,
  61. },
  62. {
  63. prop: "02",
  64. label: 2,
  65. },
  66. ],
  67. tableData: [{
  68. region: "Tom",
  69. speed: "白天",
  70. hj: 3,
  71. dqhj: 6,
  72. "01": 55,
  73. wpid: "NSS_FDC",
  74. }],
  75. tableDataExport:[]
  76. };
  77. },
  78. created() {
  79. this.monthTit(this.month);
  80. this.list(new Date(this.month).valueOf());
  81. },
  82. mounted() {
  83. this.$nextTick(() => {
  84. $(".el-table__fixed-footer-wrapper tbody td:last-child").remove();
  85. $(".el-table__fixed-footer-wrapper tbody td:last-child").attr("colspan", "2");
  86. });
  87. },
  88. methods: {
  89. list(month) {
  90. let that = this;
  91. this.API.requestData({
  92. showLoading: true,
  93. method: "GET",
  94. baseURL: "http://10.155.32.4:8082/",
  95. subUrl: "decision/ycfsdl/" + month,
  96. success(res) {
  97. let tableData = [],
  98. windpowerstationid = [],
  99. dateS = [],
  100. recodedata = [],
  101. average = [];
  102. res.data.Weatherfd.forEach((ele) => {
  103. let dd = new Date(ele.recodedata).formatDate("dd");
  104. recodedata.push(dd);
  105. if (windpowerstationid.indexOf(ele.windpowerstationid) === -1) {//判断有木有id,若没有,则加进去,并新增一条tableData的白天和夜间
  106. windpowerstationid.push(ele.windpowerstationid);
  107. tableData.push({
  108. region: ele.region,
  109. speed: "白天",
  110. [dd]: ele.speed1,
  111. wpid: ele.windpowerstationid,
  112. dqhj: "",
  113. });
  114. tableData.push({
  115. region: ele.region,
  116. speed: "夜间",
  117. [dd]: ele.speed2,
  118. wpid: ele.windpowerstationid,
  119. dqhj: "",
  120. });
  121. average.push(ele.speed1, ele.speed2);
  122. } else {//匹配tableData的id,并分别传到夜间和白天data中
  123. tableData.forEach((ele2, index2) => {
  124. if (ele2.wpid === ele.windpowerstationid) {
  125. if (ele2.speed === "白天") {
  126. tableData[index2][dd] = ele.speed1;
  127. average[index2] = average[index2] + ele.speed1;
  128. } else {
  129. tableData[index2][dd] = ele.speed2;
  130. average[index2] = average[index2] + ele.speed2;
  131. }
  132. }
  133. });
  134. }
  135. });
  136. Object.values(Array.from(new Set(recodedata))).forEach((ele) => {
  137. dateS.push({
  138. prop: ele,
  139. label: ele,
  140. });
  141. });
  142. average.forEach((ele, index) => {//合计里的平均风速
  143. tableData[index]["hj"] = (ele / dateS.length).toFixed(2);
  144. });
  145. that.dateS = dateS;
  146. that.Forecastwindspeed(res.data.Forecastwindspeed, tableData);
  147. },
  148. });
  149. },
  150. Forecastwindspeed(data, Weatherfd) {
  151. let that = this,
  152. tableData = [],
  153. pjid = [],
  154. wpid = [],
  155. dqhj = {};
  156. data.forEach((ele) => {
  157. let dd = new Date(ele.recodedate).formatDate("dd");
  158. let wpidName = ele.pjid.split("_")[0].match(/^[a-z|A-Z]+/gi)[0];
  159. let power = ele.daypower + ele.nightpower;
  160. if (pjid.indexOf(ele.pjid) === -1) {//判断有木有id,若没有,则加进去,并新增一条tableData的白天和夜间
  161. pjid.push(ele.pjid);
  162. tableData.push({
  163. region: ele.pjname,
  164. speed: "白天",
  165. [dd]: ele.daypower,
  166. wpid: ele.wpid,
  167. pjid: ele.pjid,
  168. hj: power,
  169. dqhj: 1,
  170. });
  171. tableData.push({
  172. region: ele.pjname,
  173. speed: "夜间",
  174. [dd]: ele.nightpower,
  175. wpid: ele.wpid,
  176. pjid: ele.pjid,
  177. });
  178. tableData.push({
  179. region: ele.pjname,
  180. speed: "合计",
  181. [dd]: power,
  182. wpid: ele.wpid,
  183. pjid: ele.pjid,
  184. });
  185. dqhj[wpidName] != undefined ?
  186. (dqhj[wpidName] += power) :
  187. (dqhj[wpidName] = power);
  188. } else {//匹配tableData的id,并分别传到夜间和白天data中
  189. tableData.forEach((ele2, index2) => {
  190. if (ele2.pjid === ele.pjid) {
  191. if (ele2.speed === "白天") {
  192. tableData[index2][dd] = ele.daypower;
  193. tableData[index2].hj += power;
  194. } else if (ele2.speed === "夜间") {
  195. tableData[index2][dd] = ele.nightpower;
  196. } else {
  197. tableData[index2][dd] = power;
  198. dqhj[wpidName] += power;
  199. }
  200. }
  201. });
  202. }
  203. });
  204. tableData.find((ele) => {//地区合并
  205. if (ele.dqhj == 1) {
  206. ele.dqhj = dqhj[ele.wpid.split("_")[0]];
  207. }
  208. });
  209. let arr = [];
  210. Weatherfd.forEach((ele, index) => {
  211. arr.push(ele);
  212. tableData.forEach((ele2, index2) => {
  213. if (ele.wpid === ele2.wpid && ele.speed === "夜间") {
  214. arr.push(ele2);
  215. }
  216. });
  217. });
  218. that.getSpanArr(arr);
  219. that.tableData = arr;
  220. },
  221. yhmxdbMethod({row,column,rowIndex,columnIndex}) {
  222. if (columnIndex === 0) {
  223. let _row = this.spanArr[rowIndex];
  224. const _col = _row > 0 ? 1 : 0;
  225. return [_row, _col];
  226. } else if (columnIndex === this.dateS.length + 2) {//合计列
  227. let _row = this.spanArr[rowIndex];
  228. if (_row < 3) {
  229. _row = 1;
  230. }
  231. const _col = _row > 0 ? 1 : 0;
  232. return [_row, _col];
  233. } else if (columnIndex === this.dateS.length + 3) {//地区合并列
  234. let _row = this.spanArrHj[rowIndex];
  235. const _col = _row > 0 ? 1 : 0;
  236. return [_row, _col];
  237. }
  238. },
  239. getSpanArr(data,port) {
  240. this.spanArr = [];
  241. for (var i = 0; i < data.length; i++) {
  242. if (i === 0) {
  243. this.spanArr.push(1);
  244. this.pos = 0;
  245. } else {// 判断当前元素与上一个元素是否相同
  246. if (data[i].region === data[i - 1].region) {
  247. this.spanArr[this.pos] += 1;
  248. this.spanArr.push(0);
  249. }else if (data[i].region === '合计') {
  250. this.spanArr.push(1);
  251. this.pos = i;
  252. } else {
  253. this.spanArr.push(1);
  254. this.pos = i;
  255. }
  256. }
  257. }
  258. let spanArrIndex = [], //下标为2时,获取spanArr所对应的下标
  259. idx = [];
  260. this.spanArr.forEach((ele, index) => {
  261. if (ele === 2) {
  262. idx.push(index + 2);
  263. spanArrIndex.push(1);
  264. } else if (this.spanArr[index - 1] === 2 || ele === 1) {
  265. spanArrIndex.push(1);
  266. }else {
  267. spanArrIndex.push(0);
  268. }
  269. });
  270. if(port){
  271. idx.push(this.spanArr.length + 1);
  272. }else{
  273. idx.push(this.spanArr.length + 2);
  274. }
  275. idx.forEach((ele, index) => {
  276. if (idx[index + 1]) {
  277. spanArrIndex[ele] = idx[index + 1] - (ele + 2);
  278. }
  279. });
  280. this.spanArrHj = spanArrIndex;
  281. },
  282. getSummaries(param) {
  283. const {columns,data} = param;
  284. const sums = [];
  285. columns.forEach((column, index) => {
  286. if (index === 0) {
  287. sums[index] = "合计";
  288. return;
  289. }
  290. const values = data.map((item) => {
  291. if (item.pjid && item.speed === "白天") {
  292. return Number(item[column.property]);
  293. }
  294. });
  295. if (!values.every((value) => isNaN(value))) {
  296. sums[index] = values.reduce((prev, curr) => {
  297. const value = Number(curr);
  298. if (!isNaN(value)) {
  299. return prev + curr;
  300. } else {
  301. return prev;
  302. }
  303. }, 0);
  304. } else {
  305. sums[index] = "";
  306. }
  307. });
  308. sums[sums.length - 1] = "";
  309. this.SUM = sums;
  310. return sums;
  311. },
  312. tableDataE(tableData,SUM){
  313. let tableDataExport = [...tableData],
  314. sums = {},
  315. dateS = [];
  316. this.dateS.forEach(ele=>{
  317. dateS.push(ele.prop)
  318. })
  319. SUM.forEach((ele,index)=>{
  320. if(index === 0){
  321. sums['region'] = ele;
  322. }else if(index < SUM.length - 2 && index > 1){
  323. sums[dateS[index - 2]] = ele;
  324. }else if(index === SUM.length - 2){
  325. sums['dqhj'] = ele
  326. }else if(index === SUM.length - 1){
  327. sums['hj'] = ele
  328. }
  329. })
  330. tableDataExport.push(sums);
  331. this.getSpanArr(tableDataExport,true);
  332. this.tableDataExport = tableDataExport;
  333. /* 从表生成工作簿对象 */
  334. setTimeout(()=>{
  335. let wb = XLSX.utils.table_to_book(document.querySelector('#out-table'))
  336. /* 获取二进制字符串作为输出 */
  337. var wbout = XLSX.write(wb, {bookType: 'xlsx',bookSST: true,type: 'array'})
  338. try {
  339. FileSaver.saveAs(
  340. new Blob([wbout], { type: 'application/octet-stream' }),this.monthTitle + '发电量预测.xlsx')
  341. } catch (e) {
  342. if (typeof console !== 'undefined') console.log(e, wbout)
  343. }
  344. return wbout
  345. },1500)
  346. },
  347. exportExcel() {
  348. this.tableDataE(this.tableData,this.SUM)
  349. },
  350. monthTit(e){
  351. let tit = e.split("-");
  352. if (tit[1].indexOf(0) != -1) {
  353. tit[1] = tit[1].split("0")[1];
  354. }
  355. this.monthTitle = tit[0] + "年" + tit[1] + "月";
  356. }
  357. },
  358. watch: {
  359. month(e) {
  360. this.list(new Date(e).valueOf());
  361. this.monthTit(e);
  362. },
  363. },
  364. };
  365. </script>
  366. <style lang="less">
  367. .main-body {
  368. .dlTable {
  369. .el-table thead.is-group th.el-table__cell {
  370. background-color: rgb(30, 37, 36);
  371. }
  372. .el-table__expanded-cell {
  373. background: #141e1e;
  374. }
  375. .el-table__body {
  376. tr.hover-row>td {
  377. background-color: rgb(4, 12, 11);
  378. color: #b3bdc0 !important;
  379. }
  380. td {
  381. background-color: rgb(4, 12, 11);
  382. }
  383. td[rowspan="2"],
  384. td[rowspan="3"],
  385. tr.hover-row>td[rowspan="2"],
  386. tr.hover-row>td[rowspan="3"],
  387. tr td:last-of-type,
  388. tr td:nth-last-of-type(2),
  389. tr:hover td {
  390. background-color: #141e1e !important;
  391. }
  392. }
  393. .el-table__fixed-footer-wrapper tbody td.el-table__cell {
  394. color: #b3bdc0;
  395. background-color: rgb(30, 37, 36) !important;
  396. }
  397. .el-table__footer-wrapper tbody td.el-table__cell,
  398. .el-table__header-wrapper tbody td.el-table__cell {
  399. color: white;
  400. background-color: rgb(4, 12, 11);
  401. }
  402. .el-table__cell.is-hidden>* {
  403. visibility: inherit;
  404. }
  405. }
  406. }
  407. </style>