TheEchartPanel.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719
  1. <script setup>
  2. import { ref, computed, onMounted, watch, onUnmounted } from 'vue';
  3. import { NSpin, NSelect, NDatePicker, useMessage } from "naive-ui";
  4. import * as echarts from 'echarts';
  5. import { startOfDay } from "date-fns/esm"
  6. import { controlApi } from "@/api/control"
  7. import dayjs from 'dayjs';
  8. const props = defineProps(['htfksdOne','htfksdTwo']);
  9. const isDomSizeChange = defineModel('change');
  10. const message = useMessage();
  11. let echart = null;
  12. let tempTabItemOneKey = 0;
  13. let tempTabItemTwoKey = 'jzxsOne';
  14. const currentDay = dayjs().format('YYYY-MM-DD');
  15. const yesterday = dayjs().subtract(1, 'day').format('YYYY-MM-DD');
  16. const datePickerValue = ref([yesterday, currentDay]);
  17. const dateRangeRef = ref(null);
  18. const selectValue = ref(['0']);
  19. const coefficientDataSource = ref([]);
  20. const echartDataSource = ref({});
  21. const echartRef = ref(null);
  22. const activeIndex = ref(0);
  23. const show = ref(true);
  24. const tabList = ['水质', '系数'];
  25. const selectOptions = ref([]);
  26. let selectEnum = new Map([
  27. ['0', '进水流量'],
  28. ['1', '北池-好氧池硝酸盐'],
  29. ['2', '南池-好氧池硝酸盐'],
  30. ['22', '北池-后反馈设定'],
  31. ['23', '南池-后反馈设定'],
  32. ['12', '北池-缺氧硝酸盐'],
  33. ['13', '南池-缺氧硝酸盐'],
  34. ['3', '北池-缺氧池氨氮'],
  35. ['4', '南池-缺氧池氨氮'],
  36. ['5', '进水COD-连续检测'],
  37. ['6', '进水COD-在线仪表'],
  38. ['7', '进水总氮'],
  39. ['8', '碳源投加量-北池-计算投药量'],
  40. ['9', '碳源投加量-南池-计算投药量'],
  41. ['10', '碳源投加量-北池-反馈流量'],
  42. ['11', '碳源投加量-南池-反馈流量'],
  43. ])
  44. let echartOptions = [
  45. { label: "进水流量", value: 0, style: "font-size: 12px" },
  46. { label: "#1好氧池硝酸盐", value: 1, style: "font-size: 12px" },
  47. { label: "#2好氧池硝酸盐", value: 2, style: "font-size: 12px" },
  48. { label: "#1缺氧池氨氮", value: 3, style: "font-size: 12px" },
  49. { label: "#2缺氧池氨氮", value: 4, style: "font-size: 12px" },
  50. { label: "进水COD", value: 5, style: "font-size: 12px" },
  51. { label: "进水总氮", value: 6, style: "font-size: 12px" },
  52. { label: "碳源投加量", value: 7, style: "font-size: 12px" }
  53. ]
  54. const coefficientOptions = [
  55. { label: "基准系数", value: 'jzxsOne', style: "font-size: 12px" },
  56. { label: "修正系数", value: 'xzxsOne', style: "font-size: 12px" },
  57. { label: "水量分配系数", value: 'slfpxsOne', style: "font-size: 12px" },
  58. { label: "碳源当量", value: 'tydlOne', style: "font-size: 12px" },
  59. { label: "转换系数", value: 'zhxsOne', style: "font-size: 12px" },
  60. { label: "稀释倍数", value: 'sxpsOne', style: "font-size: 12px" },
  61. { label: "密度", value: 'yymdOne', style: "font-size: 12px" },
  62. ]
  63. const selectThemeOverrides = {
  64. peers: {
  65. InternalSelection: {
  66. borderRadius: '4px'
  67. }
  68. },
  69. }
  70. const seriesName = computed(() => {
  71. let name = '';
  72. if ( activeIndex.value === 0) {
  73. name = echartOptions.find(({ value }) => selectValue.value === value).label
  74. } else {
  75. name = coefficientOptions.find(item => item.value === selectValue.value).label
  76. }
  77. return name
  78. })
  79. // 切换tab选项
  80. const handleSwitchTab = (index) => {
  81. if ( activeIndex.value === index ) return;
  82. activeIndex.value = index;
  83. if ( !index ) {
  84. // echart
  85. tempTabItemTwoKey = selectValue.value;
  86. selectValue.value = tempTabItemOneKey;
  87. selectOptions.value = echartOptions;
  88. datePickerValue.value = null;
  89. initWaterEchart();
  90. } else {
  91. // 系数
  92. tempTabItemOneKey = selectValue.value;
  93. selectValue.value = tempTabItemTwoKey;
  94. selectOptions.value = coefficientOptions;
  95. datePickerValue.value = null;
  96. intiCoefficientEchartData();
  97. }
  98. }
  99. // select option change
  100. const handleSelectOptions = (selectOptionList) => {
  101. let tempArr = selectOptionList;
  102. if ( activeIndex.value === 0 ) {
  103. if ( selectOptionList.length <= 1 && selectOptionList.every(item => item > 11) ) {
  104. const [item] = selectOptionList;
  105. if ( (item == 22 || item == 23) || (!item && item != 0) ) {
  106. selectOptions.value.forEach(item => {
  107. if ( item.value == 22 || item.value == 23 ) {
  108. item.disabled = true;
  109. }
  110. });
  111. tempArr = [];
  112. }
  113. } else {
  114. selectOptions.value.forEach(item => {
  115. if ( item.value == 22 || item.value == 23 ) {
  116. item.disabled = false;
  117. }
  118. });
  119. }
  120. if ( selectOptionList.length > 3 ) {
  121. selectValue.value = tempArr.slice(0, -1);
  122. return message.warning("数据看板最大选择项为3个")
  123. }
  124. localStorage.setItem('selectValue', JSON.stringify(tempArr));
  125. }
  126. selectValue.value = tempArr;
  127. activeIndex.value === 0 ? initWaterEchart() : intiCoefficientEchartData();
  128. }
  129. const initWaterEchart = async () => {
  130. const tempArr = [];
  131. const tempResult = selectValue.value.map(key => {
  132. if (key == 22 || key == 23) return null;
  133. const data = echartDataSource.value[key] || [];
  134. tempArr.push(data);
  135. return ({
  136. name: selectEnum.get(key),
  137. data
  138. });
  139. }).filter(Boolean);
  140. const xAxis = tempArr.flat(Infinity).reduce((acc, curr) => {
  141. if (!acc.some(item => item.time === curr.time)) {
  142. acc.push(curr);
  143. }
  144. return acc;
  145. }, []).filter(Boolean).map(item => item.time.trim()).sort((a, b) => dayjs(a).valueOf() - dayjs(b).valueOf());
  146. const options = getWaterEchartOptions({ data: tempResult, xAxis });
  147. echart.setOption(options, true);
  148. }
  149. const windowResize = () => echart.resize();
  150. const getWaterEchartOptions = ({ data, xAxis = [] }) => {
  151. const series = data.map(item => {
  152. const d1 = item.data.map(item => {
  153. return [
  154. item.time.trim(),
  155. item.val ? item.val.toFixed(2) : 0
  156. ]
  157. });
  158. return {
  159. name: item.name,
  160. showSymbol: false,
  161. smooth: true,
  162. type: 'line',
  163. symbolSize: 10,
  164. data: d1,
  165. }
  166. })
  167. if ( selectValue.value.includes('22') ) {
  168. const data = xAxis.map(time => [time, props.htfksdOne]);
  169. series.push({
  170. name: '北池-后反馈设定',
  171. showSymbol: false,
  172. smooth: true,
  173. type: 'line',
  174. symbolSize: 10,
  175. lineStyle: {
  176. color: 'red'
  177. },
  178. itemStyle: {
  179. color: 'red'
  180. },
  181. data
  182. });
  183. }
  184. if (selectValue.value.includes('13')) {
  185. const data = xAxis.map(time => [time, props.htfksdTwo]);
  186. series.push({
  187. name: '南池-后反馈设定',
  188. showSymbol: false,
  189. smooth: true,
  190. type: 'line',
  191. symbolSize: 10,
  192. lineStyle: {
  193. color: 'red'
  194. },
  195. itemStyle: {
  196. color: 'red'
  197. },
  198. data
  199. });
  200. }
  201. const option = {
  202. backgroundColor: '#FFF',
  203. legend: {
  204. x: 'center',
  205. y: 'top',
  206. show: true,
  207. left: '10px',
  208. top: '16px',
  209. itemWidth: 6,
  210. itemGap: 20,
  211. textStyle: {
  212. color: '#556677',
  213. },
  214. // data: ['直接登录平台', '扫码登录平台', '总'],
  215. },
  216. title: {
  217. show: !xAxis.length,
  218. text: '暂无数据',
  219. x: 'center',
  220. y: 'center',
  221. textStyle: {
  222. fontSize: 14,
  223. fontWeight: 'normal',
  224. }
  225. },
  226. grid: {
  227. top: '60px',
  228. bottom: '50px',
  229. left: '5%',
  230. right: '5%',
  231. },
  232. tooltip: {
  233. trigger: 'axis',
  234. label: {
  235. show: true
  236. },
  237. },
  238. xAxis: {
  239. type: 'time',
  240. boundaryGap: ['5%', '5%'],
  241. axisLine: {
  242. show: false
  243. },
  244. splitLine: {
  245. show: false
  246. },
  247. axisTick: {
  248. show: false,
  249. // alignWithLabel: true
  250. },
  251. axisLabel: {
  252. // margin: 10,
  253. // showMaxLabel: true,
  254. // rotate: 1,
  255. formatter: function (value) {
  256. return dayjs(value).format('YYYY/MM/DD')
  257. }
  258. },
  259. // data: xAxis
  260. },
  261. yAxis: {
  262. axisLine: {
  263. show: false
  264. },
  265. splitLine: {
  266. show: true,
  267. lineStyle: {
  268. type: 'dashed',
  269. color: '#E5E5E5'
  270. }
  271. },
  272. axisTick: {
  273. show: false
  274. },
  275. splitArea: {
  276. show: false,
  277. color: '#fff'
  278. },
  279. axisLabel: {
  280. formatter: function (value) {
  281. return value.toFixed(0)
  282. }
  283. }
  284. },
  285. series
  286. };
  287. return option;
  288. }
  289. const getEchartOptions = (data) => {
  290. const option = {
  291. backgroundColor: '#FFF',
  292. title: {
  293. show: !data.length,
  294. text: '暂无数据',
  295. x: 'center',
  296. y: 'center',
  297. textStyle: {
  298. fontSize: 14,
  299. fontWeight: 'normal',
  300. }
  301. },
  302. grid: {
  303. top: '40px',
  304. bottom: '50px',
  305. left: '5%',
  306. right: '5%',
  307. },
  308. tooltip: {
  309. trigger: 'axis',
  310. label: {
  311. show: true
  312. },
  313. },
  314. xAxis: {
  315. boundaryGap: false,
  316. axisLine: {
  317. show: false
  318. },
  319. splitLine: {
  320. show: false
  321. },
  322. axisTick: {
  323. show: false,
  324. alignWithLabel: true
  325. },
  326. axisLabel: {
  327. formatter: function (value) {
  328. return dayjs(value).format('YYYY/MM/DD')
  329. }
  330. },
  331. data: data.map(({ time }) => time)
  332. },
  333. yAxis: {
  334. axisLine: {
  335. show: false
  336. },
  337. splitLine: {
  338. show: true,
  339. lineStyle: {
  340. type: 'dashed',
  341. color: '#E5E5E5'
  342. }
  343. },
  344. axisTick: {
  345. show: false
  346. },
  347. splitArea: {
  348. show: false,
  349. color: '#fff'
  350. }
  351. },
  352. series: [
  353. {
  354. name: seriesName.value,
  355. showSymbol: false,
  356. smooth: true,
  357. type: 'line',
  358. symbolSize: 10,
  359. lineStyle: {
  360. color: '#17a6fa',
  361. shadowBlur: 12,
  362. shadowColor: 'rgba(0, 0, 0, 0.12)',
  363. shadowOffsetX: 0,
  364. shadowOffsetY: 4,
  365. width: 2,
  366. },
  367. itemStyle: {
  368. color: '#4080FF',
  369. borderWidth: 3,
  370. borderColor: '#4080FF'
  371. },
  372. areaStyle: {
  373. color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{
  374. offset: 0,
  375. color: 'rgba(0, 136, 212, 0.2)'
  376. }, {
  377. offset: 1,
  378. color: 'rgba(0, 136, 212, 0)'
  379. }], false),
  380. },
  381. data: data.map(({ val }) => val)
  382. }
  383. ]
  384. };
  385. return option;
  386. }
  387. // 水务相关数据格式化
  388. const initWaterEchartData = async () => {
  389. show.value = true;
  390. const [tBegin, tEnd] = datePickerValue.value || [];
  391. const timeBegin = tBegin ? dayjs(tBegin).format('YYYY-MM-DD') : null;
  392. const timeEnd = tEnd ? dayjs(tEnd).format('YYYY-MM-DD') : null;
  393. const { data } = await controlApi.getAllEchartData({ timeBegin, timeEnd });
  394. show.value = false;
  395. echartDataSource.value = data;
  396. initWaterEchart();
  397. }
  398. // 系数相关数据
  399. const intiCoefficientEchartData = async () => {
  400. show.value = true;
  401. const [timeBegin, timeEnd] = datePickerValue.value || [];
  402. const { data } = await controlApi.getBoardEchartList({ timeBegin, timeEnd });
  403. coefficientDataSource.value = data;
  404. const d = data.map(item => ({
  405. time: dayjs(item.createTime).format('YYYY/MM/DD HH'),
  406. val: item[selectValue.value]
  407. }));
  408. show.value = false;
  409. echart.setOption(getEchartOptions(d), true);
  410. }
  411. // 日期范围限制
  412. const isRangeDateDisabled = (ts, type, range) => {
  413. const d = 864e5;
  414. if (type === "start" && range !== null) {
  415. return startOfDay(range[1]).valueOf() - startOfDay(ts).valueOf() >= d * 10;
  416. }
  417. if (type === "end" && range !== null) {
  418. return startOfDay(ts).valueOf() - startOfDay(range[0]).valueOf() >= d * 10;
  419. }
  420. return false;
  421. }
  422. const onDatePickerConfirm = (ts) => {
  423. datePickerValue.value = ts.map(t => dayjs(t).format('YYYY-MM-DD'));
  424. activeIndex.value === 0 ? initWaterEchartData() : intiCoefficientEchartData();
  425. }
  426. const onDatePickerClear = () => {
  427. datePickerValue.value = null;
  428. activeIndex.value === 0 ? initWaterEchartData() : intiCoefficientEchartData();
  429. }
  430. watch(() => isDomSizeChange.value, (val) => {
  431. setTimeout(() => windowResize(), 200)
  432. });
  433. onMounted(async () => {
  434. const localCacheSelectVal = localStorage.getItem('selectValue');
  435. let tempArr = [];
  436. selectEnum.forEach(function(value, key) {
  437. tempArr.push({ label: value, value: key, style: "font-size: 12px"});
  438. })
  439. echartOptions = tempArr;
  440. selectValue.value = localCacheSelectVal ? JSON.parse(localCacheSelectVal) : ["0"]
  441. selectOptions.value = echartOptions;
  442. echart = echarts.init(document.querySelector('#echartRef'), 'light');
  443. setTimeout(async () => {
  444. await initWaterEchartData();
  445. })
  446. window.addEventListener("resize", windowResize);
  447. })
  448. onUnmounted(() => {
  449. window.removeEventListener("resize", windowResize);
  450. echart && echart.dispose();
  451. })
  452. </script>
  453. <template>
  454. <div class="echart-card_view">
  455. <div class="title">
  456. <div class="left-inner">
  457. <span class="text">数据看板</span>
  458. </div>
  459. <div class="right-inner">
  460. <ul class="custom-radio-group">
  461. <li :class="{ active: activeIndex === index }" v-for="item, index in tabList" :key="item"
  462. @click="handleSwitchTab(index)">{{ item }}</li>
  463. </ul>
  464. </div>
  465. </div>
  466. <div class="select-wrapper">
  467. <NDatePicker
  468. clearable
  469. class="w-[300px] flex-shrink-0"
  470. size="small"
  471. type="daterange"
  472. ref="dateRangeRef"
  473. :is-date-disabled="isRangeDateDisabled"
  474. :on-confirm="onDatePickerConfirm"
  475. :on-clear="onDatePickerClear"
  476. v-model:formatted-value="datePickerValue"
  477. ></NDatePicker>
  478. <NSelect
  479. class="w-[340px]"
  480. :multiple="activeIndex != 1"
  481. :options="selectOptions"
  482. :value="selectValue"
  483. :on-update:value="handleSelectOptions"
  484. :theme-overrides="selectThemeOverrides"
  485. size="small"
  486. max-tag-count="responsive"
  487. />
  488. </div>
  489. <div class="echart-wrapper">
  490. <div class="echart w-full h-full" ref="echartRef" id="echartRef" :style="{ zIndex: show ? 0 : 100 }"></div>
  491. <n-spin :show="show" class="h-full w-full" :style="{position: 'absolute', top: '0%', left: '0%', zIndex: show ? 100 : 0}">
  492. <span></span>
  493. </n-spin>
  494. </div>
  495. </div>
  496. </template>
  497. <style lang="scss" scoped>
  498. .echart-card_view {
  499. flex: 1;
  500. display: flex;
  501. flex-flow: column;
  502. // height: calc(100% - 284px);
  503. padding: 0px 16px 0 25px;
  504. border-radius: 10px;
  505. .title {
  506. flex-shrink: 0;
  507. @include flex(x, center, between);
  508. padding-bottom: 16px;
  509. .left-inner {
  510. @include flex(x, center, start);
  511. .text {
  512. color: #1A2029;
  513. font-size: 15px;
  514. font-style: normal;
  515. font-weight: 500;
  516. line-height: 24px;
  517. }
  518. .tabs {
  519. width: 240px;
  520. margin-left: 16px;
  521. }
  522. }
  523. .right-inner {
  524. @include flex(x, center, start);
  525. .custom-radio-group {
  526. @include flex(x, center, center);
  527. width: 104px;
  528. height: 24px;
  529. border-radius: 4px;
  530. border: 1px solid #D3D7DD;
  531. li {
  532. width: 50%;
  533. font-size: 12px;
  534. text-align: center;
  535. line-height: 24px;
  536. color: #333;
  537. cursor: pointer;
  538. &:nth-child(1) {
  539. border-right: 1px solid #D3D7DD;
  540. }
  541. }
  542. li.active {
  543. color: #2454FF;
  544. }
  545. }
  546. }
  547. }
  548. .select-wrapper {
  549. @include flex(x, center, between);
  550. height: 32px;
  551. }
  552. .echart-wrapper {
  553. position: relative;
  554. height: calc(100% - 72px);
  555. .echart {
  556. position: relative;
  557. z-index: 10;
  558. }
  559. .echart,
  560. .empty {
  561. width: 100%;
  562. height: 100%;
  563. }
  564. .empty {
  565. @include flex(x, center, center);
  566. }
  567. }
  568. }
  569. </style>
  570. <style lang="scss">
  571. .echart-card_view {
  572. .tabs {
  573. .n-tabs-tab--active {
  574. .n-tabs-tab__label {
  575. color: #2454FF;
  576. }
  577. }
  578. .n-tabs-tab__label {
  579. font-size: 12px;
  580. color: #333333;
  581. }
  582. }
  583. // .n-base-selection .n-base-selection-label .n-base-selection-input {
  584. // font-size: 12px;
  585. // color: #333333 !important;
  586. // }
  587. // .right-inner {
  588. // .n-base-selection__border {
  589. // border: 0;
  590. // }
  591. // .n-base-selection-label {
  592. // border: 0;
  593. // border-radius: 0;
  594. // background: #F2F3F5 !important;
  595. // }
  596. // .n-base-selection__state-border,
  597. // .n-base-selection__border {
  598. // border: 0 !important;
  599. // }
  600. // .n-base-selection .n-base-selection__border,
  601. // .n-base-selection .n-base-selection__state-border {
  602. // box-shadow: none;
  603. // }
  604. // .n-base-suffix__arrow {
  605. // color: #4E5969;
  606. // }
  607. // }
  608. .n-input--pair {
  609. background: #f0f1f3 !important;
  610. }
  611. // .n-base-selection-input__content {
  612. // font-size: 12px;
  613. // }
  614. // .n-base-selection {
  615. // background: #f0f1f3 !important;
  616. // }
  617. // .n-base-selection-label {
  618. // background: #f0f1f3;
  619. // border-radius: 3px;
  620. // }
  621. // .n-base-selection__border,
  622. // .n-base-selection__state-border {
  623. // display: none !important;
  624. // }
  625. // .n-date-picker {
  626. // .n-input__input-el,
  627. // .n-input__placeholder {
  628. // font-size: 12px !important;
  629. // }
  630. // }
  631. // .n-base-selection:not(.n-base-selection--disabled).n-base-selection--active .n-base-selection-label {
  632. // background: #f0f1f3 !important;
  633. // }
  634. }
  635. </style>