山东移动监控应用=>标准版智能安防

This commit is contained in:
aixianling
2022-04-12 16:01:21 +08:00
parent 427f890aa1
commit f4f7963fff
7 changed files with 16 additions and 13 deletions

View File

@@ -0,0 +1,308 @@
<template>
<section class="deviceSlider">
<div class="mainPane" v-if="show">
<div flex overview>
<b>监控设备</b>
<div>
<div>设备总数{{ overview.total }}</div>
<div flex>在线设备<p v-text="overview.online"/></div>
</div>
<el-progress type="circle" :width="40" :percentage="overview.percent" color="#19D286" :stroke-width="4"/>
</div>
<div flex search>
<el-select v-model="search.bind" size="mini" placeholder="全部" clearable @change="onChange">
<el-option v-for="(op,i) in dict.getDict('deviceStatus')" :key="i" :value="op.dictValue"
:label="op.dictName"/>
</el-select>
<el-input
v-model="search.name"
size="mini"
placeholder="设备名称"
v-throttle="handleTreeFilter"
prefix-icon="el-icon-search"
@clear="search.name = '', handleTreeFilter()" clearable/>
</div>
<div title>设备列表</div>
<div fill class="deviceList">
<el-tree ref="deviceTree" :render-content="renderItem" :data="treeData" :props="propsConfig"
@node-click="handleNodeClick" @node-contextmenu="nodeContextmenu"
:filter-node-method="handleFilter"/>
<ul
v-if="isShowMenu && menuInfo.node.type==1 && permissions('video_config')"
class="el-dropdown-menu el-popper"
:style="{top: menuInfo.y + 'px', left: menuInfo.x + 'px', position: 'fixed', zIndex: 2023}"
x-placement="top-end">
<li class="el-dropdown-menu__item" @click="handleTreeCommand('edit', menuInfo.node)">修改名称</li>
<!-- <li class="el-dropdown-menu__item" @click="handleTreeCommand('area', menuInfo.node)">行政地区</li> -->
<li class="el-dropdown-menu__item" @click="handleTreeCommand('locate', menuInfo.node)">地图标绘</li>
</ul>
</div>
</div>
<div class="rightBtn" :class="{show}" @click="handleShow">
<i class="iconfont iconArrow_Right"/>
</div>
</section>
</template>
<script>
export default {
name: "deviceSlider",
props: {
show: Boolean,
ins: Function,
dict: Object,
permissions: Function,
renderItem: Function
},
computed: {
overview() {
let total = this.list?.length || 0,
online = this.list?.filter(e => e.deviceStatus == 1)?.length || 0
return {
total, online,
percent: Math.ceil(online / total * 100) || 0
}
},
propsConfig() {
return {
label: 'name',
children: 'children'
}
},
treeData() {
let {list, noArea, staData} = this
let meta = [staData?.reduce((t, e) => {
return t.type <= e.type ? t : e
}, {name: '读取中...'})]
meta.map(p => this.addChild(p, [...staData, ...list].map(s => ({
...s,
parentId: s.areaId || s.parent_id
}))))
return [...meta, {
id: 'no_area',
name: '未知区划',
children: noArea
}]
}
},
data() {
return {
list: [],
noArea: [],
staData: [],
name: '',
isShowMenu: false,
search: {
bind: ''
},
menuInfo: {
x: '',
y: '',
node: {}
}
}
},
methods: {
handleShow() {
this.$emit('update:show', !this.show)
},
bindEvent() {
this.isShowMenu = false
},
getDevices() {
this.ins.post("/app/appzyvideoequipment/tree", null, {
params: {size: 999}
}).then(res => {
if (res?.data) {
this.staData = res.data.count
this.list = res.data.list
this.noArea = res.data.noArea
this.$emit('list', this.list)
}
})
},
handleTreeCommand(e, node) {
this.$emit('treeCommand', {
type: e,
node
})
},
nodeContextmenu(e, node) {
this.isShowMenu = true
let y = e.y + 6
if (y + 202 > document.body.clientHeight) {
y = y - 202
}
this.menuInfo = {
x: e.x + 16, y,
node
}
},
handleNodeClick(data) {
this.isShowMenu = false
this.$emit('select', data)
},
handleFilter(v, data) {
if (!v) {
return !this.search.bind ? true : data.deviceStatus === this.search.bind
}
return data?.name?.indexOf(v) > -1 && (!this.search.bind ? true : data.deviceStatus === this.search.bind)
},
handleTreeFilter() {
this.$refs.deviceTree?.filter(this.search.name)
},
onChange() {
this.$refs.deviceTree?.filter(this.search.name)
}
},
created() {
this.dict.load("deviceStatus")
this.getDevices()
},
mounted() {
document.querySelector('html').addEventListener('click', this.bindEvent)
}
}
</script>
<style lang="scss" scoped>
.deviceSlider {
display: flex;
align-items: center;
flex-shrink: 0;
color: #fff;
overflow: hidden;
div[flex] {
display: flex;
align-items: center;
}
.deviceList {
overflow: auto;
::v-deep .el-tree {
width: -webkit-fit-content;
width: -moz-fit-content;
width: fit-content;
min-width: 100%;
}
&::-webkit-scrollbar {
width: 10px;
height: 15px;
}
&::-webkit-scrollbar-thumb {
box-shadow: inset 0 0 3px rgba(0, 0, 0, 0.2);
background: #535353;
}
&::-webkit-scrollbar-track {
box-shadow: inset 0 0 3px rgba(0, 0, 0, 0.2);
background: #fff;
}
}
div[fill] {
flex: 1;
min-width: 0;
min-height: 0;
}
.mainPane {
width: 280px;
height: 100%;
background: #333C53;
display: flex;
flex-direction: column;
padding-top: 16px;
overflow: hidden;
box-sizing: border-box;
b {
font-size: 18px;
}
div[overview], div[search] {
box-sizing: border-box;
font-size: 12px;
justify-content: space-between;
padding: 0 16px;
gap: 4px;
margin-bottom: 16px;
::v-deep.el-input__inner {
color: #fff;
}
}
div[title] {
height: 28px;
background: #3E4A69;
padding: 0 16px;
line-height: 28px;
}
::v-deep.deviceList {
padding: 0 8px;
.el-scrollbar {
height: 100%;
.el-scrollbar__wrap {
box-sizing: content-box;
padding-bottom: 17px;
}
}
}
::v-deep .el-progress__text, p {
color: #19D286;
}
::v-deep .el-input__inner {
background: #282F45;
border: none;
}
::v-deep .el-tree {
background: transparent;
color: #fff;
.el-tree-node:focus > .el-tree-node__content, .el-tree-node__content:hover {
background: rgba(#fff, .1);
}
}
::v-deep .el-input__icon {
color: #89b;
}
}
.rightBtn {
width: 16px;
height: 80px;
background: url("https://cdn.cunwuyun.cn/monitor/drawerBtn.png");
color: #fff;
display: flex;
align-items: center;
justify-content: center;
.iconfont {
transition: transform 0.2s;
}
&.show > .iconfont {
transform: rotate(180deg);
}
}
}
</style>

View File

@@ -0,0 +1,177 @@
<template>
<section class="locateDialog">
<ai-dialog :visible.sync="dialog" title="标绘" @closed="$emit('visible',false),selected={}"
@opened="$nextTick(()=>initMap())"
@onConfirm="handleConfirm">
<ai-t-map :map.sync="map" :lib.sync="TMap"/>
<div class="poi">
<el-autocomplete ref="poiInput" v-model="search" size="small" clearable :fetch-suggestions="handleSearch"
placeholder="请输入地点" @select="handleSelect" :trigger-on-focus="false">
<template slot-scope="{item}">
<span style="direction: rtl" v-text="`${item.title}(${item.address})`"/>
</template>
</el-autocomplete>
</div>
<el-form class="selected" v-if="!!selected.location" id="result" size="mini" label-suffix=""
label-position="left">
<div class="header">
<i class="iconfont iconLocation"/>
<span v-html="[selected.location.lng, selected.location.lat].join(',')"/>
</div>
<el-form-item label="地点">{{ selected.name || "未知地名" }}</el-form-item>
<el-form-item label="类型" v-if="!!selected.type">{{ selected.type }}</el-form-item>
<el-form-item label="地址" v-if="!!selected.address">{{ selected.address }}</el-form-item>
</el-form>
</ai-dialog>
</section>
</template>
<script>
import {mapState} from "vuex";
export default {
name: "locateDialog",
model: {
prop: "visible",
event: "visible",
},
props: ['latlng', 'visible'],
data() {
return {
dialog: false,
search: "",
map: null,
selected: {},
TMap: null
}
},
computed: {
...mapState(['user'])
},
watch: {
visible(v) {
this.dialog = v
}
},
methods: {
initMap(count = 0) {
let {map, TMap} = this
if (map) {
if (!!this.latlng?.lat) {
let position = new TMap.LatLng(this.latlng.lat, this.latlng.lng)
map.setCenter(position)
this.selected.marker = new TMap.MultiMarker({map, geometries: [{position}]})
}
map.on('click', res => {
let {poi, latLng: location} = res, name = poi?.name || ""
this.selected.marker?.setMap(null)
this.selected = {location, name}
this.selected.marker = new TMap.MultiMarker({map, geometries: [{position: location}]})
})
} else {
if (count < 5) {
count++
setTimeout(() => this.initMap(count), 500)
} else {
console.error("地图渲染失败")
}
}
},
handleSearch(keyword, cb) {
let {TMap} = this
if (keyword && TMap) {
let poi = new TMap.service.Search({pageSize: 10})
poi.searchRegion({
keyword, radius: 5000, cityName: this.user.info?.areaId?.substring(0, 6) || ""
}).then(res => {
if (res?.data?.length > 0) {
cb(res.data)
} else this.$message.error("未查到有效地点")
})
}
},
handleConfirm() {
if (this.selected?.location) {
this.$emit('confirm', this.selected)
} else {
this.$message.error('请先选择坐标位置')
}
},
handleSelect(res) {
let {map, TMap} = this
if (map) {
let {title: name, location} = res
this.selected.marker?.setMap(null)
this.selected = {location, name}
this.selected.marker = new TMap.MultiMarker({map, geometries: [{position: location}]})
map.setCenter(location)
}
}
},
created() {
this.dialog = this.visible
}
}
</script>
<style lang="scss" scoped>
.locateDialog {
.color-999 {
color: #999;
}
::v-deep .el-dialog__body {
padding: 0;
height: 480px;
position: relative;
.ai-dialog__content--wrapper {
padding: 0 !important;
}
.poi {
position: absolute;
left: 10px;
top: 10px;
display: flex;
height: 32px;
flex-direction: column;
z-index: 202203281016;
width: 400px;
div {
flex-shrink: 0;
}
}
.selected {
position: absolute;
right: 16px;
top: 16px;
background: #fff;
min-width: 200px;
box-sizing: border-box;
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
.header {
color: #fff;
background: #26f;
text-align: center;
display: flex;
align-items: center;
height: 32px;
font-size: 14px;
gap: 4px;
padding: 0 8px;
}
.el-form-item {
padding: 0 8px;
margin: 0;
}
}
}
}
</style>

View File

@@ -0,0 +1,91 @@
<template>
<section class="settingDialog">
<ai-dialog :visible.sync="dialog" title="基础设置" @close="$emit('visible',false)">
<el-form ref="deviceForm" size="small" label-width="140px">
<el-form-item label="设备名称" class="full">
<el-input v-model="form.name" clearable placeholder="设备名称"/>
</el-form-item>
<el-form-item label="摄像头状态">
<el-radio v-model="form.status" label="开启"/>
<el-radio v-model="form.status" label="关闭"/>
</el-form-item>
<el-form-item label="高清视频">
<el-radio v-model="form.status" label="开启"/>
<el-radio v-model="form.status" label="关闭"/>
</el-form-item>
<el-form-item label="摄像头麦克风">
<el-radio v-model="form.status" label="开启"/>
<el-radio v-model="form.status" label="关闭"/>
</el-form-item>
<el-form-item label="状态指示灯">
<el-radio v-model="form.status" label="开启"/>
<el-radio v-model="form.status" label="关闭"/>
</el-form-item>
<el-form-item label="夜视">
<el-radio v-model="form.status" label="自动"/>
<el-radio v-model="form.status" label="开启"/>
<el-radio v-model="form.status" label="关闭"/>
</el-form-item>
<el-form-item label="旋转180°">
<el-radio v-model="form.status" label="开启"/>
<el-radio v-model="form.status" label="关闭"/>
</el-form-item>
<el-form-item label="WIFI网络" class="full">-</el-form-item>
<el-form-item label="MAC地址">(34:75:6b:c9:10)</el-form-item>
<el-form-item label="摄像头型号">C71</el-form-item>
<el-form-item label="固件">20.0326.251.2486</el-form-item>
<el-form-item label="嵌入式应用">2.3.37.8954</el-form-item>
<el-form-item label="IMEI">110003953100302</el-form-item>
</el-form>
</ai-dialog>
</section>
</template>
<script>
export default {
name: "settingDialog",
model: {
prop: "visible",
event: "visible",
},
props: {
visible: Boolean,
detail: {default: () => ({})}
},
data() {
return {
dialog: false,
form: {}
}
},
watch: {
visible(v) {
this.dialog = v
}
},
created() {
this.form = JSON.parse(JSON.stringify(this.form))
}
}
</script>
<style lang="scss" scoped>
.settingDialog {
.el-form {
display: flex;
flex-wrap: wrap;
}
::v-deep .el-form-item {
width: 50%;
.el-form-item__label {
padding-right: 40px;
}
&.full {
width: 100%;
}
}
}
</style>