index.vue 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291
  1. <template>
  2. <div class="table-box">
  3. <div class="card table-search" style="overflow: hidden;display: flex;justify-content: space-between;">
  4. <el-form ref="formRef" :model="searchParam" :inline="true" label-width="auto">
  5. <el-form-item label="自动站:" prop="as_code">
  6. <el-select v-model="pageable.as_code" filterable placeholder="请搜索自动站" remote reserve-keyword
  7. clearable style="width: 200px" @change="changeStation">
  8. <el-option v-for="item in platformList" :key="item.as_code"
  9. :label="item.as_code + ' ' + item.as_name" :value="item.as_code" />
  10. <template #prefix>
  11. <el-icon class="el-input__icon">
  12. <search />
  13. </el-icon>
  14. </template>
  15. </el-select>
  16. </el-form-item>
  17. <el-form-item label="观测时间:" prop="base">
  18. <div>
  19. <el-switch v-model="value1" @change="changetype" class="ml-2" inline-prompt active-text="正点"
  20. inactive-text="分钟" style="margin-right: 10px;" />
  21. <el-date-picker v-model="dateRange" value-format="YYYY-MM-DD HH:mm"
  22. :format="value1 ? 'YYYY-MM-DD' : 'YYYY-MM-DD HH:mm'"
  23. :type="value1 ? 'daterange' : 'datetimerange'" range-separator="-" start-placeholder="开始时间"
  24. end-placeholder="结束时间" :disabled-date="disabledDate"
  25. @focus="handleFocus" @calendar-change="handleChange" @change="handleDateChange"
  26. :default-time="[new Date(2000, 1, 1, 0, 0), new Date(2000, 1, 1, new Date().getHours(), 0)]"
  27. style="width: 300px;"></el-date-picker>
  28. </div>
  29. </el-form-item>
  30. <el-form-item>
  31. <div>
  32. <el-button type="primary" @click="getHistoryList">查询</el-button>
  33. <el-button plain @click="resetForm">重置</el-button>
  34. </div>
  35. </el-form-item>
  36. </el-form>
  37. <div class="header-button-ri">
  38. <slot name="toolButton">
  39. <img class="setting" style="margin-top: 5px;" src="@/assets/images/setting2.png"
  40. @click="handleSetting" />
  41. </slot>
  42. </div>
  43. </div>
  44. <div class="main_list">
  45. <el-row :gutter="15">
  46. <template v-if="array.length>0">
  47. <el-col :xs="24" :sm="12" :md="8" :lg="8">
  48. <div class="chart_item zhuti">
  49. <div style="font-weight: bold;" class="item_title">
  50. <span>2分钟平均风向 ( ° )/2分钟平均风速 ( m/s )</span>
  51. </div>
  52. <div class="mt5">
  53. <div ref="wind" class="ve-ring" style="height: 330px"></div>
  54. </div>
  55. </div>
  56. </el-col>
  57. <el-col :xs="24" :sm="12" :md="8" :lg="8" v-for="(item, index) in array" :key="item.data_id">
  58. <div class="chart_item zhuti">
  59. <div style="font-weight: bold;" class="item_title">
  60. <span>{{ item.data_name }} ( {{ item.data_unit ? item.data_unit : '' }} )</span>
  61. </div>
  62. <div class="mt5">
  63. <div :id="'Access' + index" class="ve-ring" style="height: 330px"></div>
  64. </div>
  65. </div>
  66. </el-col>
  67. </template>
  68. <el-col :span="24" v-else>
  69. <div class="chart_item zhuti">
  70. <el-empty description="暂无数据" />
  71. </div>
  72. </el-col>
  73. </el-row>
  74. </div>
  75. <!-- 添加或修改岗位对话框 -->
  76. <el-dialog :title="dialog.title" v-model="dialog.visible" width="1200px" append-to-body>
  77. <div style="height: 500px;display: flex">
  78. <el-tabs tab-position="left" class="demo-tabs" style="height: 100%">
  79. <el-tab-pane label="风">
  80. <div class="container-tag">
  81. <template v-for="item in elementList" :key="item.id">
  82. <div v-if="item.data_type === '风'"
  83. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  84. @click="toggleSelection(item)">{{ item.data_name }}</div>
  85. </template>
  86. </div>
  87. </el-tab-pane>
  88. <el-tab-pane label="温湿度">
  89. <div class="container-tag">
  90. <template v-for="item in elementList" :key="item.id">
  91. <div v-if="item.data_type === '温湿度'"
  92. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  93. @click="toggleSelection(item)">{{ item.data_name }}</div>
  94. </template>
  95. </div>
  96. </el-tab-pane>
  97. <el-tab-pane label="气压">
  98. <div class="container-tag">
  99. <template v-for="item in elementList" :key="item.id">
  100. <div v-if="item.data_type === '气压'"
  101. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  102. @click="toggleSelection(item)">{{ item.data_name }}</div>
  103. </template>
  104. </div>
  105. </el-tab-pane>
  106. <el-tab-pane label="降水">
  107. <div class="container-tag">
  108. <template v-for="item in elementList" :key="item.id">
  109. <div v-if="item.data_type === '降水'"
  110. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  111. @click="toggleSelection(item)">{{ item.data_name }}</div>
  112. </template>
  113. </div>
  114. </el-tab-pane>
  115. <el-tab-pane label="云">
  116. <div class="container-tag">
  117. <template v-for="item in elementList" :key="item.id">
  118. <div v-if="item.data_type === '云'"
  119. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  120. @click="toggleSelection(item)">{{ item.data_name }}</div>
  121. </template>
  122. </div>
  123. </el-tab-pane>
  124. <el-tab-pane label="能见度">
  125. <div class="container-tag">
  126. <template v-for="item in elementList" :key="item.id">
  127. <div v-if="item.data_type === '能见度'"
  128. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  129. @click="toggleSelection(item)">{{ item.data_name }}</div>
  130. </template>
  131. </div>
  132. </el-tab-pane>
  133. <el-tab-pane label="天气现象">
  134. <div class="container-tag">
  135. <template v-for="item in elementList" :key="item.id">
  136. <div v-if="item.data_type === '天气现象'"
  137. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  138. @click="toggleSelection(item)">{{ item.data_name }}</div>
  139. </template>
  140. </div>
  141. </el-tab-pane>
  142. <el-tab-pane label="积雪">
  143. <div class="container-tag">
  144. <template v-for="item in elementList" :key="item.id">
  145. <div v-if="item.data_type === '积雪'"
  146. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  147. @click="toggleSelection(item)">{{ item.data_name }}</div>
  148. </template>
  149. </div>
  150. </el-tab-pane>
  151. <el-tab-pane label="地面温度">
  152. <div class="container-tag">
  153. <template v-for="item in elementList" :key="item.id">
  154. <div v-if="item.data_type === '地面温度'"
  155. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  156. @click="toggleSelection(item)">{{ item.data_name }}</div>
  157. </template>
  158. </div>
  159. </el-tab-pane>
  160. <el-tab-pane label="大气电场">
  161. <div class="container-tag">
  162. <template v-for="item in elementList" :key="item.id">
  163. <div v-if="item.data_type === '大气电场'"
  164. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  165. @click="toggleSelection(item)">{{ item.data_name }}</div>
  166. </template>
  167. </div>
  168. </el-tab-pane>
  169. <el-tab-pane label="闪电">
  170. <div class="container-tag">
  171. <template v-for="item in elementList" :key="item.id">
  172. <div v-if="item.data_type === '闪电'"
  173. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  174. @click="toggleSelection(item)">{{ item.data_name }}</div>
  175. </template>
  176. </div>
  177. </el-tab-pane>
  178. <el-tab-pane label="日照">
  179. <div class="container-tag">
  180. <template v-for="item in elementList" :key="item.id">
  181. <div v-if="item.data_type === '日照'"
  182. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  183. @click="toggleSelection(item)">{{ item.data_name }}</div>
  184. </template>
  185. </div>
  186. </el-tab-pane>
  187. <el-tab-pane label="总辐射">
  188. <div class="container-tag">
  189. <template v-for="item in elementList" :key="item.id">
  190. <div v-if="item.data_type === '总辐射'"
  191. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  192. @click="toggleSelection(item)">{{ item.data_name }}</div>
  193. </template>
  194. </div>
  195. </el-tab-pane>
  196. <el-tab-pane label="日照文">
  197. <div class="container-tag">
  198. <template v-for="item in elementList" :key="item.id">
  199. <div v-if="item.data_type === '日照文'"
  200. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  201. @click="toggleSelection(item)">{{ item.data_name }}</div>
  202. </template>
  203. </div>
  204. </el-tab-pane>
  205. <el-tab-pane label="水文">
  206. <div class="container-tag">
  207. <template v-for="item in elementList" :key="item.id">
  208. <div v-if="item.data_type === '水文'"
  209. :class="item.isSelected ? 'item-tag-active' : 'item-tag'"
  210. @click="toggleSelection(item)">{{ item.data_name }}</div>
  211. </template>
  212. </div>
  213. </el-tab-pane>
  214. </el-tabs>
  215. <div class="selected-list-box">
  216. <div class="disposition-title" style="margin-top: 0px">已选要素</div>
  217. <div class="selected-list-c">
  218. <template v-for="(item, index) in copiedCustomizeColumns" :key="item.id">
  219. <div :class="selectedItemIndex === index ? 'selected-item-tag-active' : 'selected-item-tag'"
  220. @click="changeSelected(index)">{{ item.data_name }}</div>
  221. </template>
  222. </div>
  223. <div class="button-container">
  224. <el-button type="primary" plain class="top-left" @click="changeMoveUp">上移</el-button>
  225. <el-button type="primary" plain class="top-right" @click="changeMoveDown">下移</el-button>
  226. <el-button type="primary" plain class="bottom-left" @click="changeRemove">移除</el-button>
  227. <el-button type="primary" plain class="bottom-right" @click="changeCleared">清空</el-button>
  228. </div>
  229. </div>
  230. </div>
  231. <template #footer>
  232. <div class="dialog-footer">
  233. <el-button @click="cancel">取 消</el-button>
  234. <el-button type="primary" @click="submitForm">确 定</el-button>
  235. </div>
  236. </template>
  237. </el-dialog>
  238. </div>
  239. </template>
  240. <script setup lang="ts" name="dataSynthesis">
  241. import * as echarts from 'echarts';
  242. import { ElMessage } from "element-plus";
  243. import { useGlobalStore } from "@/stores/modules/global";
  244. import { ref, computed, onMounted, reactive, watch, onDeactivated, onActivated, nextTick } from "vue";
  245. import { getDataItemList, getPlatformList, getRgDataList, getTacRecordList } from "@/api/modules/allData";
  246. import { Platform } from "@/api/interface";
  247. import { parseTime } from '@/utils/index';
  248. import { isToday, isWithinThirtyMinutes, getTimeStamp, getStamp, isDateFuture, isDateRangeWithinDays } from "@/utils/dateTime";
  249. import {useUserStore} from "@/stores/modules/user";
  250. const value1 = ref(true)
  251. const dateRange = ref<[any, any]>([parseTime(new Date(), '{y}-{m}-{d}') + ' 00:00', parseTime(new Date(), '{y}-{m}-{d} {h}') + ':00']);
  252. const changetype = (e) => {
  253. if (!e) {
  254. dateRange.value = [parseTime(getTimeStamp(1), '{y}-{m}-{d} {h}:{i}'), parseTime(new Date(), '{y}-{m}-{d} {h}:{i}')]
  255. }else {
  256. dateRange.value = [parseTime(new Date(), '{y}-{m}-{d}') + ' 00:00', parseTime(new Date(), '{y}-{m}-{d} {h}') + ':00'];
  257. }
  258. getHistoryList()
  259. }
  260. const handleDateRangeChange = (newVal) => {
  261. const [startDate, endDate] = newVal;
  262. if (endDate && isToday(endDate)) {
  263. // 获取当前时间的整点
  264. const currentHour = new Date().getHours();
  265. const roundedEndDate = new Date(endDate);
  266. roundedEndDate.setMinutes(0, 0, 0); // 设置分钟、秒、毫秒为0
  267. const startZeroed = new Date(startDate);
  268. startZeroed.setHours(0, 0, 0, 0);
  269. dateRange.value = [parseTime(new Date(startZeroed)), parseTime(new Date(roundedEndDate))];
  270. } else {
  271. const noTodayStart = new Date(startDate).setMinutes(0, 0, 0);
  272. const noTodayEnd = new Date(endDate).setHours(23, 0, 0, 0);
  273. dateRange.value = [parseTime(new Date(noTodayStart)), parseTime(new Date(noTodayEnd))];
  274. }
  275. };
  276. const handleDateChange = async (e) => {
  277. if (isDateFuture(e[0]) || isDateFuture(e[1])) {
  278. ElMessage.error("不得选择未来的时间");
  279. } else {
  280. if (!value1.value) {
  281. // if (isWithinThirtyMinutes(e[0], e[1])) {
  282. // 获取参数
  283. getHistoryList()
  284. // } else {
  285. // ElMessage.error("分钟模式下时间间隔不能大于3小时");
  286. // }
  287. } else {
  288. // if (isDateRangeWithinDays(e[0], e[1], 7)) {
  289. await handleDateRangeChange(e)
  290. getHistoryList()
  291. // } else {
  292. // ElMessage.error("正点模式下时间间隔不能大于7天");
  293. // }
  294. }
  295. }
  296. }
  297. const pickDay = ref()
  298. const handleFocus = () => {
  299. pickDay.value = null
  300. }
  301. const limitTime = 1000 * 60 * 60 * 24 * 6
  302. const disabledDate = (time: Date) => {
  303. return time.getTime() > Date.now() - 8.64e6
  304. }
  305. const handleChange = (val: Date[]) => {
  306. const [pointDay] = val
  307. pickDay.value = pointDay
  308. }
  309. // 查询所有人工要素字典表
  310. const elementList = ref<any>([])
  311. //自定义部分表头
  312. let customizeColumns = ref<any>([
  313. {
  314. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  315. data_id: 2,
  316. data_type: "风",
  317. data_item: "ER_FEN_ZHONG_PING_JUN_FENG_XIANG",
  318. data_name: "2分钟风向",
  319. data_unit: "°",
  320. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  321. data_value: "",
  322. data_order: 0,
  323. data_condition: 0
  324. },
  325. {
  326. data_id: 1,
  327. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  328. data_type: "风",
  329. data_name: "2分钟风速",
  330. data_item: "ER_FEN_ZHONG_PING_JUN_FENG_SU",
  331. data_unit: "m/s",
  332. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  333. data_value: "",
  334. data_order: 0,
  335. data_condition: 0
  336. },
  337. {
  338. data_id: 4,
  339. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  340. data_type: "风",
  341. data_name: "10分钟风向",
  342. data_item: "SHI_FEN_ZHONG_PING_JUN_FENG_XIANG",
  343. data_unit: "°",
  344. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  345. data_value: "",
  346. data_order: 0,
  347. data_condition: 0
  348. },
  349. {
  350. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  351. data_id: 3,
  352. data_type: "风",
  353. data_item: "SHI_FEN_ZHONG_PING_JUN_FENG_SU",
  354. data_name: "10分钟风速",
  355. data_unit: "m/s",
  356. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  357. data_value: "",
  358. data_order: 0,
  359. data_condition: 0
  360. },
  361. {
  362. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  363. data_id: 501,
  364. data_type: "能见度",
  365. data_item: "YI_FEN_ZHONG_PING_JUN_NENG_JIAN_DU",
  366. data_name: "1分钟平均能见度",
  367. data_unit: "m",
  368. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  369. data_value: "",
  370. data_order: 0,
  371. data_condition: 0
  372. },
  373. {
  374. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  375. data_id: 101,
  376. data_type: "温湿度",
  377. data_item: "QI_WEN",
  378. data_name: "气温",
  379. data_unit: "℃",
  380. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  381. data_value: "",
  382. data_order: 0,
  383. data_condition: 0
  384. },
  385. {
  386. data_id: 201,
  387. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  388. data_type: "气压",
  389. data_name: "水汽压",
  390. data_item: "SHUI_QI_YA",
  391. data_unit: "hPa",
  392. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  393. data_value: "",
  394. data_order: 0,
  395. data_condition: 0
  396. },
  397. {
  398. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  399. data_id: 109,
  400. data_type: "温湿度",
  401. data_item: "LU_DIAN_WEN_DU",
  402. data_name: "露点温度",
  403. data_unit: "℃",
  404. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  405. data_value: "",
  406. data_order: 0,
  407. data_condition: 0
  408. },
  409. {
  410. data_id: 202,
  411. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  412. data_type: "气压",
  413. data_name: "本站气压",
  414. data_item: "BEN_ZHAN_QI_YA",
  415. data_unit: "hPa",
  416. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  417. data_value: "",
  418. data_order: 0,
  419. data_condition: 0
  420. }, {
  421. data_id: 207,
  422. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  423. data_type: "气压",
  424. data_name: "海平面气压",
  425. data_item: "HAI_PING_MIAN_QI_YA",
  426. data_unit: "hPa",
  427. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  428. data_value: "",
  429. data_order: 0,
  430. data_condition: 0
  431. },
  432. {
  433. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  434. data_id: 312,
  435. data_type: "降水",
  436. data_item: "XIAO_SHI_LEI_JI_JIANG_SHUI_LIANG_CHENG_ZHONG",
  437. data_name: "小时累计降水量(称重)",
  438. data_unit: "mm",
  439. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  440. data_value: "",
  441. data_order: 0,
  442. data_condition: 0
  443. }
  444. ]);
  445. const getAllRgDataList = async () => {
  446. const { data } = await getRgDataList();
  447. elementList.value = data.list
  448. };
  449. const copiedCustomizeColumns = ref<any>([...customizeColumns.value]);
  450. const sensorList = ref<any>([])
  451. // 被选中当条数据下标
  452. const selectedItemIndex = ref(0)
  453. const changeMoveUp = () => {
  454. //上移
  455. if (selectedItemIndex.value > 0) {
  456. const temp = copiedCustomizeColumns.value[selectedItemIndex.value - 1]
  457. copiedCustomizeColumns.value[selectedItemIndex.value - 1] = copiedCustomizeColumns.value[selectedItemIndex.value]
  458. copiedCustomizeColumns.value[selectedItemIndex.value] = temp
  459. selectedItemIndex.value--
  460. }
  461. }
  462. const changeMoveDown = () => {
  463. //下移
  464. if (selectedItemIndex.value < copiedCustomizeColumns.value.length - 1) {
  465. const temp = copiedCustomizeColumns.value[selectedItemIndex.value + 1]
  466. copiedCustomizeColumns.value[selectedItemIndex.value + 1] = copiedCustomizeColumns.value[selectedItemIndex.value]
  467. copiedCustomizeColumns.value[selectedItemIndex.value] = temp
  468. selectedItemIndex.value++
  469. }
  470. }
  471. const changeRemove = () => {
  472. //删除数据
  473. if (copiedCustomizeColumns.value.length > 0) {
  474. const custom = copiedCustomizeColumns.value[selectedItemIndex.value];
  475. const se_type = custom.se_type
  476. for (let i = 0; i < sensorList.value.length; i++) {
  477. if (sensorList.value[i].se_type === se_type) {
  478. sensorList.value[i].isSelected = false;
  479. }
  480. }
  481. copiedCustomizeColumns.value.splice(selectedItemIndex.value, 1)
  482. if (selectedItemIndex.value === copiedCustomizeColumns.value.length) {
  483. selectedItemIndex.value--
  484. }
  485. }
  486. }
  487. const changeCleared = () => {
  488. //清空所有数据
  489. selectedItemIndex.value = 0
  490. copiedCustomizeColumns.value = []
  491. for (let i = 0; i < sensorList.value.length; i++) {
  492. sensorList.value[i].isSelected = false;
  493. }
  494. }
  495. const changeSelected = (index) => {
  496. selectedItemIndex.value = index
  497. }
  498. // 要素的选中和取消
  499. const toggleSelection = (item) => {
  500. item.isSelected = !item.isSelected
  501. if (item.isSelected) {
  502. //添加
  503. copiedCustomizeColumns.value.push(item)
  504. } else {
  505. //删除
  506. copiedCustomizeColumns.value = copiedCustomizeColumns.value.filter(column => column.data_id !== item.data_id);
  507. }
  508. }
  509. const handleSetting = () => {
  510. copiedCustomizeColumns.value = [...customizeColumns.value]
  511. elementList.value.forEach(element => {
  512. const column = copiedCustomizeColumns.value.find(col => col.data_id === element.data_id);
  513. if (column) {
  514. element.isSelected = true;
  515. } else {
  516. element.isSelected = false;
  517. }
  518. });
  519. dialog.title = "列表字段设置"
  520. dialog.visible = true;
  521. }
  522. const submitForm = () => {
  523. dialog.visible = false;
  524. customizeColumns.value = [...copiedCustomizeColumns.value]
  525. // updateColumns()
  526. // getList()
  527. // weatherInfo()
  528. getHistoryList()
  529. saveCustomizeColumns()
  530. }
  531. const cancel = () => {
  532. dialog.visible = false;
  533. }
  534. const formRef = ref()
  535. const wind = ref()
  536. const pressureLine = ref()
  537. const radio = ref()
  538. const radio2 = ref()
  539. const searchParam = reactive({
  540. base: undefined,
  541. date: '',
  542. switch: true
  543. })
  544. const dialog = reactive<any>({
  545. visible: false,
  546. title: ''
  547. });
  548. const pageable = ref({
  549. data_type: false,
  550. time_order: 1,
  551. as_code: undefined,
  552. data_items: [],
  553. pageNum: 1,
  554. });
  555. let historyColumns = [
  556. {
  557. data_id: 202,
  558. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  559. data_type: "气压",
  560. data_name: "本站气压",
  561. data_item: "BEN_ZHAN_QI_YA",
  562. data_unit: "hPa",
  563. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  564. data_value: "",
  565. data_order: 0,
  566. data_condition: 0
  567. },
  568. {
  569. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  570. data_id: 2,
  571. data_type: "风",
  572. data_item: "ER_FEN_ZHONG_PING_JUN_FENG_XIANG",
  573. data_name: "2分钟风向",
  574. data_unit: "°",
  575. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  576. data_value: "",
  577. data_order: 0,
  578. data_condition: 0
  579. },
  580. {
  581. data_id: 1,
  582. data_r_table: "SHI_SHI_LIU_YAO_SU_SHU_JU",
  583. data_type: "风",
  584. data_name: "2分钟风速",
  585. data_item: "ER_FEN_ZHONG_PING_JUN_FENG_SU",
  586. data_unit: "m/s",
  587. data_h_table: "LI_SHI_LIU_YAO_SU_SHU_JU",
  588. data_value: "",
  589. data_order: 0,
  590. data_condition: 0
  591. },
  592. {
  593. data_id: 1203,
  594. data_r_table: "SHI_SHI_FU_SHE_SHU_JU",
  595. data_type: "总辐射",
  596. data_name: "总辐射曝辐量",
  597. data_item: "ZONG_FU_SHE_BAO_FU_LIANG",
  598. data_unit: "MJ/m²",
  599. data_h_table: "LI_SHI_FU_SHE_SHU_JU",
  600. data_value: "",
  601. data_order: 0,
  602. data_condition: 0,
  603. isSelected: true
  604. },
  605. {
  606. data_id: 1202,
  607. data_r_table: "SHI_SHI_FU_SHE_SHU_JU",
  608. data_type: "总辐射",
  609. data_name: "总辐射辐照度",
  610. data_item: "ZONG_FU_SHE_FU_ZHAO_DU",
  611. data_unit: "W/m²",
  612. data_h_table: "LI_SHI_FU_SHE_SHU_JU",
  613. data_value: "",
  614. data_order: 0,
  615. data_condition: 0,
  616. isSelected: true
  617. }
  618. ]
  619. // 查询自动站列表
  620. const queryas = ref<Platform>({
  621. pageSize: 5000,
  622. pageNum: 1,
  623. currentpage: 1
  624. })
  625. const userStore = useUserStore();
  626. const platformList =ref<any>(computed(() => userStore.stations))
  627. const selectPlatform = ref()
  628. const getPlatforms = async () => {
  629. pageable.value.as_code = platformList.value[0].as_code
  630. selectPlatform.value = platformList.value[0]
  631. getHistoryList()
  632. };
  633. const changeStation = () => {
  634. getHistoryList()
  635. }
  636. //时间戳转换为指定格式的日期字符串
  637. function formatDate(timestamp, type) {
  638. const date = new Date(timestamp * 1000); // 将时间戳转换为毫秒
  639. const year = date.getFullYear();
  640. const month = String(date.getMonth() + 1).padStart(2, '0');
  641. const day = String(date.getDate()).padStart(2, '0');
  642. const hours = String(date.getHours()).padStart(2, '0');
  643. const minutes = String(date.getMinutes()).padStart(2, '0');
  644. if (type === 1) {
  645. return `${month}/${day} ${hours}:${minutes}`;
  646. } else {
  647. return `${year}-${month}-${day} ${hours}:${minutes}`;
  648. }
  649. }
  650. const array = ref<any>([])
  651. const Access = () => {
  652. array.value?.forEach((item, i) => {
  653. // 获取最大值方法
  654. function calMax(arr) {
  655. let max = Math.max.apply(null, arr); // 获取最大值方法
  656. let maxint = Math.ceil(max / 5); // 向上以5的倍数取整
  657. let maxval = maxint * 5; // 最终设置的最大值
  658. return maxval; // 输出最大值
  659. }
  660. // 获取最小值方法
  661. function calMin(arr) {
  662. let min = Math.min.apply(null, arr); // 获取最小值方法
  663. return min
  664. }
  665. // 调用方法,获取数据的最大值&最小值
  666. let maxData1 = calMax(item.value);
  667. let minData1 = calMin(item.value);
  668. let chartDom = document.getElementById("Access" + i);
  669. if (chartDom) {
  670. let myChart = echarts.init(chartDom);
  671. let option;
  672. option = {
  673. grid: {
  674. left: '10',
  675. right: '10',
  676. bottom: '3%',
  677. top: 20,
  678. containLabel: true
  679. },
  680. // X轴
  681. xAxis: {
  682. type: 'category',
  683. splitLine: {
  684. show: false
  685. },
  686. // boundaryGap: false,
  687. axisTick: {
  688. alignWithLabel: true, // true:标签位于刻度线正下方;false:标签位于2个刻度线中间
  689. },
  690. data: timeArr.value,
  691. axisLabel: {
  692. formatter: function (value) {
  693. return value.replace(/ /g, '\n');
  694. },
  695. },
  696. axisLine: {
  697. onZero: false //-----------重点
  698. }
  699. },
  700. yAxis: {
  701. // min: minData1,
  702. // max: maxData1,
  703. type: 'value',
  704. // boundaryGap: [0, '10%']
  705. },
  706. // 鼠标悬浮提示
  707. tooltip: {
  708. trigger: "axis",
  709. },
  710. // 数据
  711. series: [
  712. {
  713. data: item.value,
  714. name: item.data_name,
  715. type: "line",
  716. smooth: true,
  717. lineStyle: {
  718. color: '#157aea',
  719. width: 1,
  720. },
  721. itemStyle: {
  722. normal: {
  723. color: '#157aea'
  724. },
  725. }
  726. },
  727. ]
  728. };
  729. myChart.setOption(option)
  730. window.addEventListener("resize", function () {
  731. myChart.resize();
  732. });
  733. }
  734. // option && myChart.setOption(option);
  735. })
  736. };
  737. const timeArr = ref()
  738. const windLineArr = ref() // 2分钟风速风向
  739. const tempLineArr = ref()
  740. const humidityLineArr = ref()
  741. const pressureLineArr = ref()
  742. const getHistoryList = async () => {
  743. if (isDateFuture(dateRange.value[0]) ||isDateFuture(dateRange.value[1])) {
  744. ElMessage.error("不得选择未来的时间");
  745. return false;
  746. }
  747. const arr = [...customizeColumns.value, ...historyColumns].reduce((acc, item) => {
  748. if (!acc.some(existingItem => existingItem.data_id === item.data_id)) {
  749. acc.push(item);
  750. }
  751. return acc;
  752. }, []);
  753. const { data } = await getDataItemList({ ...pageable.value, as_code_list: [pageable.value.as_code], data_items: arr, data_type: false, time_space: value1.value ? 60 : 1, end_time: getStamp(dateRange.value[1]), begin_time: getStamp(dateRange.value[0]) });
  754. timeArr.value = data.list?.map(item => {
  755. return formatDate(item.data_time_i, 1)
  756. })
  757. // 初始化一个空对象来存储归类后的数据
  758. const resultMap = {};
  759. // 遍历原数组
  760. data.list?.forEach(obj => {
  761. obj.data_list.forEach(item => {
  762. // 将data_value转换为数字
  763. const value = parseFloat(item.data_value);
  764. // 初始化或更新resultMap中的对应项
  765. if (!resultMap[item.data_id]) {
  766. resultMap[item.data_id] = {
  767. data_id: item.data_id,
  768. data_name: item.data_name,
  769. data_unit: item.data_unit,
  770. value: [value]
  771. };
  772. } else {
  773. resultMap[item.data_id].value.push(value);
  774. }
  775. });
  776. });
  777. array.value = Object.values(resultMap);
  778. nextTick(() => {
  779. if (array.value.length>0) Access();
  780. });
  781. // console.log(array.value, 'array');
  782. windLineArr.value = data.list?.map(item => {
  783. return {
  784. value: item[1],
  785. symbolRotate: adjustValue(item[2]),
  786. symbolRotateStr: item[2]
  787. }
  788. })
  789. nextTick(()=>{
  790. if (array.value.length>0) showWind()
  791. })
  792. };
  793. function adjustValue(value) {
  794. if (value === 'C') {
  795. return 0
  796. } else if (value <= 180) {
  797. return '-' + value;
  798. } else {
  799. return 360 - value;
  800. }
  801. }
  802. const globalStore = useGlobalStore();
  803. const isDark = computed(() => globalStore.isDark);
  804. // resetForm
  805. const resetForm = () => {
  806. dateRange.value = [parseTime(new Date(), '{y}-{m}-{d}') + ' 00:00', parseTime(new Date(), '{y}-{m}-{d} {h}') + ':00'];
  807. value1.value = true
  808. pageable.value.as_code = platformList.value[0].as_code
  809. getHistoryList()
  810. };
  811. const showWind = () => {
  812. let mychart = echarts.init(wind.value);
  813. let option = {
  814. title: {
  815. text: '暂无数据',
  816. show: !timeArr.value ? true : false,
  817. x: 'center',
  818. y: 'center',
  819. textStyle: {
  820. fontSize: 14,
  821. fontWeight: 'normal',
  822. }
  823. },
  824. grid: {
  825. left: '10',
  826. right: '10',
  827. bottom: '3%',
  828. top: 20,
  829. containLabel: true
  830. },
  831. // X轴
  832. xAxis: {
  833. type: 'category',
  834. splitLine: {
  835. show: false
  836. },
  837. boundaryGap: true,
  838. axisTick: {
  839. alignWithLabel: true, // true:标签位于刻度线正下方;false:标签位于2个刻度线中间
  840. },
  841. data: timeArr.value,
  842. axisLabel: {
  843. // interval: 4,
  844. formatter: function (value) {
  845. // if (value.index % 4 === 0) {
  846. // return value.replace(/ /g, '\n');
  847. // } else {
  848. // return '';
  849. // }
  850. return value.replace(/ /g, '\n');
  851. },
  852. },
  853. },
  854. // Y轴
  855. // yAxis: {
  856. // type: 'value',
  857. // // name: "风速(m/s)",
  858. // },
  859. yAxis: {
  860. min: 0,
  861. max: 10,
  862. type: 'value'
  863. },
  864. // 鼠标悬浮提示
  865. tooltip: {
  866. trigger: "axis",
  867. formatter(params) {
  868. let tip =
  869. "时间:" +
  870. params[0].axisValue +
  871. "<br/>风速:" +
  872. (params[0].value ? params[0].value : "--") +
  873. "m/s<br/>风向:" +
  874. (params[0].data ? params[0].data.symbolRotateStr + '°' : "--")
  875. return tip;
  876. },
  877. },
  878. // 数据
  879. series: [{
  880. data: windLineArr.value,
  881. type: 'line',
  882. smooth: true, //这句就是让曲线变平滑的
  883. symbol: 'path://M4.866,0,0,15.193l4.866-4.449L9.738,15.2Z',
  884. symbolSize: 11,
  885. //折线样式
  886. lineStyle: {
  887. color: '#a5cfff',
  888. width: 1,
  889. },
  890. // label: {
  891. // normal: {
  892. // formatter: '{c}',
  893. // show: false,
  894. // position: 'top',
  895. // distance: 15
  896. // }
  897. // },
  898. itemStyle: {
  899. normal: {
  900. },
  901. },
  902. }]
  903. };
  904. mychart.setOption(option)
  905. window.addEventListener("resize", function () {
  906. mychart.resize();
  907. });
  908. }
  909. const data3 = ref([1004, 1004, 1004, 1004, 1004, 1005, 1005, 1005, 1005, 1004.5, 1004.6, 1004.8]);
  910. const showPressureLine = () => {
  911. // 获取最大值方法
  912. function calMax(arr) {
  913. let max = Math.max.apply(null, arr); // 获取最大值方法
  914. let maxint = Math.ceil(max / 5); // 向上以5的倍数取整
  915. let maxval = maxint * 5 + 5; // 最终设置的最大值
  916. return maxval; // 输出最大值
  917. }
  918. // 获取最小值方法
  919. function calMin(arr) {
  920. let min = Math.min.apply(null, arr); // 获取最小值方法
  921. let minint = Math.floor(min / 1); // 向下以1的倍数取整
  922. let minval = minint * 1 - 5; // 最终设置的最小值
  923. return minval; // 输出最小值
  924. }
  925. // 调用方法,获取数据的最大值&最小值
  926. let maxData1 = calMax(data3.value);
  927. let minData1 = calMin(data3.value);
  928. let mychart = echarts.init(pressureLine.value);
  929. let option = {
  930. grid: {
  931. left: '10',
  932. right: '10',
  933. bottom: '3%',
  934. top: 20,
  935. containLabel: true
  936. },
  937. // X轴
  938. xAxis: {
  939. type: 'category',
  940. splitLine: {
  941. show: false
  942. },
  943. // boundaryGap: false,
  944. axisTick: {
  945. alignWithLabel: true, // true:标签位于刻度线正下方;false:标签位于2个刻度线中间
  946. },
  947. data: ['06/12 08:00', '06/12 09:00', '06/12 10:00', '06/12 11:00', '06/12 12:00', '06/12 13:00', '06/12 14:00', '06/12 15:00', '06/12 16:00', '06/12 17:00', '06/12 18:00', '06/12 19:00'],
  948. axisLabel: {
  949. interval: 1,
  950. formatter: function (value) {
  951. return value.replace(/ /g, '\n');
  952. },
  953. },
  954. },
  955. yAxis: {
  956. min: minData1,
  957. max: maxData1,
  958. type: 'value'
  959. },
  960. // 鼠标悬浮提示
  961. tooltip: {
  962. trigger: "axis",
  963. },
  964. // 数据
  965. series: [
  966. {
  967. data: data3.value,
  968. name: "气压",
  969. type: "line",
  970. smooth: true,
  971. lineStyle: {
  972. color: '#157aea',
  973. width: 1,
  974. },
  975. itemStyle: {
  976. normal: {
  977. color: '#157aea'
  978. },
  979. }
  980. },
  981. ]
  982. };
  983. mychart.setOption(option)
  984. window.addEventListener("resize", function () {
  985. mychart.resize();
  986. });
  987. }
  988. const data4 = ref([0.4, 0.5, 0.45, 0.5, 0.4, 0.5, 0.6, 0.55, 0.5, 0.5, 0.6, 0.55]);
  989. const showRadio = () => {
  990. let mychart = echarts.init(radio.value);
  991. let option = {
  992. grid: {
  993. left: '10',
  994. right: '10',
  995. bottom: '3%',
  996. top: 20,
  997. containLabel: true
  998. },
  999. // X轴
  1000. xAxis: {
  1001. type: 'category',
  1002. splitLine: {
  1003. show: false
  1004. },
  1005. // boundaryGap: false,
  1006. axisTick: {
  1007. alignWithLabel: true, // true:标签位于刻度线正下方;false:标签位于2个刻度线中间
  1008. },
  1009. data: ['06/12 08:00', '06/12 09:00', '06/12 10:00', '06/12 11:00', '06/12 12:00', '06/12 13:00', '06/12 14:00', '06/12 15:00', '06/12 16:00', '06/12 17:00', '06/12 18:00', '06/12 19:00'],
  1010. axisLabel: {
  1011. interval: 1,
  1012. formatter: function (value) {
  1013. return value.replace(/ /g, '\n');
  1014. },
  1015. },
  1016. },
  1017. yAxis: {
  1018. min: 0,
  1019. max: 1,
  1020. interval: 0.2,
  1021. type: 'value'
  1022. },
  1023. // 鼠标悬浮提示
  1024. tooltip: {
  1025. trigger: "axis",
  1026. },
  1027. // 数据
  1028. series: [
  1029. {
  1030. name: "曝辐量",
  1031. type: "line",
  1032. smooth: true,
  1033. data: data4.value,
  1034. lineStyle: {
  1035. color: '#157aea',
  1036. width: 1,
  1037. },
  1038. itemStyle: {
  1039. normal: {
  1040. color: '#157aea'
  1041. },
  1042. }
  1043. },
  1044. ]
  1045. };
  1046. mychart.setOption(option)
  1047. window.addEventListener("resize", function () {
  1048. mychart.resize();
  1049. });
  1050. }
  1051. const data5 = ref([800, 810, 800, 820, 800, 850, 800, 830, 800, 850, 800, 820]);
  1052. const showRadio2 = () => {
  1053. let mychart = echarts.init(radio2.value);
  1054. let option = {
  1055. grid: {
  1056. left: '10',
  1057. right: '10',
  1058. bottom: '3%',
  1059. top: 20,
  1060. containLabel: true
  1061. },
  1062. // X轴
  1063. xAxis: {
  1064. type: 'category',
  1065. splitLine: {
  1066. show: false
  1067. },
  1068. // boundaryGap: false,
  1069. axisTick: {
  1070. alignWithLabel: true, // true:标签位于刻度线正下方;false:标签位于2个刻度线中间
  1071. },
  1072. data: ['06/12 08:00', '06/12 09:00', '06/12 10:00', '06/12 11:00', '06/12 12:00', '06/12 13:00', '06/12 14:00', '06/12 15:00', '06/12 16:00', '06/12 17:00', '06/12 18:00', '06/12 19:00'],
  1073. axisLabel: {
  1074. interval: 0,
  1075. formatter: function (value) {
  1076. return value.replace(/ /g, '\n');
  1077. },
  1078. },
  1079. },
  1080. yAxis: {
  1081. min: 0,
  1082. max: 1200,
  1083. interval: 200,
  1084. type: 'value'
  1085. },
  1086. // 鼠标悬浮提示
  1087. tooltip: {
  1088. trigger: "axis",
  1089. },
  1090. // 数据
  1091. series: [
  1092. {
  1093. name: "辐照度",
  1094. type: "line",
  1095. smooth: true,
  1096. data: data5.value,
  1097. lineStyle: {
  1098. color: '#157aea',
  1099. width: 1,
  1100. },
  1101. itemStyle: {
  1102. normal: {
  1103. color: '#157aea'
  1104. },
  1105. }
  1106. },
  1107. ]
  1108. };
  1109. mychart.setOption(option)
  1110. window.addEventListener("resize", function () {
  1111. mychart.resize();
  1112. });
  1113. }
  1114. // 保存customizeColumns数组到localStorage
  1115. function saveCustomizeColumns() {
  1116. const jsonString = JSON.stringify(customizeColumns.value);
  1117. localStorage.setItem('customizeColumns', jsonString);
  1118. }
  1119. // 从localStorage中获取customizeColumns数组
  1120. function loadCustomizeColumns() {
  1121. const jsonString = localStorage.getItem('customizeColumns');
  1122. if (jsonString) {
  1123. customizeColumns.value = JSON.parse(jsonString);
  1124. }
  1125. }
  1126. onMounted(() => {
  1127. loadCustomizeColumns()
  1128. getAllRgDataList()
  1129. getPlatforms()
  1130. nextTick(() => {
  1131. Access();
  1132. });
  1133. // showWind()
  1134. // showPressureLine()
  1135. // showRadio()
  1136. // showRadio2()
  1137. })
  1138. </script>
  1139. <style scoped lang="scss">
  1140. .el-row .el-col {
  1141. margin-bottom: 15px;
  1142. }
  1143. .main_list {
  1144. background: transparent !important;
  1145. box-shadow: none !important;
  1146. padding: 0 !important;
  1147. height: calc(100vh - 150px);
  1148. margin-bottom: 10px;
  1149. overflow-y: scroll;
  1150. overflow-x: hidden;
  1151. .chart_item {
  1152. border-radius: 10px;
  1153. padding: 10px;
  1154. background-color: #fff;
  1155. // height: 100%;
  1156. // margin-bottom: 15px;
  1157. .item_title {
  1158. display: flex;
  1159. justify-content: space-between;
  1160. align-items: center;
  1161. }
  1162. }
  1163. .info_box {
  1164. .info_title {
  1165. color: #999999;
  1166. font-size: 16px;
  1167. }
  1168. .info_content {
  1169. height: 40px;
  1170. font-weight: bold;
  1171. color: #000;
  1172. font-size: 26px;
  1173. }
  1174. }
  1175. }
  1176. .el-button.is-link {
  1177. height: 32px;
  1178. }
  1179. </style>