FanDeviceEcharts.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481
  1. <template>
  2. <div class="charts-container">
  3. <a-select ref="select" v-model:value="chartsType" size="small" style="position: absolute; z-index: 99; top: 2px; left: 2px; width: 98px">
  4. <a-select-option value="listMonitor">实时监测</a-select-option>
  5. <a-select-option value="detail">详情监测</a-select-option>
  6. <a-select-option value="history">历史记录</a-select-option>
  7. </a-select>
  8. <div class="charts-box" v-if="chartsType === 'listMonitor'" style="position: absolute; top: 20px">
  9. <div class="echarts-group">
  10. <BarAndLine
  11. class="echarts-item"
  12. :chartsColumnsType="chartsColumnsType"
  13. :xAxisPropType="xAxisPropType"
  14. :dataSource="dataSource"
  15. height="100%"
  16. chartsType="listMonitor"
  17. :option="echartsOption"
  18. :chartsColumns="fan1ChartsColumns"
  19. style="border-right: 1px solid #fff"
  20. />
  21. <BarAndLine
  22. class="echarts-item"
  23. :chartsColumnsType="chartsColumnsType"
  24. :xAxisPropType="xAxisPropType"
  25. :dataSource="dataSource"
  26. height="100%"
  27. chartsType="listMonitor"
  28. :option="echartsOption"
  29. :chartsColumns="fan2ChartsColumns"
  30. />
  31. </div>
  32. </div>
  33. <div class="charts-box" v-else-if="chartsType === 'detail' && deviceListApi">
  34. <Select
  35. :options="options"
  36. :fieldNames="{ label: 'strname', value: 'deviceID' }"
  37. v-model:value="deviceId"
  38. placeholder="请选择查看的设备"
  39. size="small"
  40. style="position: absolute; z-index: 99; left: 102px; width: 150px; top: 2px"
  41. />
  42. <div class="echarts-group">
  43. <BarAndLine
  44. class="echarts-item"
  45. :chartsColumnsType="chartsColumnsType"
  46. :xAxisPropType="resultXAxisPropType"
  47. :dataSource="detailDataSource"
  48. height="100%"
  49. :option="echartsOption2"
  50. chartsType="detail"
  51. :chartsColumns="fan1ChartsColumns"
  52. style="border-right: 1px solid #fff"
  53. />
  54. <BarAndLine
  55. class="echarts-item"
  56. :chartsColumnsType="chartsColumnsType"
  57. :xAxisPropType="resultXAxisPropType"
  58. :dataSource="detailDataSource"
  59. height="100%"
  60. :option="echartsOption2"
  61. chartsType="detail"
  62. :chartsColumns="fan2ChartsColumns"
  63. />
  64. </div>
  65. </div>
  66. <div class="charts-box" v-else-if="chartsType === 'history'">
  67. <Select
  68. :options="options"
  69. :fieldNames="{ label: 'strname', value: 'deviceID' }"
  70. v-model:value="deviceId"
  71. placeholder="请选择查看的设备"
  72. size="small"
  73. style="position: absolute; z-index: 99; left: 102px; width: 150px; top: 2px"
  74. />
  75. <template v-if="globalConfig.History_Type == 'vent'">
  76. <a-date-picker
  77. v-model:value="historyParams.ttime_begin"
  78. valueFormat="YYYY-MM-DD HH:mm:ss"
  79. placeholder="请选择开始日期"
  80. size="small"
  81. :showTime="true"
  82. style="position: absolute; z-index: 99; left: 254px; width: 170px; top: 2px"
  83. />
  84. <a-date-picker
  85. v-model:value="historyParams.ttime_end"
  86. valueFormat="YYYY-MM-DD HH:mm:ss"
  87. placeholder="请选择结束日期"
  88. size="small"
  89. :showTime="true"
  90. style="position: absolute; z-index: 99; left: 426px; width: 170px; top: 2px"
  91. />
  92. <a-select
  93. ref="select"
  94. v-model:value="historyParams.skip"
  95. placeholder="请选择间隔时间"
  96. size="small"
  97. style="position: absolute; z-index: 99; top: 2px; left: 598px; width: 150px"
  98. >
  99. <a-select-option value="1">1秒</a-select-option>
  100. <a-select-option value="2">5秒</a-select-option>
  101. <a-select-option value="3">10秒</a-select-option>
  102. <a-select-option value="4">30分钟</a-select-option>
  103. <a-select-option value="5">1分钟</a-select-option>
  104. <a-select-option value="6">10分钟</a-select-option>
  105. <a-select-option value="7">30分钟</a-select-option>
  106. <a-select-option value="8">1小时</a-select-option>
  107. </a-select>
  108. </template>
  109. <template v-else>
  110. <a-date-picker
  111. v-model:value="historyParams.startTime"
  112. valueFormat="YYYY-MM-DD HH:mm:ss"
  113. placeholder="开始时间"
  114. size="small"
  115. :showTime="true"
  116. style="position: absolute; z-index: 99; left: 254px; width: 170px; top: 2px"
  117. />
  118. <a-date-picker
  119. v-model:value="historyParams.endTime"
  120. valueFormat="YYYY-MM-DD HH:mm:ss"
  121. placeholder="结束时间"
  122. size="small"
  123. :showTime="true"
  124. style="position: absolute; z-index: 99; left: 426px; width: 170px; top: 2px"
  125. />
  126. <a-select
  127. ref="select"
  128. v-model:value="historyParams.interval"
  129. placeholder="请选择间隔时间"
  130. size="small"
  131. style="position: absolute; z-index: 99; top: 2px; left: 598px; width: 150px"
  132. >
  133. <a-select-option value="1s">1秒</a-select-option>
  134. <a-select-option value="5s">5秒</a-select-option>
  135. <a-select-option value="10s">10秒</a-select-option>
  136. <a-select-option value="30s">30秒</a-select-option>
  137. <a-select-option value="1m">1分钟</a-select-option>
  138. <a-select-option value="10m">5分钟</a-select-option>
  139. <a-select-option value="30m">10分钟</a-select-option>
  140. <a-select-option value="1h">1小时</a-select-option>
  141. </a-select>
  142. </template>
  143. <Pagination
  144. size="small"
  145. v-model:current="currentPage"
  146. v-model:page-size="pageSize"
  147. :total="total"
  148. :show-total="(total) => `共 ${total} 条`"
  149. style="position: absolute; z-index: 99; top: 2px; right: 30px"
  150. />
  151. <div class="echarts-group">
  152. <BarAndLine
  153. class="echarts-item"
  154. :chartsColumnsType="chartsColumnsType"
  155. :xAxisPropType="resultXAxisPropType"
  156. :dataSource="resultDataSource"
  157. height="100%"
  158. :option="echartsOption1"
  159. chartsType="history"
  160. style="margin-top: 20px; border-right: 1px solid #fff"
  161. :chartsColumns="fan1ChartsColumns"
  162. />
  163. <BarAndLine
  164. class="echarts-item"
  165. :chartsColumnsType="chartsColumnsType"
  166. :xAxisPropType="resultXAxisPropType"
  167. :dataSource="resultDataSource"
  168. height="100%"
  169. :option="echartsOption1"
  170. chartsType="history"
  171. style="margin-top: 20px"
  172. :chartsColumns="fan2ChartsColumns"
  173. />
  174. </div>
  175. </div>
  176. </div>
  177. </template>
  178. <script lang="ts">
  179. import { ref, defineComponent, watch, reactive, onMounted, watchEffect, inject } from 'vue';
  180. import BarAndLine from '/@/components/chart/BarAndLine.vue';
  181. import dayjs from 'dayjs';
  182. import { defHttp } from '/@/utils/http/axios';
  183. import { Select, Pagination } from 'ant-design-vue';
  184. export default defineComponent({
  185. name: 'DeviceEcharts',
  186. components: { BarAndLine, Select, Pagination },
  187. props: {
  188. fan1ChartsColumns: {
  189. type: Array,
  190. default: () => [],
  191. },
  192. fan2ChartsColumns: {
  193. type: Array,
  194. default: () => [],
  195. },
  196. chartsColumnsHistory: {
  197. type: Array,
  198. default: () => [],
  199. },
  200. chartsColumnsType: {
  201. type: String,
  202. required: true,
  203. },
  204. dataSource: {
  205. type: Array,
  206. default: () => [],
  207. },
  208. deviceListApi: {
  209. type: Function,
  210. required: true,
  211. },
  212. deviceType: {
  213. type: String,
  214. required: true,
  215. },
  216. option: {
  217. type: Object,
  218. default: () => ({}),
  219. },
  220. xAxisPropType: {
  221. type: String,
  222. required: true,
  223. },
  224. },
  225. setup(props) {
  226. const globalConfig = inject('globalConfig');
  227. let historyList;
  228. const chartsType = ref('history');
  229. const deviceId = ref('');
  230. const options = ref([]);
  231. const historyParams = reactive({
  232. ttime_begin: dayjs().startOf('date').format('YYYY-MM-DD HH:mm:ss'),
  233. ttime_end: dayjs().format('YYYY-MM-DD HH:mm:ss'),
  234. skip: '8',
  235. startTime: dayjs().startOf('date').format('YYYY-MM-DD HH:mm:ss'),
  236. endTime: dayjs().format('YYYY-MM-DD HH:mm:ss'),
  237. interval: '1h',
  238. });
  239. const resultXAxisPropType = ref('');
  240. const resultDataSource = ref<any[]>([]);
  241. const detailDataSource = ref<any[]>([]);
  242. const currentPage = ref<number>(1);
  243. const pageSize = ref<number>(20);
  244. const total = ref(0);
  245. const echartsOption = {
  246. grid: {
  247. top: '50px',
  248. left: '10px',
  249. right: props.fan1ChartsColumns.length * 15 + 'px',
  250. bottom: '15px',
  251. containLabel: true,
  252. },
  253. toolbox: {
  254. feature: {},
  255. },
  256. xAxis: {
  257. axisLabel: {
  258. interval: 0,
  259. },
  260. },
  261. };
  262. const echartsOption1 = {
  263. grid: {
  264. top: '60px',
  265. left: '10px',
  266. right: props.fan1ChartsColumns.length * 15 + 'px',
  267. bottom: '35px',
  268. containLabel: true,
  269. },
  270. toolbox: {
  271. feature: {},
  272. },
  273. legend: {
  274. top: 30,
  275. },
  276. xAxis: {
  277. interval: 0,
  278. },
  279. };
  280. const echartsOption2 = {
  281. grid: {
  282. top: '70px',
  283. left: '10px',
  284. right: props.fan1ChartsColumns.length * 15 + 'px',
  285. bottom: '5px',
  286. containLabel: true,
  287. },
  288. toolbox: {
  289. feature: {},
  290. },
  291. };
  292. const onChange = (pageNumber: number) => {
  293. console.log('Page: ', pageNumber);
  294. };
  295. watch(
  296. [chartsType, deviceId, historyParams, pageSize, currentPage],
  297. async (
  298. [newChartsType, newDeviceId, newHistoryParams, newPageSize, newCurrentPage],
  299. [oldChartsType, oldDeviceId, oldHistoryParams, oldPageSize, oldCurrentPage]
  300. ) => {
  301. console.log('[ historyParams ] >', historyParams.ttime, dayjs(historyParams.ttime).format('HH:mm:ss'));
  302. if (newChartsType === 'listMonitor') {
  303. // 实时监测所有
  304. resultDataSource.value = props.dataSource;
  305. } else if (newChartsType === 'history') {
  306. resultDataSource.value = [];
  307. // 历史
  308. if (newChartsType !== oldChartsType || newDeviceId !== oldDeviceId) {
  309. currentPage.value = 1;
  310. }
  311. const device = options.value.find((device) => device['deviceID'] === newDeviceId);
  312. if (device) {
  313. let res;
  314. if (props.dataSource['stationtype'] !== 'redis') {
  315. resultXAxisPropType.value = 'ttime';
  316. historyList = (params) => defHttp.get({ url: '/safety/ventanalyMonitorData/listdays', params });
  317. const datas1 = await historyList({
  318. ttime_begin: newHistoryParams.ttime_begin,
  319. ttime_end: newHistoryParams.ttime_end,
  320. strtype: device.deviceType,
  321. gdeviceid: newDeviceId,
  322. skip: historyParams.skip,
  323. pageSize: pageSize.value,
  324. pageNo: currentPage.value,
  325. column: 'createTime',
  326. deviceNum: 'Fan1',
  327. });
  328. const datas2 = await historyList({
  329. ttime_begin: newHistoryParams.ttime_begin,
  330. ttime_end: newHistoryParams.ttime_end,
  331. strtype: device.deviceType,
  332. gdeviceid: newDeviceId,
  333. skip: historyParams.skip,
  334. pageSize: pageSize.value,
  335. pageNo: currentPage.value,
  336. column: 'createTime',
  337. deviceNum: 'Fan2',
  338. });
  339. res = datas1['datalist']['records'];
  340. const res2 = datas2['datalist']['records'];
  341. if (res && res.length > 0) {
  342. resultDataSource.value = res.map((item, index) => Object.assign(item, item.readData, res2[index]['readData']));
  343. } else {
  344. resultDataSource.value = [];
  345. }
  346. total.value = datas1['datalist'].total;
  347. } else {
  348. historyList = (params) => defHttp.post({ url: '/monitor/history/getHistoryData', params });
  349. resultXAxisPropType.value = 'time';
  350. res = await historyList({
  351. pageSize: pageSize.value,
  352. pageNum: currentPage.value,
  353. startTime: newHistoryParams.ttime_begin,
  354. endTime: newHistoryParams.ttime_end,
  355. deviceId: newDeviceId,
  356. strtype: device.deviceType,
  357. interval: historyParams.interval,
  358. column: 'createTime',
  359. deviceNum: 'Fan2',
  360. });
  361. if (res && res.records && res.records.length > 0) {
  362. resultDataSource.value = res.records.map((item) => Object.assign(item, item.readData));
  363. } else {
  364. resultDataSource.value = [];
  365. }
  366. total.value = res.total;
  367. }
  368. }
  369. } else if (newChartsType === 'detail') {
  370. // 设备详情
  371. resultXAxisPropType.value = 'readTime';
  372. if (newDeviceId !== oldDeviceId) {
  373. detailDataSource.value = [];
  374. }
  375. }
  376. }
  377. );
  378. watchEffect(() => {
  379. if (chartsType.value === 'detail') {
  380. const currentData = props.dataSource.find((item: any) => item.deviceID === deviceId.value);
  381. if (currentData) {
  382. const isHas = detailDataSource.value.find((item) => item[resultXAxisPropType.value] === currentData[resultXAxisPropType.value]);
  383. if (!isHas) {
  384. if (detailDataSource.value.length < 15) {
  385. detailDataSource.value.push(currentData);
  386. } else {
  387. detailDataSource.value.shift();
  388. detailDataSource.value.push(currentData);
  389. }
  390. }
  391. }
  392. }
  393. });
  394. onMounted(async () => {
  395. const res = await props.deviceListApi();
  396. // debugger;
  397. if (res['msgTxt'] && res['msgTxt'][0] && res['msgTxt'][0]['datalist']) {
  398. options.value = res['msgTxt'][0]['datalist'];
  399. deviceId.value = options.value[0]['deviceID'];
  400. }
  401. });
  402. return {
  403. chartsType,
  404. deviceId,
  405. resultDataSource,
  406. historyParams,
  407. options,
  408. resultXAxisPropType,
  409. detailDataSource,
  410. currentPage,
  411. pageSize,
  412. total,
  413. echartsOption,
  414. echartsOption1,
  415. echartsOption2,
  416. onChange,
  417. globalConfig,
  418. };
  419. },
  420. });
  421. </script>
  422. <style lang="less">
  423. :deep(.vent-select-dropdown) {
  424. color: #000 !important;
  425. .vent-select-item {
  426. color: #000 !important;
  427. }
  428. }
  429. </style>
  430. <style lang="less" scoped>
  431. @import '/@/design/theme.less';
  432. .charts-container {
  433. position: relative;
  434. height: 100%;
  435. .charts-box {
  436. width: 100%;
  437. height: 100%;
  438. position: absolute;
  439. bottom: 0;
  440. top: 0px;
  441. .echarts-group {
  442. height: 100%;
  443. display: flex;
  444. :deep(div) {
  445. border-right: 1px solid #ffffff22 !important;
  446. }
  447. }
  448. }
  449. .@{ventSpace}-picker,
  450. .@{ventSpace}-select-selector {
  451. background: #00000017 !important;
  452. border: 1px solid @vent-form-item-border !important;
  453. input,
  454. .@{ventSpace}-select-selection-item,
  455. .@{ventSpace}-picker-suffix {
  456. color: #fff !important;
  457. }
  458. .@{ventSpace}-select-selection-placeholder {
  459. color: #b7b7b7 !important;
  460. }
  461. }
  462. .@{ventSpace}-select-arrow,
  463. .@{ventSpace}-picker-separator {
  464. color: #fff !important;
  465. }
  466. }
  467. :deep(.@{ventSpace}-select-dropdown) {
  468. color: #000 !important;
  469. .@{ventSpace}-select-item {
  470. color: #000 !important;
  471. }
  472. }
  473. </style>