widget.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491
  1. <template>
  2. <com-side-page>
  3. <template slot="side">
  4. <div class="side-title">内涝点水位监测点</div>
  5. <div class="side-content">
  6. <div class="searchInput">
  7. <el-input
  8. placeholder="支持搜索监测点名称"
  9. suffix-icon="el-icon-search"
  10. size="small"
  11. v-model="searchDevice"
  12. clearable
  13. >
  14. </el-input>
  15. </div>
  16. <el-tree
  17. :ref="`tree`"
  18. class="dataListTree"
  19. show-checkbox
  20. default-expand-all
  21. :default-checked-keys="defaultcheckedList"
  22. node-key="id"
  23. :data="treeData"
  24. :props="defaultProps"
  25. :filter-node-method="filterNode"
  26. @check="handleCheckChange"
  27. >
  28. </el-tree>
  29. </div>
  30. </template>
  31. <template slot="main">
  32. <div class="main-title">
  33. <div class="mt-bottom">
  34. <label>数据时间:</label>
  35. <div class="switch">
  36. <div
  37. class="switch-item"
  38. v-for="item in switchType"
  39. :key="item.type"
  40. :class="{ 'switch-active': currentShowType == item.type }"
  41. @click="currentShowType = item.type"
  42. >
  43. <template>{{ item.name }}</template>
  44. </div>
  45. </div>
  46. <el-date-picker
  47. class="el-date-picker"
  48. v-model="customDate"
  49. type="datetimerange"
  50. range-separator="至"
  51. start-placeholder="开始日期"
  52. end-placeholder="结束日期"
  53. size="mini"
  54. style="margin-right: 20px"
  55. >
  56. </el-date-picker>
  57. <el-button type="primary" size="small" class="search-button" @click="getData()">统计</el-button>
  58. <!-- <el-button type="primary" size="small" class="export-button" icon="el-icon-upload2">导出</el-button> -->
  59. </div>
  60. </div>
  61. <div class="main-content">
  62. <com-title :title="'每日最高水位统计'">
  63. <template slot="title-append">
  64. <div class="button-group">
  65. <el-button size="mini" v-for="item in typeList" :key="item.type" @click="changeShowNum(item.type)">
  66. <img :src="currentChartShowNum == item.type ? item.himg : item.img" />
  67. </el-button>
  68. <el-checkbox style="margin-left: 60px" v-model="onlyRainDay">只看降雨日</el-checkbox>
  69. </div>
  70. </template>
  71. </com-title>
  72. <div class="chart-content">
  73. <el-carousel
  74. class="chartCarousel"
  75. ref="chartCarousel"
  76. :autoplay="false"
  77. style="height: 100%; width: 100%"
  78. v-if="paramsList.length != 0"
  79. >
  80. <el-carousel-item
  81. v-for="(item, index) in paramsList"
  82. :key="index"
  83. :class="currentChartShowNum"
  84. style="height: 100%; width: 100%"
  85. >
  86. <template v-if="reload">
  87. <div class="chart-item" v-for="citem of item.child" :key="citem.id">
  88. <CompositeBarChart v-bind="$attrs" :chartData="citem" />
  89. </div>
  90. </template>
  91. </el-carousel-item>
  92. </el-carousel>
  93. <el-empty v-else :image="require('@/assets/icon/null.png')" description=" "></el-empty>
  94. </div>
  95. </div>
  96. </template>
  97. </com-side-page>
  98. </template>
  99. <script lang='ts'>
  100. import { Vue, Component, Prop, Watch } from 'vue-property-decorator'
  101. import { comTitle, comSidePage, CompositeBarChart } from '../../component/index'
  102. import { typeList } from '../../common/settings'
  103. import { getStagnantWaterStatistics, querySiteInfoByType } from '../../common/requestapis'
  104. import moment from 'moment'
  105. //内涝积水分析
  106. @Component({ name: 'WaterloggingAnalysis', components: { comTitle, comSidePage, CompositeBarChart } })
  107. export default class WaterloggingAnalysis extends Vue {
  108. onlyRainDay = true
  109. indexValue = []
  110. switchType = [
  111. { name: '近30天', type: '30' },
  112. { name: '近6个月', type: '6' },
  113. { name: '近1年', type: '1' }
  114. ]
  115. currentShowType: String = '30'
  116. searchDevice: String = ''
  117. customDate = []
  118. dateRange = null
  119. //树结构参数
  120. defaultcheckedList = []
  121. treeData = []
  122. defaultProps = {
  123. children: 'children',
  124. label: 'label'
  125. }
  126. selectStationList = []
  127. indexList = []
  128. //
  129. reload = true
  130. currentChartShowNum: String = 'fourth'
  131. paramsList = []
  132. get typeList() {
  133. return typeList
  134. }
  135. @Watch('searchDevice')
  136. filterText(val) {
  137. ;(this.$refs[`tree`] as Element[]).filter(val)
  138. }
  139. @Watch('customDate')
  140. onCustomeTimeChange(val) {
  141. if (val) {
  142. this.dateRange = {
  143. beginTime: moment(val[0]).format('YYYY-MM-DD HH:mm:ss'),
  144. endTime: moment(val[1]).format('YYYY-MM-DD HH:mm:ss')
  145. }
  146. this.currentShowType = null
  147. } else {
  148. this.currentShowType = '30'
  149. }
  150. }
  151. @Watch('currentShowType', { immediate: true })
  152. onTimeChangeMethod(val) {
  153. if (!val) return
  154. this.customDate = null
  155. let beginTime = null,
  156. endTime = null
  157. switch (val) {
  158. case '30':
  159. beginTime = moment().add(-1, 'M')
  160. endTime = moment(new Date())
  161. break
  162. case '6':
  163. beginTime = moment().add(-6, 'M')
  164. endTime = moment(new Date())
  165. break
  166. case '1':
  167. beginTime = moment().add(-1, 'y')
  168. endTime = moment(new Date())
  169. break
  170. }
  171. if (!beginTime || !endTime) return
  172. this.dateRange = {
  173. beginTime: beginTime.format('YYYY-MM-DD HH:mm:ss'),
  174. endTime: endTime.format('YYYY-MM-DD HH:mm:ss')
  175. }
  176. }
  177. @Watch('selectStationList')
  178. onSelectChange() {
  179. this.getData()
  180. }
  181. mounted() {
  182. this.getSites()
  183. }
  184. async getSites() {
  185. const res = await querySiteInfoByType({ type: 'pointFlood' })
  186. const { code, result } = res || {}
  187. if (code != 1 || result.length == 0) return
  188. let list = []
  189. result['pointFlood'].forEach((item, index) => {
  190. list.push({
  191. ...item,
  192. label: item.siteName,
  193. children: []
  194. })
  195. this.defaultcheckedList.push(item.id)
  196. })
  197. this.selectStationList = this.defaultcheckedList
  198. list = [{ label: '全部', children: list }]
  199. this.treeData = list
  200. }
  201. async getData() {
  202. if (!this.dateRange) {
  203. this.$message.warning('请选择统计时间')
  204. return
  205. }
  206. if (this.selectStationList.length == 0) {
  207. this.$message.warning('请选择至少一个雨量站')
  208. this.paramsList = []
  209. return
  210. }
  211. let params = {
  212. isRain: this.onlyRainDay ? 1 : 0,
  213. siteId: this.selectStationList.join(),
  214. start: this.dateRange.beginTime,
  215. end: this.dateRange.endTime
  216. }
  217. const res = await getStagnantWaterStatistics(params)
  218. this.indexList = res.result.map((i) => {
  219. const { rain, water, siteName } = i
  220. let xData = [],
  221. rainData = [],
  222. waterData = [],
  223. tableData = []
  224. //获取x轴数据
  225. rain.map((e) => xData.push(e.time))
  226. water.map((e) => xData.push(e.time))
  227. xData = [...new Set(xData)]
  228. xData.sort((a, b) => {
  229. return a.localeCompare(b)
  230. })
  231. //
  232. xData.forEach((item) => {
  233. let hasRain = rain.find((r) => r.time == item)
  234. let hasWater = water.find((w) => w.time == item)
  235. rainData.push(hasRain ? hasRain.sumVal : 0)
  236. waterData.push(hasWater ? hasWater.sumVal : 0)
  237. tableData.push({
  238. siteName,
  239. water: hasWater ? hasWater.sumVal : 0,
  240. rain: hasRain ? hasRain.sumVal : 0,
  241. time: item
  242. })
  243. })
  244. tableData.sort((a, b) => {
  245. return b.time.localeCompare(a.time)
  246. })
  247. return { ...i, xData, rainData, waterData, tableData }
  248. })
  249. this.assembleParamsList()
  250. }
  251. assembleParamsList() {
  252. let type = this.currentChartShowNum
  253. let list = this.indexList
  254. this.paramsList = []
  255. if (type == 'single') {
  256. list.forEach((item) => {
  257. this.paramsList.push({ child: [item] })
  258. })
  259. } else {
  260. let dividend = type == 'fourth' ? 4 : type == 'sixth' ? 6 : type == 'ninth' ? 9 : 1
  261. let temp = [],
  262. symbol = 0
  263. list.forEach((item, index) => {
  264. if ((index + 1) % dividend != 0) {
  265. temp[symbol] ? temp[symbol].list.push({ ...item }) : temp.push({ list: [{ ...item }] })
  266. } else {
  267. temp[symbol].list.length < dividend ? temp[symbol].list.push({ ...item }) : temp.push({ list: [{ ...item }] })
  268. symbol++
  269. }
  270. })
  271. temp.forEach((item) => {
  272. this.paramsList.push({ child: item.list })
  273. })
  274. }
  275. }
  276. //树形列表过滤查找
  277. filterNode(value, data) {
  278. if (!value) return true
  279. return data.label.indexOf(value) !== -1
  280. }
  281. handleCheckChange(current, checkedData) {
  282. this.selectStationList = checkedData.checkedKeys.filter((i) => i)
  283. }
  284. tagClose(tag) {
  285. this.indexValue.splice(
  286. this.indexValue.findIndex((e) => e == tag),
  287. 1
  288. )
  289. }
  290. //图表显示数量
  291. changeShowNum(type) {
  292. if (this.paramsList.length == 0) return
  293. this.reload = false
  294. this.currentChartShowNum = type
  295. this.assembleParamsList()
  296. //
  297. ;(this.$refs[`chartCarousel`] as any).setActiveItem(0)
  298. //重新渲染界面
  299. this.$nextTick(() => {
  300. this.reload = true
  301. })
  302. }
  303. }
  304. </script>
  305. <style lang="scss" scoped>
  306. @mixin center() {
  307. display: flex;
  308. align-items: center;
  309. justify-content: center;
  310. }
  311. @mixin fontSet() {
  312. font-family: Source Han Sans CN;
  313. font-style: normal;
  314. font-weight: 500;
  315. font-size: 14px;
  316. }
  317. $bcolor: #f8f8f8;
  318. .side-title,
  319. .side-content {
  320. background: $bcolor;
  321. }
  322. .side-title {
  323. @include center();
  324. height: 44px;
  325. width: 100%;
  326. margin-bottom: 4px;
  327. font-family: Alibaba PuHuiTi;
  328. font-style: normal;
  329. font-weight: 500;
  330. font-size: 16px;
  331. color: #333333;
  332. }
  333. .side-content {
  334. height: calc(100% - 48px);
  335. width: 100%;
  336. overflow: auto;
  337. .searchInput {
  338. padding: 10px;
  339. }
  340. .dataListTree {
  341. background-color: $bcolor;
  342. }
  343. }
  344. .main-title {
  345. display: flex;
  346. width: 100%;
  347. margin-bottom: 14px;
  348. position: relative;
  349. flex-flow: column;
  350. border-bottom: 1px solid #dedede;
  351. .mt-top {
  352. display: flex;
  353. align-items: center;
  354. margin-bottom: 20px;
  355. }
  356. .mt-bottom {
  357. display: flex;
  358. align-items: center;
  359. margin-bottom: 15px;
  360. .switch {
  361. @include center();
  362. border: 1px solid #dedede;
  363. border-radius: 3px;
  364. overflow: hidden;
  365. margin-right: 20px;
  366. .switch-item {
  367. @include center();
  368. width: 60px;
  369. height: 34px;
  370. white-space: nowrap;
  371. line-height: 21px;
  372. letter-spacing: 0.01em;
  373. color: #333333;
  374. cursor: pointer;
  375. }
  376. .switch-item:not(:last-child) {
  377. border-right: 1px solid #dedede;
  378. padding: 3px 10px;
  379. }
  380. .switch-active {
  381. color: #ffffff;
  382. background: #2083e8;
  383. }
  384. }
  385. .el-checkbox {
  386. margin-right: 15px;
  387. }
  388. .search-button {
  389. margin-left: 20px;
  390. background-color: #2d74e7;
  391. border-color: #2d74e7;
  392. }
  393. .export-button {
  394. right: 0;
  395. position: absolute;
  396. background-color: #e1ebfc;
  397. border-color: #e1ebfc;
  398. border-radius: 4px;
  399. color: #555555;
  400. >>> i {
  401. color: #2d74e7;
  402. font-weight: bold;
  403. }
  404. }
  405. }
  406. }
  407. .main-content {
  408. margin-top: 14px;
  409. height: calc(100% - 112px);
  410. width: 100%;
  411. .button-group {
  412. margin-left: 60px;
  413. display: flex;
  414. align-items: center;
  415. .el-button {
  416. margin-left: 20px;
  417. line-height: 0;
  418. padding: 0;
  419. img {
  420. object-fit: contain;
  421. }
  422. }
  423. >>> .el-checkbox__label {
  424. font-family: Source Han Sans CN-Medium;
  425. }
  426. }
  427. .chart-content {
  428. width: 100%;
  429. height: calc(100% - 24px);
  430. overflow: auto;
  431. .chart-item {
  432. height: 100%;
  433. width: 100%;
  434. overflow: hidden;
  435. }
  436. >>> .el-carousel {
  437. height: 100%;
  438. width: 100%;
  439. .el-carousel__container {
  440. height: 100%;
  441. width: 100%;
  442. }
  443. .el-carousel__item {
  444. height: 100%;
  445. width: 100%;
  446. display: grid;
  447. }
  448. .single {
  449. grid-template-rows: repeat(1, 1fr);
  450. grid-template-columns: repeat(1, 1fr);
  451. }
  452. .fourth {
  453. grid-template-rows: repeat(2, 1fr);
  454. grid-template-columns: repeat(2, 1fr);
  455. }
  456. .sixth {
  457. grid-template-rows: repeat(2, 1fr);
  458. grid-template-columns: repeat(3, 1fr);
  459. }
  460. .ninth {
  461. grid-template-rows: repeat(3, 1fr);
  462. grid-template-columns: repeat(3, 1fr);
  463. }
  464. }
  465. }
  466. }
  467. </style>