index.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. <script setup>
  2. import { ref, computed, onMounted, onUnmounted } from 'vue';
  3. import { useRouter } from 'vue-router';
  4. import { NPopselect, useMessage } from 'naive-ui';
  5. import * as echarts from 'echarts';
  6. import dayjs from 'dayjs';
  7. import autofit from 'autofit.js'
  8. import { carbonApi } from '@/api/carbon'
  9. import { echartLineOptions, echartGraphOptions, echart3DOptions, echartColumnarOptions } from './echartOptions';
  10. import SvgIcon from '@/components/SvgIcon';
  11. import { TheDrawerForm, TheDrawerList, TheModalTable } from './components';
  12. import DataBorad from './components/screen/DataBorad.vue'
  13. import DataScroll from './components/screen/DataScroll.vue'
  14. import LayoutCard from './components/screen/LayoutCard.vue'
  15. const router = useRouter();
  16. const message = useMessage();
  17. let timerId = null;
  18. let echartMonth = null;
  19. let echartSewage = null;
  20. let echartEnergy = null;
  21. let echartMdu = null;
  22. const tDay = dayjs();
  23. const currentTimer = ref(tDay.format('HH:mm:ss'));
  24. const currentMonth = tDay.format('YYYY.MM.DD');
  25. const allMonth = ref([]);
  26. const inpVal = ref('')
  27. const historyDetails = ref({});
  28. const echartMonthRef = ref(null);
  29. const echartSewageRef = ref(null);
  30. const echartEnergyRef = ref(null);
  31. const echartMudRef = ref(null);
  32. const isVisibleDrawerForm = ref(false);
  33. const isVisibleDrawerList = ref(false);
  34. const isVisibleDrawerTable = ref(false);
  35. const scrollData = ref([]);
  36. const scrollMapping = [
  37. { label: '化石源的CO₂排放量', key: 'wsHsytkhCo2', value: '' },
  38. { label: '污水处理阶段排放的CH₄的CO₂当量', key: 'wsSjclCh4Co2', value: '' },
  39. { label: '污水处理阶段排放的N₂O的CO₂当量', key: 'wsTdN2oCo2', value: '' },
  40. { label: '污水厂电力消耗产生的总碳排放量', key: 'nyyjDlxhZhdlCo2', value: '' },
  41. { label: '进水泵房碳排', key: 'nyyjDlxhJsbfCo2', value: '' },
  42. { label: '鼓风机房碳排', key: 'nyyjDlxhGfjfCo2', value: '' },
  43. { label: '脱水机房碳排', key: 'nyyjDlxhTsjfCo2', value: '' },
  44. { label: '消毒间碳排', key: 'nyyjDlxhXdjCo2', value: '' },
  45. { label: '深度处理用电碳排', key: 'nyyjDlxhSdclCo2', value: '' },
  46. { label: '污泥处置用电碳排', key: 'nyyjDlxhWnczCo2', value: '' },
  47. { label: '生活区碳排', key: 'nyyjDlxhShqCo2', value: '' },
  48. { label: '其他碳排', key: 'nyyjDlxhQtCo2', value: '' },
  49. { label: '燃料消耗产生的碳排放量', key: 'nyyjGrrlCo2', value: '' },
  50. { label: '污泥厌氧消化过程排放的CH₄的CO₂当量', key: 'wnclYyzqCo2', value: '' },
  51. { label: '单独处理污泥厌氧消化产生沼液的CO₂当量', key: 'wnclDdclCo2', value: '' },
  52. { label: '污泥好氧发酵过程排放的N₂O的CO₂当量', key: 'wnclHyfjN2oCo2', value: '' },
  53. { label: '污泥好氧发酵过程排放的CH₄的CO₂当量', key: 'wnclWnfsCo2', value: '' },
  54. { label: '污泥焚烧过程排放的CH₄的CO₂当量', key: 'wnclBwqrsCh4Co2', value: '' },
  55. { label: '污泥焚烧过程排放的N₂O的CO₂当量', key: 'wnclBwqrsN2oCo2', value: '' },
  56. { label: '污泥热解过程排放的CH₄的CO₂当量', key: 'wnclWnrjByhCo2', value: '' },
  57. { label: '污泥热解过程排放的N₂O的CO₂当量', key: 'wnclWnrjQtN2oCo2', value: '' },
  58. { label: '光伏发电替碳量', key: 'thGfCo2', value: '' },
  59. { label: '回用水替代供水的替碳量', key: 'thZssCo2', value: '' },
  60. { label: '污泥产物土地利用的替碳量', key: 'thWnClhCo2', value: '' },
  61. ];
  62. const selectOptions = computed(() => allMonth.value.map(item => ({ label: item, value: item })));
  63. const handleType = async ( type ) => {
  64. if (type === 'down') {
  65. return message.warning("该功能暂未开通");
  66. }
  67. if (type === 'info') {
  68. isVisibleDrawerTable.value = true;
  69. }
  70. if (type === 'list') {
  71. isVisibleDrawerList.value = true;
  72. }
  73. if (type === 'add') {
  74. isVisibleDrawerForm.value = true;
  75. }
  76. }
  77. const handleBackPage = () => router.back();
  78. const getCurrentTimer = () => dayjs().format('HH:mm:ss');
  79. const initAllMonth = async () => {
  80. const { data: month } = await carbonApi.getAllMonth();
  81. allMonth.value = month;
  82. return month;
  83. }
  84. const initEchart = () => {
  85. const {
  86. recent7Datas, extraMap, wsTdN2oCo2, wsSjclCh4Co2, wjtykuCo2, wsHsytkhCo2,
  87. wnclYyzqCo2,
  88. wnclDdclCo2,
  89. wnclHyfjCh4Co2, wnclHyfjN2oCo2,
  90. wnclWnfsCo2,
  91. wnclBwqrsCh4Co2, wnclBwqrsN2oCo2,
  92. wnclWnrjQtCh4Co2, wnclWnrjQtN2oCo2,
  93. nyyjDlxhZhdlCo = 0, wscljdyjxhCo = 0, gdsnyrlCo2 = 0, nyyjGrrlCo = 0
  94. } = historyDetails.value;
  95. // 近7月碳排放趋势
  96. echartMonth.setOption(echartLineOptions(recent7Datas));
  97. // 污水处理生化反应
  98. echartSewage.setOption(echartGraphOptions([wsTdN2oCo2, wsSjclCh4Co2, wjtykuCo2, wsHsytkhCo2]));
  99. // echartEnergy.dispose();
  100. // 能源、药剂
  101. echartEnergy.setOption(echart3DOptions([nyyjDlxhZhdlCo, wscljdyjxhCo, gdsnyrlCo2, nyyjGrrlCo]));
  102. // 污泥处理-生化反应
  103. echartMdu.setOption(echartColumnarOptions([
  104. wnclYyzqCo2,
  105. wnclDdclCo2,
  106. wnclHyfjCh4Co2 + wnclHyfjN2oCo2,
  107. wnclWnfsCo2,
  108. wnclBwqrsCh4Co2 + wnclBwqrsN2oCo2,
  109. wnclWnrjQtCh4Co2 + wnclWnrjQtN2oCo2
  110. ]));
  111. const scrollList = Object.entries(extraMap).map(([k, v], i) => {
  112. return v.map((child, index) => {
  113. return {
  114. label: child.dictLabel,
  115. key: index + i,
  116. value: child.val
  117. }
  118. })
  119. }).flat(Infinity);
  120. // 滚动区域
  121. scrollData.value = [...scrollList, ...scrollMapping].map(item => {
  122. const temp = { ...item };
  123. Object.entries(historyDetails.value).forEach(([k, v]) => {
  124. if ( temp.key === k ) {
  125. temp.value = v;
  126. }
  127. })
  128. return temp;
  129. }).filter(({ value }) => value);
  130. }
  131. const initPageData = async () => {
  132. const { data } = await carbonApi.getDetails(inpVal.value);
  133. historyDetails.value = data;
  134. echartMonth = echarts.init(echartMonthRef.value, 'light');
  135. echartSewage = echarts.init(echartSewageRef.value, 'light');
  136. echartEnergy = echarts.init(echartEnergyRef.value, 'light');
  137. echartMdu = echarts.init(echartMudRef.value, 'light');
  138. initEchart();
  139. }
  140. // select 的 options选中
  141. const handleSelectChecked = (time) => {
  142. inpVal.value = time;
  143. initPageData();
  144. }
  145. const onSubmit = async () => {
  146. isVisibleDrawerForm.value = false;
  147. if ( !allMonth.value.length ) {
  148. const [ firstMonth ] = await initAllMonth();
  149. inpVal.value = firstMonth;
  150. initPageData();
  151. } else {
  152. initAllMonth();
  153. initPageData();
  154. }
  155. }
  156. const windowResize = () => {
  157. echartMonth.resize();
  158. echartSewage.resize();
  159. echartEnergy.resize();
  160. echartMdu.resize();
  161. };
  162. onMounted(async () => {
  163. const [ firstMonth ] = await initAllMonth();
  164. autofit.init({
  165. dw: 1920,
  166. dh: 1080,
  167. el:"#screenViewport",
  168. resize: true
  169. })
  170. if ( !firstMonth ) {
  171. return message.warning("当前未查询到对应的数据,请新增核算");
  172. }
  173. inpVal.value = firstMonth;
  174. initPageData();
  175. window.addEventListener("resize", windowResize);
  176. timerId = setInterval(() => currentTimer.value = getCurrentTimer(), 1000)
  177. })
  178. onUnmounted(() => {
  179. window.removeEventListener("resize", windowResize);
  180. clearInterval(timerId);
  181. echartMonth && echartMonth.dispose();
  182. echartSewage && echartSewage.dispose();
  183. echartEnergy && echartEnergy.dispose();
  184. echartMdu && echartMdu.dispose();
  185. autofit.off()
  186. })
  187. </script>
  188. <template>
  189. <div class="screen-viewprot" id="screenViewport">
  190. <header class="header">
  191. <div class="flex items-center">
  192. <NPopselect :options="selectOptions" v-model:value="inpVal" scrollable :on-update:value="handleSelectChecked">
  193. <div class="calendar space-x-[4px]">
  194. <span class="calendar-text">{{ inpVal }}</span>
  195. </div>
  196. </NPopselect>
  197. <div class="timer-inner">
  198. <p class="time">{{ currentTimer }}</p>
  199. <p class="day text-[13px]">{{ currentMonth }}</p>
  200. </div>
  201. <button class="back-btn_group" @click="handleBackPage">
  202. <SvgIcon name="carbon-icon-back" size="36"></SvgIcon>
  203. <span class="btn">返回</span>
  204. </button>
  205. </div>
  206. </header>
  207. <main class="main">
  208. <LayoutCard title="近7月碳排放趋势">
  209. <div ref="echartMonthRef" class="w-full h-full"></div>
  210. </LayoutCard>
  211. <DataBorad :data="historyDetails"></DataBorad>
  212. <LayoutCard title="污水处理生化反应">
  213. <div ref="echartSewageRef" class="w-full h-full"></div>
  214. </LayoutCard>
  215. <LayoutCard title="能源、药剂">
  216. <div class="energy-inner">
  217. <div ref="echartEnergyRef" class="w-full h-full"></div>
  218. </div>
  219. </LayoutCard>
  220. <DataScroll :data="scrollData"></DataScroll>
  221. <LayoutCard title="污泥处理-生化反应">
  222. <div ref="echartMudRef" class="w-full h-full"></div>
  223. </LayoutCard>
  224. </main>
  225. <ul class="footer">
  226. <li><img src="@/assets/images/carbon/bg-btn-1.png" alt="下载报告" class="img" @click="handleType('down')"></li>
  227. <li><img src="@/assets/images/carbon/bg-btn-2.png" alt="申报信息" class="img" @click="handleType('list')"></li>
  228. <li><img src="@/assets/images/carbon/bg-btn-3.png" alt="核算历史" class="img" @click="handleType('info')"></li>
  229. <li><img src="@/assets/images/carbon/bg-btn-4.png" alt="新增核算" class="img" @click="handleType('add')"></li>
  230. </ul>
  231. </div>
  232. <TheDrawerForm v-model="isVisibleDrawerForm" @on-submit="onSubmit" :all-month="allMonth"/>
  233. <TheDrawerList v-model="isVisibleDrawerList" :data="historyDetails" />
  234. <TheModalTable v-model="isVisibleDrawerTable"/>
  235. </template>
  236. <style lang="scss" scoped>
  237. $bg-path: '@/assets/images/carbon/';
  238. $text-color_primary: #C2D8EF;
  239. .screen-viewprot {
  240. display: grid;
  241. grid-template-columns: repeat(1, 1fr);
  242. grid-template-rows: 100px 1fr 100px;
  243. width: 100vw;
  244. height: 100vh;
  245. background: url("@/assets/images/carbon/bg.png") 50% / cover no-repeat;
  246. }
  247. .header {
  248. @include flex(x, start, end);
  249. padding: 0 18px;
  250. background: url("@/assets/images/carbon/bg-header.png") left top no-repeat;
  251. background-size: cover;
  252. .calendar {
  253. width: 90px;
  254. height: 30px;
  255. padding-left: 8px;
  256. margin-right: 54px;
  257. background: url("@/assets/images/carbon/bg-calendar.png") no-repeat,
  258. url("@/assets/images/carbon/bg-calendar-direction.png") 62px center no-repeat;
  259. background-size: 100% 100%, 10px 10px;
  260. font-size: 14px;
  261. color: #fff;
  262. text-shadow: 0px 1px 0px rgba(0, 22, 35, 0.75);
  263. line-height: 30px;
  264. font-family: D-DIN-PRO-700-Bold;
  265. cursor: pointer;
  266. span {
  267. box-shadow: 0px 1px 0px 0px #001623BF;
  268. }
  269. }
  270. .timer-inner {
  271. padding: 16px 8px;
  272. margin-right: 40px;
  273. font-family: 'D-DIN-PRO-700-Bold';
  274. font-weight: bold;
  275. color: $text-color_primary;
  276. text-align: right;
  277. .time {
  278. width: 80px;
  279. font-size: 24px;
  280. line-height: 24px;
  281. box-shadow: 0px 1px 3px 0px #050C198A;
  282. }
  283. }
  284. .back-btn_group {
  285. @include flex(x, center, center);
  286. font-size: 14px;
  287. line-height: 18px;
  288. color: $text-color_primary;
  289. }
  290. }
  291. .main {
  292. display: grid;
  293. grid-gap: 26px 26px;
  294. padding: 30px 48px 0 48px;
  295. grid-template-columns: 490px 1fr 490px;
  296. grid-template-rows: minmax(300px, 50%) minmax(300px, 50%);
  297. color: #fff;
  298. .energy-inner {
  299. width: 100%;
  300. height: 100%;
  301. font-family: 'D-DIN-PRO-700-Bold';
  302. // background: url('@/assets/images/carbon/bg-3d.png') 17% 80% no-repeat;
  303. }
  304. }
  305. .footer {
  306. @include flex(x, center, center);
  307. background: url("@/assets/images/carbon/Group 1321315504 (1).png") center center no-repeat;
  308. background-size: cover;
  309. li {
  310. width: 180px;
  311. cursor: pointer;
  312. .img {
  313. width: 100%;
  314. }
  315. }
  316. }
  317. </style>
  318. <style lang="scss">
  319. .calendar {
  320. .el-input__wrapper {
  321. padding: 0;
  322. padding-left: 16px;
  323. background: transparent;
  324. box-shadow: none;
  325. .el-input__prefix,
  326. .el-input__suffix {
  327. display: none;
  328. }
  329. .el-input__inner {
  330. font-family: 'D-DIN-PRO-700-Bold';
  331. font-size: 12px;
  332. color: #fff;
  333. cursor: pointer;
  334. }
  335. &:hover {
  336. box-shadow: none;
  337. }
  338. }
  339. .el-input__wrapper.is-focus {
  340. box-shadow: none;
  341. }
  342. }
  343. .el-month-table td.current:not(.disabled) .el-date-table-cell__text, .el-year-table td.current:not(.disabled) .el-date-table-cell__text {
  344. background: #3153f5;
  345. }
  346. </style>