TheEchartPanel.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  1. <script setup>
  2. import { ref, computed, onMounted, unref, onUnmounted } from 'vue';
  3. import { NTabs, NTab, NSelect, NDatePicker } 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 modelValue = defineModel("height");
  9. let echart = null;
  10. let tempTabItemOneKey = 0;
  11. let tempTabItemTwoKey = 'jzxs';
  12. const datePickerValue = ref(null);
  13. const dateRangeRef = ref(null);
  14. const tabs = ref([]);
  15. const tabActive = ref(null);
  16. const selectValue = ref(0);
  17. const coefficientDataSource = ref([]);
  18. const echartDataSource = ref({});
  19. const echartRef = ref(null);
  20. const isEmpty = ref(false);
  21. const activeIndex = ref(0);
  22. const tabList = ['水质', '系数'];
  23. const selectOptions = ref([]);
  24. const echartOptions = [
  25. { label: "进水流量", value: 0, style: "font-size: 12px" },
  26. { label: "#1好氧池硝酸盐", value: 1, style: "font-size: 12px" },
  27. { label: "#2好氧池硝酸盐", value: 2, style: "font-size: 12px" },
  28. { label: "#1缺氧池氨氮", value: 3, style: "font-size: 12px" },
  29. { label: "#2缺氧池氨氮", value: 4, style: "font-size: 12px" },
  30. { label: "进水COD", value: 5, style: "font-size: 12px" },
  31. { label: "进水总氮", value: 6, style: "font-size: 12px" },
  32. { label: "碳源投加量", value: 7, style: "font-size: 12px" }
  33. ]
  34. const coefficientOptions = [
  35. { label: "基准系数", value: 'jzxs', style: "font-size: 12px" },
  36. { label: "修正系数", value: 'xzxs', style: "font-size: 12px" },
  37. { label: "水量分配系数", value: 'slfpxs', style: "font-size: 12px" },
  38. { label: "碳源当量", value: 'tydl', style: "font-size: 12px" },
  39. { label: "转换系数", value: 'zhxs', style: "font-size: 12px" },
  40. { label: "稀释倍数", value: 'sxps', style: "font-size: 12px" },
  41. { label: "密度", value: 'yymd', style: "font-size: 12px" },
  42. ]
  43. const seriesName = computed(() => {
  44. let name = '';
  45. if ( activeIndex.value === 0) {
  46. name = echartOptions.find(({ value }) => selectValue.value === value).label
  47. } else {
  48. name = coefficientOptions.find(item => item.value === selectValue.value).label
  49. }
  50. return name
  51. })
  52. // 切换tab选项
  53. const handleSwitchTab = (index) => {
  54. if ( activeIndex.value === index ) return;
  55. activeIndex.value = index;
  56. if ( !index ) {
  57. // echart
  58. tempTabItemTwoKey = selectValue.value;
  59. selectValue.value = tempTabItemOneKey;
  60. selectOptions.value = echartOptions;
  61. datePickerValue.value = null;
  62. initWaterEchartData();
  63. } else {
  64. // 系数
  65. tempTabItemOneKey = selectValue.value;
  66. selectValue.value = tempTabItemTwoKey;
  67. selectOptions.value = coefficientOptions;
  68. datePickerValue.value = null;
  69. intiCoefficientEchartData();
  70. }
  71. }
  72. // select option change
  73. const handleSelectOptions = (val) => {
  74. selectValue.value = val;
  75. activeIndex.value === 0 ? initWaterEchartData() : intiCoefficientEchartData();
  76. }
  77. const windowResize = () => echart.resize();
  78. const getEchartOptions = (data, type) => {
  79. const option = {
  80. backgroundColor: '#FFF',
  81. title: {
  82. show: !data.length,
  83. text: '暂无数据',
  84. x: 'center',
  85. y: 'center',
  86. textStyle: {
  87. fontSize: 14,
  88. fontWeight: 'normal',
  89. }
  90. },
  91. grid: {
  92. top: '40px',
  93. bottom: '50px',
  94. left: '6%',
  95. right: '6%',
  96. },
  97. tooltip: {
  98. trigger: 'axis',
  99. label: {
  100. show: true
  101. },
  102. },
  103. xAxis: {
  104. boundaryGap: false,
  105. axisLine: {
  106. show: false
  107. },
  108. splitLine: {
  109. show: false
  110. },
  111. axisTick: {
  112. show: false,
  113. alignWithLabel: true
  114. },
  115. axisLabel: {
  116. formatter: function (value) {
  117. return type ? dayjs(value).format('YYYY/MM/DD') : value
  118. }
  119. },
  120. data: data.map(({ time }) => time)
  121. },
  122. yAxis: {
  123. axisLine: {
  124. show: false
  125. },
  126. splitLine: {
  127. show: true,
  128. lineStyle: {
  129. type: 'dashed',
  130. color: '#E5E5E5'
  131. }
  132. },
  133. axisTick: {
  134. show: false
  135. },
  136. splitArea: {
  137. show: false,
  138. color: '#fff'
  139. }
  140. },
  141. series: [
  142. {
  143. name: seriesName.value,
  144. showSymbol: false,
  145. smooth: true,
  146. type: 'line',
  147. symbolSize: 10,
  148. lineStyle: {
  149. color: '#17a6fa',
  150. shadowBlur: 12,
  151. shadowColor: 'rgba(0, 0, 0, 0.12)',
  152. shadowOffsetX: 0,
  153. shadowOffsetY: 4,
  154. width: 2,
  155. },
  156. itemStyle: {
  157. color: '#4080FF',
  158. borderWidth: 3,
  159. borderColor: '#4080FF'
  160. },
  161. areaStyle: {
  162. color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{
  163. offset: 0,
  164. color: 'rgba(0, 136, 212, 0.2)'
  165. }, {
  166. offset: 1,
  167. color: 'rgba(0, 136, 212, 0)'
  168. }], false),
  169. },
  170. data: data.map(({ val }) => val)
  171. }
  172. ]
  173. };
  174. return option;
  175. }
  176. const onSwitchEchart = (item) => {
  177. const echartData = echartDataSource.value[item.value];
  178. isEmpty.value = !!echartData.length
  179. echart.setOption(getEchartOptions(echartData));
  180. }
  181. // 水务相关数据格式化
  182. const initWaterEchartData = async () => {
  183. const [tBegin, tEnd] = datePickerValue.value || [];
  184. const timeBegin = tBegin ? dayjs(tBegin).format('YYYY/MM/DD') : null;
  185. const timeEnd = tEnd ? dayjs(tEnd).format('YYYY/MM/DD') : null;
  186. const { data: echartData } = await controlApi.getEchartData(unref(selectValue), { timeBegin, timeEnd });
  187. const enumSource = {
  188. YB: '在线仪表',
  189. HY: '连续检测',
  190. YC: '预测'
  191. };
  192. tabs.value = Object.keys(echartData).map((key, index) => {
  193. if (index === 0) {
  194. tabActive.value = key + '-' + selectValue.value + '-' + index;
  195. }
  196. if (echartData[key].length) {
  197. return ({ label: enumSource[key], value: key });
  198. }
  199. }).filter(Boolean);
  200. echartDataSource.value = echartData;
  201. onSwitchEchart({ value: tabActive.value.substring(0, tabActive.value.indexOf('-')) });
  202. }
  203. // 系数相关数据
  204. const intiCoefficientEchartData = async () => {
  205. const [timeBegin, timeEnd] = datePickerValue.value || [];
  206. const { data } = await controlApi.getEchartList({ timeBegin, timeEnd });
  207. coefficientDataSource.value = data;
  208. const d = data.map(item => ({
  209. time: dayjs(item.createTime).format('YYYY/MM/DD HH'),
  210. val: item[selectValue.value]
  211. }));
  212. echart.setOption(getEchartOptions(d));
  213. }
  214. // 日期范围限制
  215. const isRangeDateDisabled = (ts, type, range) => {
  216. const d = 864e5;
  217. if (type === "start" && range !== null) {
  218. return startOfDay(range[1]).valueOf() - startOfDay(ts).valueOf() >= d * 10;
  219. }
  220. if (type === "end" && range !== null) {
  221. return startOfDay(ts).valueOf() - startOfDay(range[0]).valueOf() >= d * 10;
  222. }
  223. return false;
  224. }
  225. const onDatePickerConfirm = (ts) => {
  226. datePickerValue.value = ts.map(t => dayjs(t).format('YYYY-MM-DD'));
  227. activeIndex.value === 0 ? initWaterEchartData() : intiCoefficientEchartData();
  228. }
  229. const onDatePickerClear = () => {
  230. datePickerValue.value = null;
  231. activeIndex.value === 0 ? initWaterEchartData() : intiCoefficientEchartData();
  232. }
  233. onMounted(async () => {
  234. selectOptions.value = echartOptions;
  235. echart = echarts.init(echartRef.value, 'light');
  236. await initWaterEchartData();
  237. window.addEventListener("resize", windowResize);
  238. })
  239. onUnmounted(() => {
  240. window.removeEventListener("resize", windowResize);
  241. echart && echart.dispose();
  242. })
  243. </script>
  244. <template>
  245. <div class="echart-card_view">
  246. <div class="title">
  247. <div class="left-inner">
  248. <span class="text">数据看板</span>
  249. </div>
  250. <div class="right-inner">
  251. <ul class="custom-radio-group">
  252. <li :class="{ active: activeIndex === index }" v-for="item, index in tabList" :key="item"
  253. @click="handleSwitchTab(index)">{{ item }}</li>
  254. </ul>
  255. </div>
  256. </div>
  257. <div class="select-wrapper">
  258. <div>
  259. <n-tabs
  260. animated
  261. type="segment"
  262. size="small"
  263. class="tabs"
  264. style="width: 200px;"
  265. v-model:value="tabActive"
  266. v-show="activeIndex === 0"
  267. >
  268. <n-tab
  269. v-for="item, index in tabs"
  270. :key="item.value"
  271. :name="item.value + '-' + selectValue + '-' + index"
  272. @click="onSwitchEchart(item)"
  273. >{{ item.label }}</n-tab>
  274. </n-tabs>
  275. </div>
  276. <div class="flex space-x-[10px]">
  277. <NDatePicker
  278. clearable
  279. class="w-[300px]"
  280. size="small"
  281. type="daterange"
  282. ref="dateRangeRef"
  283. :is-date-disabled="isRangeDateDisabled"
  284. :on-confirm="onDatePickerConfirm"
  285. :on-clear="onDatePickerClear"
  286. v-model:formatted-value="datePickerValue"
  287. ></NDatePicker>
  288. <!-- v-model:formatted-value="datePickerValue" -->
  289. <!-- -->
  290. <!-- v-model:value="selectValue" -->
  291. <NSelect
  292. class="w-[150px]"
  293. :options="selectOptions"
  294. :value="selectValue"
  295. :on-update:value="handleSelectOptions"
  296. size="small" />
  297. </div>
  298. </div>
  299. <div class="echart-wrapper">
  300. <div class="echart" ref="echartRef"></div>
  301. </div>
  302. </div>
  303. </template>
  304. <style lang="scss" scoped>
  305. .echart-card_view {
  306. display: flex;
  307. flex-flow: column;
  308. height: calc(100% - 256px);
  309. padding: 0px 16px 0 25px;
  310. border-radius: 10px;
  311. .title {
  312. flex-shrink: 0;
  313. @include flex(x, center, between);
  314. padding-bottom: 16px;
  315. .left-inner {
  316. @include flex(x, center, start);
  317. .text {
  318. color: #1A2029;
  319. font-size: 15px;
  320. font-style: normal;
  321. font-weight: 500;
  322. line-height: 24px;
  323. }
  324. .tabs {
  325. width: 240px;
  326. margin-left: 16px;
  327. }
  328. }
  329. .right-inner {
  330. @include flex(x, center, start);
  331. .custom-radio-group {
  332. @include flex(x, center, center);
  333. width: 104px;
  334. height: 24px;
  335. border-radius: 4px;
  336. border: 1px solid #D3D7DD;
  337. li {
  338. width: 50%;
  339. font-size: 12px;
  340. text-align: center;
  341. line-height: 24px;
  342. color: #333;
  343. cursor: pointer;
  344. &:nth-child(1) {
  345. border-right: 1px solid #D3D7DD;
  346. }
  347. }
  348. li.active {
  349. color: #2454FF;
  350. }
  351. }
  352. }
  353. }
  354. .select-wrapper {
  355. @include flex(x, center, between);
  356. height: 32px;
  357. }
  358. .echart-wrapper {
  359. height: calc(100% - 72px);
  360. .echart,
  361. .empty {
  362. width: 100%;
  363. height: 100%;
  364. }
  365. .empty {
  366. @include flex(x, center, center);
  367. }
  368. }
  369. }
  370. </style>
  371. <style lang="scss">
  372. .echart-card_view {
  373. .tabs {
  374. .n-tabs-tab--active {
  375. .n-tabs-tab__label {
  376. color: #2454FF;
  377. }
  378. }
  379. .n-tabs-tab__label {
  380. font-size: 12px;
  381. color: #333333;
  382. }
  383. }
  384. .n-base-selection .n-base-selection-label .n-base-selection-input {
  385. font-size: 12px;
  386. color: #333333 !important;
  387. }
  388. .right-inner {
  389. .n-base-selection__border {
  390. border: 0;
  391. }
  392. .n-base-selection-label {
  393. border: 0;
  394. border-radius: 0;
  395. background: #F2F3F5 !important;
  396. }
  397. .n-base-selection__state-border,
  398. .n-base-selection__border {
  399. border: 0 !important;
  400. }
  401. .n-base-selection .n-base-selection__border,
  402. .n-base-selection .n-base-selection__state-border {
  403. box-shadow: none;
  404. }
  405. .n-base-suffix__arrow {
  406. color: #4E5969;
  407. }
  408. }
  409. .n-input--pair {
  410. background: #f0f1f3 !important;
  411. }
  412. .n-base-selection-input__content {
  413. font-size: 12px;
  414. }
  415. .n-base-selection {
  416. background: #f0f1f3 !important;
  417. }
  418. .n-base-selection-label {
  419. background: #f0f1f3;
  420. border-radius: 3px;
  421. }
  422. .n-base-selection__border,
  423. .n-base-selection__state-border {
  424. display: none !important;
  425. }
  426. .n-date-picker {
  427. .n-input__input-el,
  428. .n-input__placeholder {
  429. font-size: 12px !important;
  430. }
  431. }
  432. .n-base-selection:not(.n-base-selection--disabled).n-base-selection--active .n-base-selection-label {
  433. background: #f0f1f3 !important;
  434. }
  435. }
  436. </style>