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
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
-
14:22
-
红花岗区大队
-
划龙桥路77号
-
-
- } - 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 -
-
-
65%
-
真警占比率
-
-
-
-
-
-
电话联系数量
-
479
+ const getContent = () => { + return
+ {emergencyList.xfjy?.map(item =>
+
+
+ 救援队伍 + 负责人及电话
-
-
-
-
-
真警数量
-
273
-
-
-
-
-
-
误报数量
-
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 = {