| | |
| | | <template> |
| | | <el-row> |
| | | <el-col span="2" class="flex-col"> |
| | | <el-col v-if="direction == 'right'" span="2" class="flex-col"> |
| | | <el-row justify="end"> |
| | | <CardButton |
| | | direction="left" |
| | | name="动态溯源" |
| | | @click="() => (show = !show)" |
| | | @click="handleDisplayChange" |
| | | ></CardButton> |
| | | </el-row> |
| | | <el-row class="flex-col"> |
| | |
| | | </template> |
| | | </BaseCard> |
| | | </el-col> |
| | | <el-col v-if="direction == 'left'" span="2" class="flex-col"> |
| | | <el-row justify="start"> |
| | | <CardButton |
| | | direction="right" |
| | | name="动态溯源" |
| | | @click="handleDisplayChange" |
| | | ></CardButton> |
| | | </el-row> |
| | | <el-row class="flex-col"> |
| | | <PollutedExceptionItem |
| | | :item="selectedException" |
| | | @showMarksAndPolygon="showMarksAndPolygon" |
| | | ></PollutedExceptionItem> |
| | | </el-row> |
| | | </el-col> |
| | | <PollutedClueItem |
| | | v-model="clueDialog" |
| | | :item="selectedClue" |
| | |
| | | import moment from 'moment'; |
| | | |
| | | import websocket from '@/api/websocket'; |
| | | import dataAnalysisApi from '@/api/dataAnalysisApi'; |
| | | import sector from '@/utils/map/sector'; |
| | | import mapUtil from '@/utils/map/util'; |
| | | import { sceneTypes, sceneIcon } from '@/constant/scene-types'; |
| | |
| | | |
| | | const NO_SCENE = 'no_scene'; |
| | | |
| | | const MODE_REALTIME = 'realtime'; |
| | | const MODE_HISTORY = 'history'; |
| | | |
| | | const props = defineProps({ |
| | | factorType: String |
| | | direction: { |
| | | type: String, |
| | | default: 'left' |
| | | }, |
| | | factorType: String, |
| | | // 模式,realtime:实时模式;history:历史数据模式 |
| | | mode: { |
| | | type: String, |
| | | default: 'realtime' |
| | | }, |
| | | missionCode: String, |
| | | deviceCode: String |
| | | }); |
| | | |
| | | const emits = defineEmits(['update:factorType']); |
| | | |
| | | const height = `30vh`; |
| | | const width = `60vh`; |
| | | |
| | | const show = ref(false); |
| | | const clueDialog = ref(false); |
| | |
| | | const selectedSceneTypes = ref([]); |
| | | const sceneOptions = ref([]); |
| | | |
| | | function handleDisplayChange() { |
| | | show.value = !show.value; |
| | | if ( |
| | | !show.value && |
| | | selectedException.value && |
| | | selectedException.value.showMore |
| | | ) { |
| | | showMarksAndPolygon(selectedException.value); |
| | | } |
| | | } |
| | | |
| | | function scrollToBottom() { |
| | | const h1 = scrollContentRef.value.clientHeight + 100; |
| | | setTimeout(() => { |
| | |
| | | }, 100); |
| | | } |
| | | |
| | | const streams = reactive([]); |
| | | const streams = ref([]); |
| | | const filterStreams = computed(() => { |
| | | return streams.filter((v) => { |
| | | return streams.value.filter((v) => { |
| | | // 判断消息类型是否选中 |
| | | const b1 = selectedMsgTypes.value.indexOf(v._type) != -1; |
| | | let b2, b3; |
| | |
| | | type2: 0, |
| | | type3: 0 |
| | | }; |
| | | streams.forEach((v) => { |
| | | streams.value.forEach((v) => { |
| | | switch (v._type) { |
| | | case '1': |
| | | count.type1++; |
| | |
| | | function dealMsg(data) { |
| | | const { type, content } = websocketMsgParser.parseMsg(data); |
| | | const obj = reactive(JSON.parse(content)); |
| | | if (obj.deviceCode == props.deviceCode) { |
| | | obj._type = type; |
| | | dealObj(obj); |
| | | } |
| | | } |
| | | |
| | | function dealObj(obj) { |
| | | // 污染线索 PollutedClue |
| | | if (type == '1') { |
| | | if (obj._type == '1') { |
| | | obj.showMore = false; |
| | | console.log('污染异常切片: ', obj); |
| | | |
| | |
| | | // scrollToTop(); |
| | | // drawPolygon(obj.pollutedArea); |
| | | parseChartData(obj); |
| | | } else if (type == '2') { |
| | | } else if (obj._type == '2') { |
| | | // const obj = JSON.parse(content); |
| | | // obj._type = type; |
| | | console.log('污染线索结果: ', obj); |
| | | obj._timestr = timeFormatter(obj.time); |
| | | addNewMsg(obj); |
| | | } else if (type == '3') { |
| | | } else if (obj._type == '3') { |
| | | console.log('污染提醒切片: ', obj); |
| | | addNewMsg(obj); |
| | | parseChartData(obj); |
| | |
| | | const leftMsgList = []; |
| | | function addNewMsg(msg, inside) { |
| | | if (!addNewMsgTask && (leftMsgList.length == 0 || inside)) { |
| | | streams.splice(0, 0, msg); |
| | | streams.value.splice(0, 0, msg); |
| | | addNewMsgTask = setTimeout(() => { |
| | | clearTimeout(addNewMsgTask); |
| | | addNewMsgTask = undefined; |
| | |
| | | } |
| | | } |
| | | |
| | | function fetchPollutionTraceHistory() { |
| | | dataAnalysisApi.fetchPollutionTraceHistory(props.missionCode).then((res) => { |
| | | const objList = JSON.parse(res.data); |
| | | objList.forEach((obj) => { |
| | | obj._type = obj.msgType + ''; |
| | | if (obj._type == '1') { |
| | | obj.showMore = false; |
| | | show.value = true; |
| | | parseChartData(obj); |
| | | } else if (obj._type == '2') { |
| | | obj._timestr = timeFormatter(obj.time); |
| | | } else if (obj._type == '3') { |
| | | parseChartData(obj); |
| | | } |
| | | optionsFilte(obj); |
| | | }); |
| | | streams.value = objList; |
| | | }); |
| | | } |
| | | |
| | | onMounted(() => { |
| | | if (props.mode == MODE_REALTIME) { |
| | | websocket.registerReceiveEvent(dealMsg); |
| | | } else if (props.missionCode) { |
| | | fetchPollutionTraceHistory(); |
| | | } |
| | | }); |
| | | onUnmounted(() => { |
| | | websocket.removeReceiveEvent(dealMsg); |
| | |
| | | selectedClue.value._selected = false; |
| | | } |
| | | }); |
| | | |
| | | watch( |
| | | () => props.missionCode, |
| | | (nV, oV) => { |
| | | if (nV != oV) { |
| | | fetchPollutionTraceHistory(); |
| | | } |
| | | } |
| | | ); |
| | | |
| | | function handleOpen(item) { |
| | | switch (item._type) { |
| | |
| | | } |
| | | |
| | | function hideAll() { |
| | | streams.forEach((s) => { |
| | | streams.value.forEach((s) => { |
| | | if (polygonMap.has(s.guid)) { |
| | | s.showMore = false; |
| | | map.remove(polygonMap.get(s.guid)); |
| | |
| | | |
| | | function parseChartData(obj) { |
| | | // console.log('折线图:start'); |
| | | obj.pollutedData._startTime = moment(obj.pollutedData.startTime).format( |
| | | 'HH:mm:ss' |
| | | ); |
| | | obj.pollutedData._endTime = moment(obj.pollutedData.endTime).format( |
| | | 'HH:mm:ss' |
| | | ); |
| | | const factorDatas = new FactorDatas(); |
| | | factorDatas.setData(obj.pollutedData.historyDataList, 0, () => { |
| | | obj._chartOptions = factorDataParser.parseData(factorDatas, [ |
| | |
| | | |
| | | function timeFormatter(time) { |
| | | return moment(time).format('YYYY-MM-DD HH:mm:ss'); |
| | | } |
| | | |
| | | /******************************************************************************************************************** */ |
| | | |
| | | /** |
| | | * 添加一条工作流信息 |
| | | * @param {*} data |
| | | */ |
| | | const putWorkStream = (data) => { |
| | | data.substring(); |
| | | const obj = JSON.parse(data); |
| | | console.log('sourcetrace: ', obj); |
| | | |
| | | obj._statusStr = exceptionStatus(obj.status); |
| | | |
| | | if (streams.length == 0) { |
| | | streams.push(obj); |
| | | } else { |
| | | const index = streams.findIndex((v) => { |
| | | return v.guid == obj.guid; |
| | | }); |
| | | if (index != -1) { |
| | | const old = streams[index]; |
| | | obj.showMore = old.showMore; |
| | | old.relatedSceneList.forEach((s) => { |
| | | const index = obj.relatedSceneList.findIndex((v) => { |
| | | return v.guid == s.guid; |
| | | }); |
| | | if (index == -1) { |
| | | obj.relatedSceneList.push(s); |
| | | } |
| | | }); |
| | | streams.splice(index, 1, obj); |
| | | } else { |
| | | streams.unshift(obj); |
| | | } |
| | | |
| | | show.value = true; |
| | | } |
| | | |
| | | // scrollToBottom(); |
| | | scrollToTop(); |
| | | |
| | | const excObj = streams.find((v) => { |
| | | return v.factorId + '' == props.factorType; |
| | | }); |
| | | if (excObj) { |
| | | drawSector(excObj); |
| | | } |
| | | }; |
| | | |
| | | function exceptionStatus(value) { |
| | | switch (value) { |
| | | case 1: |
| | | return '异常发生中'; |
| | | case 2: |
| | | return '异常已结束'; |
| | | default: |
| | | return '异常已结束'; |
| | | } |
| | | } |
| | | |
| | | let lastDrawObjGuid; |
| | | function drawSector(exceptionObj) { |
| | | emits('update:factorType', exceptionObj.factorId + ''); |
| | | setTimeout(() => { |
| | | if (exceptionObj.guid != lastDrawObjGuid) { |
| | | sector.clearSectorPt(); |
| | | lastDrawObjGuid = exceptionObj.guid; |
| | | } |
| | | // 1. 绘制新扇形区域 |
| | | const datavo = exceptionObj.midData; |
| | | const factorDatas = new FactorDatas(); |
| | | factorDatas.setData([datavo], 0, () => { |
| | | const pr = sector.drawSectorPt(factorDatas, 0); |
| | | // 调整视角居中显示 |
| | | // mapUtil.setCenter(pr.p); |
| | | drawMarks(exceptionObj.relatedSceneList); |
| | | }); |
| | | }, 1000); |
| | | } |
| | | |
| | | let layer = undefined; |
| | | function drawMarks(sceneList) { |
| | | if (layer != undefined) { |
| | | mapUtil.removeViews(layer); |
| | | // layer = undefined; |
| | | } |
| | | if (sceneList.length != 0) { |
| | | const icons = []; |
| | | sceneList.forEach((s) => { |
| | | icons.push(sceneIcon(s.typeId)); |
| | | }); |
| | | layer = marks.createLabelMarks(icons, sceneList, true); |
| | | } |
| | | } |
| | | </script> |
| | | <style scoped> |
| | |
| | | .scrollbar { |
| | | width: 400px; |
| | | /* max-width: 60vw; */ |
| | | height: 45vh; |
| | | height: 65vh; |
| | | /* color: #02ffea; */ |
| | | padding-right: 10px; |
| | | } |