diff --git a/api/.vscode/launch.json b/api/.vscode/launch.json
index 5433cf2..6bf3afa 100644
--- a/api/.vscode/launch.json
+++ b/api/.vscode/launch.json
@@ -22,6 +22,7 @@
"-d postgres/example/10.8.30.160/30432",
"-w https://smartwater.anxinyun.cn",
"-a https://smartworksafety.anxinyun.cn",
+ "--tf http://91mogo.com/onecity/v5",
]
},
{
diff --git a/api/app/lib/controllers/superScreen/fire.js b/api/app/lib/controllers/superScreen/fire.js
index 8bc189a..f39a986 100644
--- a/api/app/lib/controllers/superScreen/fire.js
+++ b/api/app/lib/controllers/superScreen/fire.js
@@ -1,4 +1,6 @@
'use strict';
+const request = require("superagent");
+const moment = require("moment");
function getFireAlarmList(opts) {
return async function (ctx, next) {
@@ -56,8 +58,53 @@ function updateAlarm(opts) {
}
}
+// 获取消防设备
+function getFireDevice(opts) {
+ return async function (ctx, next) {
+ let rslt = {};
+ try {
+ let url = `${opts.tfApi}/device/alarm/statistic?_timeStamp=${new Date().getTime()}`;
+ const res = (await request.get(url).set("X-API-TOKEN", "9911510d-9dae-438d-9b6e-0545ef003c58")).body.data;
+ rslt = res.items;
+ ctx.status = 200;
+ ctx.body = rslt;
+ } catch (error) {
+ ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
+ ctx.status = 400;
+ ctx.body = { message: '获取消防设备失败' }
+ }
+ }
+}
+
+// 获取火情趋势
+function getFireTrend(opts) {
+ return async function (ctx, next) {
+ let rslt = [];
+ try {
+ let url = `${opts.tfApi}/alarms/trend?key=1`;
+ const res = (await request.get(url).set("X-API-TOKEN", "9911510d-9dae-438d-9b6e-0545ef003c58")).body.data;
+ let start = moment().add(-6, 'day').startOf('day').valueOf();
+ let end = moment().endOf('day').valueOf();
+ res.map(item => {
+ const time = moment(item.time).valueOf();
+ if (time >= start && time <= end) {
+ rslt.push({ time: item.time, count: item.val })
+ }
+ })
+ ctx.status = 200;
+ ctx.body = rslt;
+ } catch (error) {
+ ctx.fs.logger.error(`path: ${ctx.path}, error: ${error}`);
+ ctx.status = 400;
+ ctx.body = { message: '获取火情趋势失败' }
+ }
+ }
+}
+
module.exports = {
addAlarm,
updateAlarm,
- getFireAlarmList
+ getFireAlarmList,
+ getFireDevice,
+ getFireTrend
}
\ No newline at end of file
diff --git a/api/app/lib/middlewares/authenticator.js b/api/app/lib/middlewares/authenticator.js
index 07df3ec..cc625f3 100644
--- a/api/app/lib/middlewares/authenticator.js
+++ b/api/app/lib/middlewares/authenticator.js
@@ -64,6 +64,8 @@ let isPathExcluded = function (opts, path, method) {
excludeOpts.push({ p: '/fire/alarm', o: 'GET' });
excludeOpts.push({ p: '/fire/alarm', o: 'POST' });
excludeOpts.push({ p: '/fire/alarm/:id', o: 'PUT' });
+ excludeOpts.push({ p: '/fire/device', o: 'GET' });
+ excludeOpts.push({ p: '/fire/trend', o: 'GET' });
excludes = new ExcludesUrls(excludeOpts);
}
diff --git a/api/app/lib/routes/superScreen/fire.js b/api/app/lib/routes/superScreen/fire.js
index b134d95..dfa6c17 100644
--- a/api/app/lib/routes/superScreen/fire.js
+++ b/api/app/lib/routes/superScreen/fire.js
@@ -15,4 +15,12 @@ module.exports = function (app, router, opts, AuthCode) {
//修改消防告警状态
app.fs.api.logAttr['PUT/fire/alarm/:id'] = { content: '修改消防告警状态', visible: true };
router.put('/fire/alarm/:id', fire.updateAlarm(opts));
+
+ //获取消防设备
+ app.fs.api.logAttr['GET/fire/device'] = { content: '获取消防设备', visible: true };
+ router.get('/fire/device', fire.getFireDevice(opts));
+
+ //获取火情趋势
+ app.fs.api.logAttr['GET/fire/trend'] = { content: '获取火情趋势', visible: true };
+ router.get('/fire/trend', fire.getFireTrend(opts));
};
diff --git a/api/config.js b/api/config.js
index 155f150..8b4200d 100644
--- a/api/config.js
+++ b/api/config.js
@@ -15,6 +15,7 @@ args.option(['s', 'kubesphere'], 'kubesphere地址');
args.option(['d', 'dbconfig'], '后台同步数据库host示例:postgres/example/10.8.30.160/30432 格式:用户名/密码/host/port');
args.option(['w', 'water'], '水环境api地址');
args.option(['a', 'worksafety'], '安监api地址');
+args.option('tf', '消防第三方接口 THIRD_FIRECONTROL');
const flags = args.parse(process.argv);
@@ -33,7 +34,9 @@ const DATABASE_CONFIG = process.env.DATABASE_HOST || flags.dbconfig;//同步数
const WATER_URL = process.env.WATER_URL || flags.water;
const WORKSAFETY_URL = process.env.WORKSAFETY_URL || flags.worksafety;
-if (!DB || !BACKUPS_URL || !KUBESPHERE_URL || !DATABASE_CONFIG || !WATER_URL || !WORKSAFETY_URL) {
+const THIRD_FIRECONTROL = process.env.THIRD_FIRECONTROL || flags.tf;
+
+if (!DB || !BACKUPS_URL || !KUBESPHERE_URL || !DATABASE_CONFIG || !WATER_URL || !WORKSAFETY_URL || !THIRD_FIRECONTROL) {
console.log('缺少启动参数,异常退出');
args.showHelp();
process.exit(-1);
@@ -81,7 +84,7 @@ const product = {
backupsUrl: BACKUPS_URL,
k8s: KUBESPHERE_URL,
dbConfig: DATABASE_CONFIG,
-
+ tfApi: THIRD_FIRECONTROL,
}
}, {
entry: require('./app/lib/middlewares/proxy').entry,
diff --git a/super-screen/client/src/sections/fire-control/actions/fire.js b/super-screen/client/src/sections/fire-control/actions/fire.js
index 3b5b12c..4a823a0 100644
--- a/super-screen/client/src/sections/fire-control/actions/fire.js
+++ b/super-screen/client/src/sections/fire-control/actions/fire.js
@@ -40,3 +40,23 @@ export function modifyFireAlarm(id, params) {
},
});
}
+
+export function getFireDevice() {
+ return dispatch => basicAction({
+ type: 'get',
+ dispatch: dispatch,
+ actionType: 'GET_FIRE_DEVICE',
+ url: `${ApiTable.getFireDevice}`,
+ msg: { error: '获取消防设备失败' },
+ });
+}
+
+export function getFireTrend() {
+ return dispatch => basicAction({
+ type: 'get',
+ dispatch: dispatch,
+ actionType: 'GET_FIRE_TREND',
+ url: `${ApiTable.getFireTrend}`,
+ msg: { error: '获取火情趋势失败' },
+ });
+}
diff --git a/super-screen/client/src/sections/fire-control/components/left-bottom.js b/super-screen/client/src/sections/fire-control/components/left-bottom.js
index 37af56e..973d446 100644
--- a/super-screen/client/src/sections/fire-control/components/left-bottom.js
+++ b/super-screen/client/src/sections/fire-control/components/left-bottom.js
@@ -1,49 +1,118 @@
import React from 'react'
-import { Box, AutoRollComponent } from '$components';
+import { Box, NoData } from '$components';
+import ReactEcharts from 'echarts-for-react';
+import moment from 'moment';
function Infrastructure(props) {
+ const { fireTrend } = props;
+ let Ydata = fireTrend.map(t => t.count);
+ let Xdata = fireTrend.map(t => moment(t.time).format('MM月DD日'));
- const getContent = () => {
- return
- }
- return
-
+ const option = {
+ tooltip: {
+ trigger: "axis",
+ axisPointer: {
+ type: "shadow",
+ },
+ backgroundColor: "rgba(255,255,255,0.75)",
+ extraCssText: "box-shadow: 2px 2px 4px 0px rgba(0,0,0,0.3);",
+ textStyle: {
+ fontSize: 14,
+ color: "#000",
+ },
+ formatter: (params) => {
+ const item = params[0];
+ return item.name + " : " + item.value + " 次";
+ },
+ },
+ xAxis: [
+ {
+ type: "category",
+ boundaryGap: false,
+ axisLabel: {
+ interval: 0,
+ color: "rgba(195, 230, 255, 1)",
+ fontSize: 10,
+ },
+ axisLine: {
+ show: true,
+ lineStyle: {
+ type: 'solid',
+ color: "rgba(184, 185, 188, 0.5)",
+ width: 1,
+ },
+ },
+ data: Xdata,
+ },
+ ],
+ yAxis: [
+ {
+ splitNumber: 5, // 刻度段数
+ type: "value",
+ nameTextStyle: {
+ color: "rgba(195, 230, 255, 1)",
+ fontWeight: 400,
+ fontSize: 14,
+ padding: [-20, 20, 0, 0]
+ },
+ splitLine: {
+ show: true,
+ lineStyle: {
+ type: 'dashed',
+ color: 'rgba(89, 153, 200, 0.5)'
+ }
+ },
+ axisLabel: {
+ show: true,
+ fontSize: 12,
+ color: "rgba(195, 230, 255, 1)",
+ },
+ },
+ ],
+ series: [
+ {
+ type: "line",
+ symbol: "none",
+ barWidth: 16,
+ label: {
+ show: false,
+ position: "top",
+ color: "#00A8FF",
+ },
+ areaStyle: {//区域填充渐变颜色
+ color: {
+ type: 'linear',
+ x: 0,
+ y: 0,
+ x2: 0,
+ y2: 1,
+ colorStops: [{
+ offset: 0, color: '#4CA1FF40' // 0% 处的颜色
+ }, {
+ offset: 1, color: '#4CA1FF00' // 100% 处的颜色
+ }],
+ global: false // 缺省为 false
+ }
+ },
+ data: Ydata,
+ },
+ ],
+ grid: {
+ x: 45,
+ y: 25,
+ x2: 40,
+ y2: 20
+ },
+ };
+ return
+ {
+ fireTrend.length ? :
+ }
}
diff --git a/super-screen/client/src/sections/fire-control/components/left-middle.js b/super-screen/client/src/sections/fire-control/components/left-middle.js
index c03c914..2b757ae 100644
--- a/super-screen/client/src/sections/fire-control/components/left-middle.js
+++ b/super-screen/client/src/sections/fire-control/components/left-middle.js
@@ -1,141 +1,166 @@
import React from 'react'
-import { Box } from '$components';
-import ReactEcharts from 'echarts-for-react';
+import { Box, AutoRollComponent, NoData } from '$components';
+// import ReactEcharts from 'echarts-for-react';
+import { Progress } from 'antd';
-function PopulationDynamics() {
+function PopulationDynamics(props) {
+ const { fireDevice } = props;
+ const total = fireDevice.reduce((p, n) => p + n.device_count, 0);
- let Ydata = ['火灾扑救', '抢险救援', '公务执勤', '社会救助', '其他出动'];
- let Xdata = [12, 19, 19, 13, 15]
- const options = {
- xAxis: {
- type: 'value',
- show: false,
- },
- grid: {
- left: -10,
- top: 20,
- bottom: 0,
- right: 20,
- containLabel: true,
- },
- yAxis: [
- {
- type: 'category',
- inverse: true,
- axisLabel: {
- show: true,
- margin: 25,
- // textStyle: {
- color: '#ECF7FF',
- fontSize: 12,
- // },
- // 调整左侧文字的3个属性,缺一不可
- verticalAlign: 'center',
- align: 'left',
- //调整文字上右下左
- padding: [0, 0, 0, -30],
+ // let Ydata = fireDevice.map(d => d.type_name);
+ // let Xdata = fireDevice.map(d => d.device_count);
+ // const options = {
+ // xAxis: {
+ // type: 'value',
+ // show: false,
+ // },
+ // grid: {
+ // left: -10,
+ // top: 20,
+ // bottom: 0,
+ // right: 20,
+ // containLabel: true,
+ // },
+ // yAxis: [
+ // {
+ // type: 'category',
+ // inverse: true,
+ // axisLabel: {
+ // show: true,
+ // margin: 25,
+ // // textStyle: {
+ // color: '#ECF7FF',
+ // fontSize: 12,
+ // // },
+ // // 调整左侧文字的3个属性,缺一不可
+ // verticalAlign: 'center',
+ // align: 'left',
+ // //调整文字上右下左
+ // padding: [0, 0, 0, -30],
+ // },
+ // splitLine: {
+ // show: false,
+ // },
+ // axisTick: {
+ // show: false,
+ // },
+ // axisLine: {
+ // show: false,
+ // },
+ // data: Ydata
+ // },
+ // {
+ // inverse: true,
+ // // y轴最右侧的文字
+ // axisTick: "none",
+ // axisLine: "none",
+ // type: "category",
+ // axisLabel: {
+ // // margin: 10,
+ // // textStyle: {
+ // color: "#24DCF7",
+ // fontSize: "12",
+ // // },
+ // rich: {
+ // a1: {
+ // color: '#24DCF7',
+ // width: 5,
+ // height: 5,
+ // fontSize: 16,
+ // },
+ // a2: {
+ // color: '#5999C8',
+ // width: 5,
+ // height: 5,
+ // fontSize: 12,
+ // },
+ // },
+ // formatter: function (value) {
+ // return [`{a1|${value}} {a2|个}`];
+ // },
+ // },
+ // data: Xdata,
+ // },
+ // ],
+ // series: [
+ // {
+ // type: 'bar',
+ // barWidth: 12,
+ // zlevel: 2,
+ // z: 2,
+ // showBackground: true,
+ // backgroundStyle: {
+ // color: '#2B375C'
+ // },
+ // color: '#005AC6',
- },
- splitLine: {
- show: false,
- },
- axisTick: {
- show: false,
- },
- axisLine: {
- show: false,
- },
- data: Ydata
- },
- {
- inverse: true,
- // y轴最右侧的文字
- axisTick: "none",
- axisLine: "none",
- type: "category",
- axisLabel: {
- // margin: 10,
- // textStyle: {
- color: "#24DCF7",
- fontSize: "12",
- // },
- rich: {
+ // label: {
+ // show: false,
- a1: {
- color: '#24DCF7',
- width: 5,
- height: 5,
- fontSize: 16,
- },
- a2: {
- color: '#5999C8',
- width: 5,
- height: 5,
- fontSize: 12,
- },
- },
- formatter: function (value) {
- return [`{a1|${value}} {a2|次}`];
- },
- },
- data: Xdata,
- },
+ // },
+ // data: Xdata,
+ // },
+ // {
+ // type: "pictorialBar",
+ // // symbol: 'image://',
+ // symbolSize: [15, 15],
+ // symbolOffset: [0, 0],
+ // symbolPosition: "right",
+ // z: 20,
+ // zlevel: 20,
+ // itemStyle: {
+ // // normal: {
+ // color: "#fff"
+ // // }
+ // },
+ // data: (function () {
+ // var list = [];
+ // for (var i = 0; i < Xdata.length; i++) {
+ // list.push(2.02 * Xdata[i]);
+ // }
+ // console.log(list)
+ // return list;
+ // })()
+ // },
+ // ],
+ // };
+ const getContent = () => {
+ return {
+ fireDevice.map(d =>
+
{d.type_name}
+
+
{d.device_count}个
+
)
+ }
+ //
+ }
- ],
- series: [
- {
- type: 'bar',
- barWidth: 12,
- zlevel: 2,
- z: 2,
- showBackground: true,
- backgroundStyle: {
- color: '#2B375C'
- },
- color: '#005AC6',
-
- label: {
- show: false,
-
- },
- data: Xdata,
- },
- {
- type: "pictorialBar",
- // symbol: 'image://',
- symbolSize: [15, 15],
- symbolOffset: [0, 0],
- symbolPosition: "right",
- z: 20,
- zlevel: 20,
- itemStyle: {
- // normal: {
- color: "#fff"
- // }
- },
- data: (function () {
- var list = [];
- for (var i = 0; i < Xdata.length; i++) {
- list.push(2.02 * Xdata[i]);
- }
- console.log(list)
- return list;
- })()
- },
-
- ],
-
- };
-
- return
-
+ return
+ {
+ fireDevice.length
+ ?
+ :
+ }
}
diff --git a/super-screen/client/src/sections/fire-control/components/left-top.js b/super-screen/client/src/sections/fire-control/components/left-top.js
index 95250ff..bae8684 100644
--- a/super-screen/client/src/sections/fire-control/components/left-top.js
+++ b/super-screen/client/src/sections/fire-control/components/left-top.js
@@ -1,38 +1,30 @@
import React from 'react'
-import { Box } from '$components';
+import { Box, AutoRollComponent, NoData } from '$components';
-function BasicInfo() {
+function BasicInfo(props) {
+ const { emergencyList } = props;
- return
-
-
-
-
-
-
-
电话联系数量
-
479个
+ const getContent = () => {
+ return
+ {emergencyList.xfjy?.map(item =>
+
-
-
-
-
-
误报数量
-
276个
+
+ {item.teamName}
+ {item.leaderContactPhone}
-
+
)}
+ }
+ return
+ {emergencyList?.xfjy?.length
+ ?
+ :
+ }
}
diff --git a/super-screen/client/src/sections/fire-control/components/style.less b/super-screen/client/src/sections/fire-control/components/style.less
index 862ad98..55681ed 100644
--- a/super-screen/client/src/sections/fire-control/components/style.less
+++ b/super-screen/client/src/sections/fire-control/components/style.less
@@ -246,94 +246,94 @@
}
-//警情数据分析
-.alarm_data_analysis {
- width: 100%;
- height: 100%;
- display: flex;
- justify-content: space-around;
- align-items: center;
+//消防救援队伍
+.xfjy_data {
+ padding-left: 28px;
+ padding-right: 28px;
- ._item_left {
- width: 209px;
- height: 100%;
+ .xfjy_item {
+ display: flex;
+ background: url('/assets/images/homepage/fire/realtimebg.png') no-repeat;
+ background-size: 100% 100%;
+ height: 110px;
+ width: 100%;
+ margin-bottom: 5%;
+ align-items: center;
+ justify-content: center;
- ._img {
- width: 209px;
- height: 209px;
- background: url('/assets/images/homepage/fire/realalarmbg.png') no-repeat;
- background-size: 100% 100%;
+ ._text {
+ width: calc(100% - 40px);
display: flex;
- align-items: center;
- justify-content: center;
- font-family: D-DINExp-Italic;
- font-weight: Italic;
- font-size: 48px;
- color: #FFFFFF;
- letter-spacing: 0;
- }
+ flex-direction: column;
- ._text {
- font-family: YouSheBiaoTiHei;
- font-size: 20px;
- color: #FFFFFF;
- letter-spacing: 0;
- margin-top: -30px;
- text-align: center;
+ ._row1 {
+ color: #C3E6FF;
+ width: 100%;
+ height: 22px;
+ background: rgba(0, 88, 204, 0.5);
+ justify-content: space-around;
+ display: flex;
+ }
+
+ ._row2 {
+ justify-content: space-around;
+ display: flex;
+ height: 42px;
+ font-family: SourceHanSansSC-Regular;
+ font-weight: 400;
+ font-size: 16px;
+ color: #ECF7FF;
+ letter-spacing: 0.67px;
+ padding-top: 4px;
+ }
}
}
+}
- ._item_right {
- width: 200px;
+// 接入消防设备
+.fire-device-item {
+ display: flex;
+ height: 30px;
+ align-items: center;
+ justify-content: space-around;
- ._iconphone {
- width: 50.96px;
- height: 45px;
- background: url('/assets/images/homepage/fire/iconphone.png') no-repeat;
- background-size: 100% 100%;
- }
+ ._name {
+ width: 80px;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ font-family: SourceHanSansSC-Regular;
+ font-weight: 400;
+ font-size: 14px;
+ color: #ECF7FF;
+ letter-spacing: 0;
+ text-align: right;
+ }
- ._iconreal {
- width: 50.96px;
- height: 45px;
- background: url('/assets/images/homepage/fire/iconreal.png') no-repeat;
- background-size: 100% 100%;
- }
+ ._progress {
+ position: relative;
+ width: 200px;
- ._iconmistake {
- width: 50.96px;
- height: 45px;
- background: url('/assets/images/homepage/fire/iconmistake.png') no-repeat;
- background-size: 100% 100%;
+ ._round {
+ position: absolute;
+ width: 12px;
+ height: 12px;
+ background: #fff;
+ border-radius: 6px;
+ top: 7px;
}
+ }
- .right_item_right {
- width: 155px;
- margin-left: 16px;
-
- .alarm_title {
- font-size: 14px;
- color: #ECF7FF;
- letter-spacing: 0;
- }
-
- .alarm_number {
- font-family: D-DINExp-Italic;
- font-weight: Italic;
- font-size: 28px;
- color: #24DCF7;
- letter-spacing: 0;
- margin-right: 50px;
- }
+ ._count {
+ width: 75px;
+ font-family: D-DINExp-Italic;
+ font-weight: Italic;
+ font-size: 18px;
+ color: #24DCF7;
- .alarm_unit {
- font-size: 14px;
- color: #5999C8;
- letter-spacing: 0;
- }
+ ._unit {
+ font-size: 14px;
}
-
-
}
}
diff --git a/super-screen/client/src/sections/fire-control/containers/homePage.js b/super-screen/client/src/sections/fire-control/containers/homePage.js
index cc83341..9e88019 100644
--- a/super-screen/client/src/sections/fire-control/containers/homePage.js
+++ b/super-screen/client/src/sections/fire-control/containers/homePage.js
@@ -23,6 +23,8 @@ function homePage(props) {
const [tab, setTab] = useState('overview')
const [emengencyTab, setEmengencyTab] = useState('xfyjwz');
const [alarmInfo, setAlarmInfo] = useState({})
+ const [fireDevice, setFireDevice] = useState([])
+ const [fireTrend, setFireTrend] = useState([])
const { data: emergencyList = {} } = useFsRequest({ url: 'water/emergency' });
const endEvent = () => {
@@ -33,6 +35,24 @@ function homePage(props) {
})
}
+ useEffect(() => {
+ getFireData();
+ }, [])
+
+ const getFireData = () => {
+ dispatch(actions.firecontrol?.getFireDevice()).then(res => {
+ if (res?.payload?.data?.length) {
+ const filterData = res.payload.data.filter(d => d.device_count)
+ setFireDevice(filterData)
+ }
+ })
+ dispatch(actions.firecontrol?.getFireTrend()).then(res => {
+ if (res?.payload?.data) {
+ setFireTrend(res.payload.data)
+ }
+ })
+ }
+
return <>
@@ -50,13 +70,13 @@ function homePage(props) {
{tab == 'overview' ?
<>
-
+
-
+
-
+
>
:
diff --git a/super-screen/client/src/sections/water-prevention/components/left-bottom.js b/super-screen/client/src/sections/water-prevention/components/left-bottom.js
index b236c20..4f6561c 100644
--- a/super-screen/client/src/sections/water-prevention/components/left-bottom.js
+++ b/super-screen/client/src/sections/water-prevention/components/left-bottom.js
@@ -33,9 +33,9 @@ function LeftBottom(props) {
];
const data = [
- { name: '泵站站点', value: 12 },
+ { name: '泵站站点', value: SHUI_ZHAN.length },
{ name: '雨量站点', value: 1 },
- { name: '水位站点', value: 23 }
+ { name: '水位站点', value: waterLevelLength }
]
return
diff --git a/super-screen/client/src/utils/webapi.js b/super-screen/client/src/utils/webapi.js
index 61a1281..c5f0123 100644
--- a/super-screen/client/src/utils/webapi.js
+++ b/super-screen/client/src/utils/webapi.js
@@ -134,6 +134,8 @@ export const ApiTable = {
//消防
getFireAlarmList: 'fire/alarm',
modifyFireAlarm: 'fire/alarm/{id}',
+ getFireDevice: 'fire/device',
+ getFireTrend: 'fire/trend',
};
export const RouteTable = {