You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1463 lines
42 KiB
Vue
1463 lines
42 KiB
Vue
<template>
|
|
<div class="app-container">
|
|
<div class="top">
|
|
<div class="title">探测系统</div>
|
|
|
|
<div class="items">
|
|
<div
|
|
class="item"
|
|
:class="{ click: itemsId === 1 }"
|
|
@click="setItemsId(1)"
|
|
>
|
|
<span>主界面</span>
|
|
</div>
|
|
|
|
<div
|
|
class="item"
|
|
:class="{ click: itemsId === 2 }"
|
|
@click="setItemsId(2)"
|
|
>
|
|
<span>区域标定</span>
|
|
</div>
|
|
|
|
<div
|
|
class="item"
|
|
:class="{ click: itemsId === 3 }"
|
|
@click="setItemsId(3)"
|
|
>
|
|
<span>软件设置</span>
|
|
</div>
|
|
|
|
<div
|
|
class="item"
|
|
:class="{ click: itemsId === 4 }"
|
|
@click="setItemsId(4)"
|
|
>
|
|
<span>历史记录</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="left">
|
|
<Ruler
|
|
ref="rulerRef"
|
|
:width="areaData.width"
|
|
:height="areaData.height"
|
|
:boxPos="boxPos"
|
|
:dots="dots"
|
|
:dotIndex="dotIndex"
|
|
/>
|
|
</div>
|
|
|
|
<div class="right">
|
|
<div v-if="itemsId === 1" class="tabsItem">
|
|
<!-- 区域选择 -->
|
|
<el-card shadow="always">
|
|
<el-form inline :model="form1" class="demo-form-inline">
|
|
<el-form-item>
|
|
<el-select
|
|
v-model="form1.region"
|
|
placeholder="请选择机场名称"
|
|
style="width: 200px"
|
|
clearable
|
|
@change="selectUpdate"
|
|
>
|
|
<el-option
|
|
v-for="i in options1"
|
|
:key="i.id"
|
|
:label="i.name"
|
|
:value="i.id"
|
|
/>
|
|
</el-select>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-select
|
|
v-model="form1.region1"
|
|
placeholder="请选择区域名称"
|
|
style="width: 200px"
|
|
clearable
|
|
>
|
|
<el-option
|
|
v-for="i in options2"
|
|
:key="i.id"
|
|
:label="i.name"
|
|
:value="i.id"
|
|
/>
|
|
</el-select>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-button type="primary" @click="getAreaData">获取区域</el-button>
|
|
</el-form-item>
|
|
</el-form>
|
|
</el-card>
|
|
|
|
<!-- 雷达启动控制 -->
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<el-button
|
|
type="primary"
|
|
:disabled="radarWorkState === 1"
|
|
@click="StartWorkBtn"
|
|
>
|
|
启动雷达
|
|
</el-button>
|
|
<el-button
|
|
type="danger"
|
|
:disabled="radarWorkState === 0"
|
|
@click="StopWorkBtn"
|
|
>
|
|
关闭雷达
|
|
</el-button>
|
|
<el-button type="warning" @click="RestartWorkBtn">重启雷达</el-button>
|
|
<el-button type="info" @click="ShutDownWorkBtn">关闭雷达</el-button>
|
|
</el-card>
|
|
|
|
<!-- 通讯状态 -->
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<div
|
|
v-if="radarWorkState === 1"
|
|
style="width: calc(50% - 60px); position: relative; border-radius: 5px; background-color: #01CE69; height: 44px; display: inline-block; padding: 0 20px; margin-right: 20px;"
|
|
>
|
|
<span
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); left: 20%; color: #fff; letter-spacing: 2px;"
|
|
>雷达通讯状态</span
|
|
>
|
|
<el-icon
|
|
size="20px"
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); right: 40px;"
|
|
>
|
|
<success-filled color="#fff"/>
|
|
</el-icon>
|
|
</div>
|
|
|
|
<div
|
|
v-if="radarWorkState === 0"
|
|
style="width: calc(50% - 60px); position: relative; border-radius: 5px; background-color: #E8370D; height: 44px; display: inline-block; padding: 0 20px; margin-right: 20px;"
|
|
>
|
|
<span
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); left: 20%; color: #fff; letter-spacing: 2px;"
|
|
>雷达通讯状态</span
|
|
>
|
|
<el-icon
|
|
size="20px"
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); right: 40px;"
|
|
>
|
|
<warning-filled color="#fff"/>
|
|
</el-icon>
|
|
</div>
|
|
|
|
<div
|
|
v-if="true"
|
|
style="width: calc(50% - 60px); position: relative; border-radius: 5px; background-color: #01CE69; height: 44px; display: inline-block; padding: 0 20px; margin-right: 20px;"
|
|
>
|
|
<span
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); left: 20%; color: #fff; letter-spacing: 2px;"
|
|
>雷达扫描状态</span
|
|
>
|
|
<el-icon
|
|
size="20px"
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); right: 40px;"
|
|
>
|
|
<success-filled color="#fff"/>
|
|
</el-icon>
|
|
</div>
|
|
|
|
<div
|
|
v-if="false"
|
|
style="width: calc(50% - 60px); position: relative; border-radius: 5px; background-color: #E8370D; height: 44px; display: inline-block; padding: 0 20px; margin-right: 20px;"
|
|
>
|
|
<span
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); left: 20%; color: #fff; letter-spacing: 2px;"
|
|
>雷达扫描状态</span
|
|
>
|
|
<el-icon
|
|
size="20px"
|
|
style="position: absolute; top: 50%; transform: translateY(-50%); right: 40px;"
|
|
>
|
|
<warning-filled color="#fff"/>
|
|
</el-icon>
|
|
</div>
|
|
|
|
</el-card>
|
|
|
|
<!-- 目标表格1 -->
|
|
<el-card shadow="always" style="margin-top: 12px;position:relative;">
|
|
<div style="position: absolute;right: 30px;top: 20px;z-index: 100;">
|
|
<el-popover placement="bottom" :width="300" trigger="click">
|
|
<template #reference>
|
|
<el-button style="margin-right: 16px">批量处理</el-button>
|
|
</template>
|
|
<el-button style="margin-right: 16px" @click="batchDisposal">处置</el-button>
|
|
<el-button style="margin-right: 16px" @click="batchDel">删除</el-button>
|
|
<el-button style="margin-right: 16px" @click="batchShild">固定物</el-button>
|
|
</el-popover>
|
|
</div>
|
|
<el-table
|
|
ref="table1Ref"
|
|
highlight-current-row
|
|
:data="tableData1.slice((currentPage1 - 1) * pageSize1, currentPage1 * pageSize1)"
|
|
style="width: 100%"
|
|
@row-click="table1Current"
|
|
>
|
|
<el-table-column type="selection" width="55"/>
|
|
<el-table-column label="序号" type="index" width="60"/>
|
|
<el-table-column prop="Lon" label="目标经度" width="100"/>
|
|
<el-table-column prop="Lat" label="目标纬度" width="100"/>
|
|
<el-table-column prop="Strength" label="强度信息" width="100"/>
|
|
<el-table-column label="处理方式" show-overflow-tooltip width="200" fixed="right">
|
|
<template #default="scope">
|
|
<el-button type="primary" @click.stop="openHandleObstacleDialog(scope)" link>处置</el-button>
|
|
<el-popconfirm
|
|
class="box-item"
|
|
title="确定删除这个点位吗?"
|
|
placement="top"
|
|
@confirm="delPoint(scope)"
|
|
>
|
|
<template #reference>
|
|
<el-button type="primary" @click.stop link>删除</el-button>
|
|
</template>
|
|
</el-popconfirm>
|
|
<el-button type="primary" @click.stop="openShildDialog(scope)" link>固定物</el-button>
|
|
</template>
|
|
</el-table-column>
|
|
</el-table>
|
|
<el-pagination
|
|
v-model:current-page="currentPage1"
|
|
v-model:page-size="pageSize1"
|
|
:pager-count="4"
|
|
:page-sizes="[10,20,50,100, 200, 300, 400]"
|
|
layout="total, sizes, prev, pager, next, jumper"
|
|
:total="tableData1.length"
|
|
/>
|
|
</el-card>
|
|
|
|
<!-- 目标表格2 -->
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<el-table
|
|
highlight-current-row
|
|
:data="tableData2"
|
|
style="width: 100%"
|
|
>
|
|
<!-- <el-table-column type="selection" width="55"/>-->
|
|
<el-table-column label="序号" type="index" width="60"/>
|
|
<el-table-column prop="name" label="名称" width="100"/>
|
|
<el-table-column prop="lon" label="目标经度" width="100"/>
|
|
<el-table-column prop="lat" label="目标纬度" width="100"/>
|
|
<el-table-column prop="strength" label="强度信息" width="100"/>
|
|
<el-table-column label="处理方式" show-overflow-tooltip width="160" fixed="right">
|
|
<template #default="scope">
|
|
<el-button type="primary" link @click.stop="openShildDialog1(scope)">修改</el-button>
|
|
<el-popconfirm
|
|
class="box-item"
|
|
title="确定删除这个点位吗?"
|
|
placement="top"
|
|
@confirm="delShildItem(scope)"
|
|
>
|
|
<template #reference>
|
|
<el-button type="primary" @click.stop link>删除</el-button>
|
|
</template>
|
|
</el-popconfirm>
|
|
</template>
|
|
</el-table-column>
|
|
</el-table>
|
|
<el-pagination
|
|
v-model:current-page="currentPage2"
|
|
v-model:page-size="pageSize2"
|
|
:pager-count="4"
|
|
:page-sizes="[10,20,50,100, 200, 300, 400]"
|
|
layout="total, sizes, prev, pager, next, jumper"
|
|
:total="total2"
|
|
@change="getShildTableList"
|
|
/>
|
|
</el-card>
|
|
</div>
|
|
<div v-if="itemsId === 2" class="tabsItem">
|
|
<el-card shadow="always">
|
|
<el-form inline :model="form5" class="demo-form-inline">
|
|
<el-form-item>
|
|
<el-input
|
|
v-model="form5.region"
|
|
placeholder="请输入机场名称"
|
|
style="width: 200px"
|
|
/>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-input
|
|
v-model="form5.region1"
|
|
placeholder="请输入区域名称"
|
|
style="width: 200px"
|
|
/>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-button type="primary" @click="onSubmit">添加</el-button>
|
|
</el-form-item>
|
|
</el-form>
|
|
|
|
<el-form
|
|
inline
|
|
:model="form6"
|
|
class="demo-form-inline"
|
|
style="margin-top: 12px"
|
|
>
|
|
<el-form-item>
|
|
<el-select
|
|
v-model="form6.region"
|
|
placeholder="请选择机场名称"
|
|
style="width: 200px"
|
|
clearable
|
|
@change="select2Update"
|
|
>
|
|
<el-option
|
|
v-for="i in options1"
|
|
:key="i.id"
|
|
:label="i.name"
|
|
:value="i.id"
|
|
/>
|
|
</el-select>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-select
|
|
v-model="form6.region1"
|
|
placeholder="请选择区域名称"
|
|
style="width: 200px"
|
|
clearable
|
|
@change="select3Update"
|
|
>
|
|
<el-option
|
|
v-for="i in options2"
|
|
:key="i.id"
|
|
:label="i.name"
|
|
:value="i.id"
|
|
/>
|
|
</el-select>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-button type="primary" @click="getArea2">确定</el-button>
|
|
</el-form-item>
|
|
</el-form>
|
|
</el-card>
|
|
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<div
|
|
v-for="(i, k) in regionLabeling"
|
|
:key="k"
|
|
style="margin-bottom: 12px"
|
|
>
|
|
<el-input
|
|
disabled
|
|
v-model="i.name"
|
|
placeholder="点位名称"
|
|
style="width: 100px"
|
|
/>
|
|
<el-input
|
|
v-model="i.lon"
|
|
placeholder="经度"
|
|
style="width: 150px; margin-left: 20px"
|
|
/>
|
|
<el-input
|
|
v-model="i.lat"
|
|
placeholder="纬度"
|
|
style="width: 150px; margin-left: 20px"
|
|
/>
|
|
<el-button
|
|
type="primary"
|
|
@click="getPoint(k)"
|
|
style="margin-left: 20px"
|
|
>
|
|
获取
|
|
</el-button>
|
|
<el-button
|
|
type="danger"
|
|
@click="DeleteCData({ id: parseFloat(i.id) }).then(() => getArea2())"
|
|
style="margin-left: 20px"
|
|
>
|
|
删除
|
|
</el-button>
|
|
</div>
|
|
|
|
<el-button
|
|
type="primary"
|
|
style="text-align: center"
|
|
@click=" InsertCListData([{
|
|
name: `点位${regionLabeling.length + 1}`,
|
|
lon: '1',
|
|
lat: '1',
|
|
fodAreaId: form6.region1,
|
|
fodAirId: form6.region
|
|
}]).then(() => {
|
|
getArea2()
|
|
}) "
|
|
>
|
|
添加
|
|
</el-button>
|
|
</el-card>
|
|
|
|
<div style="text-align: center; margin-top: 12px">
|
|
<el-button
|
|
type="primary"
|
|
style="margin-left: 20px"
|
|
@click="saveAreaPoint"
|
|
>
|
|
提交
|
|
</el-button>
|
|
</div>
|
|
</div>
|
|
<div v-if="itemsId === 3" class="tabsItem">
|
|
<!-- 成像设置 -->
|
|
<div>成像设置</div>
|
|
<el-form :model="parasSignalproForm" inline label-width="auto" label-position="top">
|
|
<el-form-item
|
|
v-for='(key, idx) in ["imaging_rangemin", "imaging_rangemax", "imaging_rangeres", "imaging_azimuthlength", "imaging_azimuthres"]'
|
|
:key="idx"
|
|
:label="key"
|
|
style="width: calc(33% - 32px)"
|
|
>
|
|
<el-input
|
|
v-model="parasSignalproForm.imaging[key]"
|
|
/>
|
|
</el-form-item>
|
|
</el-form>
|
|
|
|
<!-- 检测设置 -->
|
|
<div>检测设置</div>
|
|
<el-form :model="parasSignalproForm" inline label-width="auto" label-position="top">
|
|
<el-form-item
|
|
v-for='(key, idx) in ["alpha", "decstartr", "decwidth", "sk_r", "sk_a", "lk_r", "lk_a"]'
|
|
:key="idx"
|
|
:label="key"
|
|
style="width: calc(33% - 32px)"
|
|
>
|
|
<el-input
|
|
v-model="parasSignalproForm.dectection[key]"
|
|
/>
|
|
</el-form-item>
|
|
|
|
<el-form-item style="width: 100%">
|
|
<div style="text-align: center; width: 100%">
|
|
<el-button type="primary" @click="saveParasSignalproForm">保存</el-button>
|
|
</div>
|
|
</el-form-item>
|
|
</el-form>
|
|
|
|
<!-- 路线设置 -->
|
|
<div>路线设置</div>
|
|
<el-form :model="parasPosForm" inline label-width="auto" label-position="top">
|
|
<el-form-item style="width: calc(33% - 32px)" label="路线数量">
|
|
<el-input-number
|
|
v-model="parasPosForm.runwaynum.startpos_num"
|
|
:precision="0"
|
|
:step="1"
|
|
:min="0"
|
|
@change="updateRunwayNum"
|
|
/>
|
|
</el-form-item>
|
|
|
|
<div
|
|
v-for="(item, k) in parasPosForm.runwaynum.startpos_num ? Array(parasPosForm.runwaynum.startpos_num) : []"
|
|
:key="k"
|
|
style="width: 100%"
|
|
>
|
|
<div>路线{{ k + 1 }}设置</div>
|
|
<el-form-item
|
|
v-for='(field, idx) in ["startpos_lon_a", "startpos_lat_a", "startpos_alt_a", "startpos_lon_b", "startpos_lat_b", "startpos_alt_b", "startpos_ori"]'
|
|
:key="idx"
|
|
:label="field"
|
|
style="width: calc(33% - 32px); display: inline-block"
|
|
>
|
|
<el-input
|
|
v-model="parasPosForm.runwayedges[k][field]"
|
|
/>
|
|
</el-form-item>
|
|
</div>
|
|
|
|
<el-form-item style="width: 100%">
|
|
<div style="text-align: center; width: 100%">
|
|
<el-button type="primary" @click="saveParasPosForm">保存</el-button>
|
|
</div>
|
|
</el-form-item>
|
|
</el-form>
|
|
|
|
<!--屏蔽信息设置-->
|
|
<div>屏蔽信息设置</div>
|
|
<el-form :model="shieldInfoForm" inline label-width="auto" label-position="top">
|
|
<el-form-item style="width: calc(33% - 32px)" label="偏差值">
|
|
<el-input v-model="shieldInfoForm.deviation"/>
|
|
</el-form-item>
|
|
<el-form-item style="width: calc(33% - 32px)" label="范围">
|
|
<el-input v-model="shieldInfoForm.shielding"/>
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item style="width: 100%">
|
|
<div style="text-align: center; width: 100%">
|
|
<el-button type="primary" @click="saveShieldInfoForm">保存</el-button>
|
|
</div>
|
|
</el-form-item>
|
|
</el-form>
|
|
</div>
|
|
<div v-if="itemsId === 4" class="tabsItem">
|
|
<!-- 时间范围选择 -->
|
|
<el-card shadow="always">
|
|
<el-date-picker
|
|
v-model="date1"
|
|
type="datetimerange"
|
|
range-separator="到"
|
|
start-placeholder="选择开始时间"
|
|
end-placeholder="选择结束时间"
|
|
style="width: calc(100% - 20px)"
|
|
/>
|
|
</el-card>
|
|
|
|
<!-- 区域选择 -->
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<el-form inline :model="form3" class="demo-form-inline">
|
|
<el-form-item v-for="(key, idx) in regionKeys" :key="idx">
|
|
<el-select
|
|
v-model="form3[key]"
|
|
:placeholder="key === 'region2' ? '请选择固定目标' : key === 'region' ? '请选择机场' : '请选择区域'"
|
|
style="width: 170px"
|
|
clearable
|
|
>
|
|
<el-option
|
|
v-for="option in getOptions(key)"
|
|
:key="option.value"
|
|
:label="option.label"
|
|
:value="option.value"
|
|
/>
|
|
</el-select>
|
|
</el-form-item>
|
|
</el-form>
|
|
</el-card>
|
|
|
|
<!-- 检测输入 -->
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<el-form inline :model="form4" class="demo-form-inline">
|
|
<el-form-item>
|
|
<el-input
|
|
v-model="form4.input1"
|
|
placeholder="请输入检测序号"
|
|
style="width: 170px"
|
|
/>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-date-picker
|
|
v-model="form4.date1"
|
|
type="date"
|
|
placeholder="选择检测时间"
|
|
clearable
|
|
style="width: 170px"
|
|
/>
|
|
</el-form-item>
|
|
|
|
<el-form-item>
|
|
<el-input
|
|
v-model="form4.input2"
|
|
placeholder="请输入检测人员"
|
|
style="width: 170px"
|
|
/>
|
|
</el-form-item>
|
|
</el-form>
|
|
</el-card>
|
|
|
|
<!-- 数据表格 -->
|
|
<el-card shadow="always" style="margin-top: 12px">
|
|
<el-table :data="tableData2" style="width: 100%">
|
|
<el-table-column label="序号" type="index" width="60"/>
|
|
<el-table-column prop="coordinate" label="目标坐标" width="120"/>
|
|
<el-table-column label="告警等级" width="160">
|
|
<template #default="scope">
|
|
<el-rate :model-value="scope.row.rate" disabled/>
|
|
</template>
|
|
</el-table-column>
|
|
<el-table-column label="处理状态" show-overflow-tooltip>
|
|
<template #default="scope">
|
|
<span v-if="scope.row.type === 1">已处理</span>
|
|
<span v-else-if="scope.row.type === 2">不处理</span>
|
|
<span v-else-if="scope.row.type === 3">未发现</span>
|
|
</template>
|
|
</el-table-column>
|
|
</el-table>
|
|
</el-card>
|
|
</div>
|
|
</div>
|
|
|
|
<el-button style="position: fixed;top: 10px;right: 10px" type="primary" @click="toAdmin">管理</el-button>
|
|
|
|
<el-dialog v-model="dialog1" title="处置障碍物" width="500">
|
|
<el-form :model="dialog1Form">
|
|
<el-form-item label="发现时间" label-width="120px">
|
|
<el-input v-model="dialog1Form.time" disabled autocomplete="off"/>
|
|
</el-form-item>
|
|
<el-form-item label="处理信息描述" label-width="120px">
|
|
<el-input
|
|
v-model="dialog1Form.text"
|
|
style="width: 100%"
|
|
:rows="2"
|
|
type="textarea"
|
|
/>
|
|
</el-form-item>
|
|
<el-form-item label="处理照片" label-width="120px">
|
|
<el-upload @click.stop="onSubmit"
|
|
ref="uploadRef"
|
|
:http-request="uploadFiles"
|
|
:file-list="dialog1Form.fileList.concat({type:'camera'})"
|
|
list-type="picture-card">
|
|
|
|
<template #trigger>
|
|
<el-icon>
|
|
<Plus/>
|
|
</el-icon>
|
|
</template>
|
|
|
|
<template #file="{ file }">
|
|
<div v-if="file.type !=='camera'">
|
|
<img class="el-upload-list__item-thumbnail" :src="file.url" alt=""/>
|
|
<span class="el-upload-list__item-actions">
|
|
<span
|
|
class="el-upload-list__item-preview"
|
|
@click="handlePictureCardPreview(file)"
|
|
>
|
|
<el-icon><zoom-in/></el-icon>
|
|
</span>
|
|
<span
|
|
class="el-upload-list__item-delete"
|
|
@click="handleDownload(file)"
|
|
>
|
|
<el-icon><Download/></el-icon>
|
|
</span>
|
|
<span
|
|
class="el-upload-list__item-delete"
|
|
@click="handleRemove(file)"
|
|
>
|
|
<el-icon><Delete/></el-icon>
|
|
</span>
|
|
</span>
|
|
</div>
|
|
<div @click="openCamera" v-if="file.type ==='camera'" style="border:none" class="el-upload--picture-card">
|
|
<el-icon>
|
|
<Camera/>
|
|
</el-icon>
|
|
</div>
|
|
</template>
|
|
</el-upload>
|
|
</el-form-item>
|
|
</el-form>
|
|
<template #footer>
|
|
<div class="dialog-footer">
|
|
<el-button @click="dialog1 = false">关闭</el-button>
|
|
<el-button type="primary" @click="handleObstacle">
|
|
处理
|
|
</el-button>
|
|
</div>
|
|
</template>
|
|
</el-dialog>
|
|
|
|
<el-dialog v-model="imgDialog">
|
|
<img width="100%" :src="dialogImageUrl" alt="Preview Image"/>
|
|
</el-dialog>
|
|
|
|
<el-dialog v-model="shildDialog">
|
|
<el-form :model="shildForm">
|
|
<el-form-item label="固定物名称" label-width="120px">
|
|
<el-input v-model="shildForm.name" autocomplete="off"/>
|
|
</el-form-item>
|
|
<el-form-item label="偏差值" label-width="120px">
|
|
<el-input v-model="shildForm.deviation" autocomplete="off"/>
|
|
</el-form-item>
|
|
<el-form-item label="屏蔽范围" label-width="120px">
|
|
<el-input v-model="shildForm.shielding" autocomplete="off"/>
|
|
</el-form-item>
|
|
</el-form>
|
|
<template #footer>
|
|
<div class="dialog-footer">
|
|
<el-button @click="shildDialog = false">关闭</el-button>
|
|
<el-button type="primary" @click="addShildItem">
|
|
确定
|
|
</el-button>
|
|
</div>
|
|
</template>
|
|
</el-dialog>
|
|
<input
|
|
ref="cameraInput"
|
|
type="file"
|
|
accept="image/*"
|
|
capture="camera"
|
|
style="display: none"
|
|
@change="handleCameraChange"
|
|
/>
|
|
</div>
|
|
</template>
|
|
<script setup lang="jsx">
|
|
import Ruler from "@/components/ruler.vue";
|
|
import {onMounted, ref, watch} from "vue";
|
|
import {
|
|
addShildData,
|
|
DeleteCData,
|
|
delShildData,
|
|
getAllAirPort,
|
|
GetAllCData,
|
|
getArea,
|
|
getParasPos,
|
|
getParasSignalpro, GetSetting,
|
|
getShildList,
|
|
InserDealObstacleModel,
|
|
InsertCListData, PosSetting,
|
|
RestartWork,
|
|
setParasPos,
|
|
setParasSignalpro,
|
|
ShutDownWork,
|
|
StartWork,
|
|
StopWork,
|
|
UpdateCData,
|
|
UpdateFodScopeData,
|
|
UpdateImage,
|
|
updateShildData
|
|
} from "@/api/api";
|
|
import {useRouter} from "vue-router";
|
|
import {ElMessage} from "element-plus";
|
|
|
|
|
|
const router = useRouter();
|
|
|
|
const rulerRef = ref()
|
|
|
|
const getSocket = () => {
|
|
|
|
// const socket = new WebSocket("ws://192.168.1.123:7789/ws");
|
|
const socket = new WebSocket("ws://192.168.1.123:7789/ws");
|
|
|
|
// 2. 连接成功时触发
|
|
socket.addEventListener("open", () => {
|
|
console.log("✅ WebSocket 连接成功");
|
|
});
|
|
|
|
// 3. 接收消息
|
|
socket.addEventListener("message", (event) => {
|
|
processData(JSON.parse(event.data))
|
|
});
|
|
|
|
// 4. 连接关闭时触发
|
|
socket.addEventListener("close", () => {
|
|
console.log("❌ WebSocket 已关闭");
|
|
});
|
|
|
|
// 5. 出错时触发
|
|
socket.addEventListener("error", (error) => {
|
|
console.error("⚠️ WebSocket 出错:", error);
|
|
});
|
|
}
|
|
|
|
const processData = (data) => {
|
|
if (data.FodName === '雷达信息') {
|
|
radarWorkState.value = data.radar_work_state
|
|
}
|
|
if (data.FodName === '图像数据') {
|
|
}
|
|
if (data.FodName === 'fod信息') {
|
|
console.log('障碍物信息', data)
|
|
setDot(data)
|
|
}
|
|
if (data.ty === 3) {
|
|
}
|
|
if (data.FodName === '车辆信息') {
|
|
currentPosition.value = {
|
|
lon: data.Longitude,
|
|
lat: data.Latitude,
|
|
}
|
|
const rectInfo = calcRectangleFromPoints(areaPoints.value);
|
|
const point = {
|
|
lon: data.Longitude,
|
|
lat: data.Latitude,
|
|
rotate: data.HeadingAngle
|
|
};
|
|
const local = correctDeviation(getLocalPositionRelativeToP4(point, areaPoints.value, rectInfo))
|
|
|
|
|
|
boxPos.value = local
|
|
|
|
}
|
|
};
|
|
|
|
// 默认方法
|
|
const onSubmit = (e) => {
|
|
console.log(e)
|
|
}
|
|
|
|
// 当前菜单id
|
|
const itemsId = ref(1)
|
|
// 菜单id跳转
|
|
const setItemsId = (e) => {
|
|
itemsId.value = e
|
|
}
|
|
// 雷达状态
|
|
const radarWorkState = ref(0)
|
|
// 车辆位置
|
|
const boxPos = ref({x: 100, y: 150, rotate: 0})
|
|
// 左侧区域宽高
|
|
const areaData = ref({
|
|
width: 500,
|
|
height: 1000,
|
|
angle: 0
|
|
})
|
|
// 障碍物点位
|
|
const dots = ref([])
|
|
// 当前的目标障碍物
|
|
const dotIndex = ref(-1)
|
|
// 跳转到管理界面
|
|
const toAdmin = () => {
|
|
router.push('/serve')
|
|
}
|
|
// 检测&成像设置
|
|
const parasSignalproForm = ref({
|
|
imaging: {},
|
|
dectection: {}
|
|
})
|
|
// 路线设置
|
|
const parasPosForm = ref({
|
|
runwaynum: {
|
|
startpos_num: 1
|
|
},
|
|
runwayedges: [
|
|
{
|
|
startpos_lon_a: '',
|
|
startpos_lat_a: '',
|
|
startpos_alt_a: '',
|
|
startpos_lon_b: '',
|
|
startpos_lat_b: '',
|
|
startpos_alt_b: '',
|
|
startpos_ori: ''
|
|
}
|
|
]
|
|
})
|
|
// 屏蔽信息设置
|
|
const shieldInfoForm = ref({
|
|
deviation: '',
|
|
shielding: '',
|
|
})
|
|
// 获取配置文件
|
|
const getOption = () => {
|
|
getParasPos().then((e) => {
|
|
parasPosForm.value = {
|
|
runwaynum: e.runwaynum,
|
|
runwayedges: e.runwayedges,
|
|
}
|
|
})
|
|
getParasSignalpro().then((e) => {
|
|
parasSignalproForm.value = {
|
|
imaging: e.imaging,
|
|
dectection: e.dectection,
|
|
}
|
|
})
|
|
GetSetting().then((e) => {
|
|
shieldInfoForm.value = {
|
|
deviation: e.deviation,
|
|
shielding: e.shielding,
|
|
}
|
|
})
|
|
}
|
|
|
|
// 保存成像&检测设置
|
|
const saveParasSignalproForm = () => {
|
|
setParasSignalpro(parasSignalproForm.value).then(e => {
|
|
})
|
|
}
|
|
// 保存路线设置
|
|
const saveParasPosForm = () => {
|
|
setParasPos(parasPosForm.value).then(e => {
|
|
})
|
|
}
|
|
// 保存屏蔽信息
|
|
const saveShieldInfoForm = () => {
|
|
PosSetting(shieldInfoForm.value).then((e) => {
|
|
})
|
|
}
|
|
// 更新路线数量
|
|
const updateRunwayNum = (e) => {
|
|
let arr = new Array(e).fill(0)
|
|
.map((item, index) => {
|
|
if (index <= parasPosForm.value.runwayedges.length - 1) {
|
|
return JSON.parse(JSON.stringify(parasPosForm.value.runwayedges[index]))
|
|
} else {
|
|
return {
|
|
startpos_lon_a: '',
|
|
startpos_lat_a: '',
|
|
startpos_alt_a: '',
|
|
startpos_lon_b: '',
|
|
startpos_lat_b: '',
|
|
startpos_alt_b: '',
|
|
startpos_ori: ''
|
|
}
|
|
}
|
|
})
|
|
parasPosForm.value.runwayedges = JSON.parse(JSON.stringify(arr))
|
|
}
|
|
|
|
// 启动雷达
|
|
const StartWorkBtn = () => {
|
|
StartWork()
|
|
// rulerRef.value.setDot([], true)
|
|
setDot([], true)
|
|
tableData1.value = []
|
|
}
|
|
// 停止雷达
|
|
const StopWorkBtn = () => {
|
|
StopWork()
|
|
}
|
|
// 重启雷达
|
|
const RestartWorkBtn = () => {
|
|
RestartWork()
|
|
}
|
|
// 关闭雷达
|
|
const ShutDownWorkBtn = () => {
|
|
ShutDownWork()
|
|
}
|
|
|
|
const form1 = ref({})
|
|
const form2 = ref({})
|
|
const form3 = ref({})
|
|
const form4 = ref({})
|
|
const form5 = ref({})
|
|
const form6 = ref({})
|
|
|
|
// 固定物
|
|
// 固定物列表
|
|
const tableData2 = ref([])
|
|
// 当前页
|
|
const currentPage2 = ref(1)
|
|
// 每页数量
|
|
const pageSize2 = ref(10)
|
|
const total2 = ref(0)
|
|
const shildDialog = ref(false)
|
|
const shildForm = ref({})
|
|
// 打开固定物模态框
|
|
const openShildDialog = (e) => {
|
|
shildDialog.value = true
|
|
shildForm.value = {
|
|
$index: e.row.$index,
|
|
lon: e.row.Lon,
|
|
lat: e.row.Lat,
|
|
strength: e.row.Strength,
|
|
name: Math.random().toString().split('.')[1],
|
|
deviation: shieldInfoForm.value.deviation,
|
|
shielding: shieldInfoForm.value.shielding,
|
|
}
|
|
}
|
|
const openShildDialog1 = (e) => {
|
|
shildDialog.value = true
|
|
shildForm.value = e.row
|
|
}
|
|
// 设置固定物
|
|
const addShildItem = () => {
|
|
if (shildForm.value.id) {
|
|
updateShildData(shildForm.value).then(e => {
|
|
shildDialog.value = false
|
|
shildForm.value = {}
|
|
getShildTableList()
|
|
})
|
|
} else {
|
|
tableData1.value.splice((currentPage1.value - 1) * pageSize1.value + shildForm.value.$index, 1);
|
|
addShildData(shildForm.value).then(e => {
|
|
shildDialog.value = false
|
|
shildForm.value = {}
|
|
getShildTableList()
|
|
})
|
|
}
|
|
}
|
|
// 删除固定物
|
|
const delShildItem = (e) => {
|
|
delShildData(e.row).then(e => {
|
|
getShildTableList()
|
|
})
|
|
}
|
|
// 获取固定物列表
|
|
const getShildTableList = () => {
|
|
getShildList({
|
|
pageIndex: currentPage2.value,
|
|
pageSize: pageSize2.value,
|
|
}).then((e) => {
|
|
tableData2.value = e.data
|
|
total2.value = e.totalCount
|
|
|
|
})
|
|
}
|
|
|
|
// 障碍物列表
|
|
const tableData1 = ref([])
|
|
// 上传图片
|
|
const uploadRef = ref(null)
|
|
const uploadFiles = async (e) => {
|
|
const res = await UpdateImage({files: [e.file]})
|
|
}
|
|
// 图片放大
|
|
const handlePictureCardPreview = (file) => {
|
|
dialogImageUrl.value = file.url
|
|
imgDialog.value = true
|
|
}
|
|
// 图片下载
|
|
const handleDownload = (file) => {
|
|
console.log(file)
|
|
}
|
|
// 图片删除
|
|
const handleRemove = (file) => {
|
|
console.log(file)
|
|
dialog1Form.value.fileList = dialog1Form.value.fileList.filter(e => e.url !== file.url)
|
|
}
|
|
// 拍照
|
|
const cameraInput = ref(null)
|
|
const openCamera = () => {
|
|
cameraInput.value?.click()
|
|
}
|
|
const handleCameraChange = (e) => {
|
|
const file = e.target.files[0]
|
|
if (!file) return
|
|
const url = URL.createObjectURL(file)
|
|
UpdateImage({files: [file]})
|
|
dialog1Form.fileList.value.push({
|
|
name: file.name,
|
|
url,
|
|
raw: file
|
|
})
|
|
// 重置 input
|
|
e.target.value = ''
|
|
}
|
|
// 处置模态框
|
|
const dialog1 = ref(false)
|
|
const dialog1Form = ref({})
|
|
const dialogImageUrl = ref('')
|
|
const imgDialog = ref(false)
|
|
// 删除
|
|
const delPoint = (e) => {
|
|
let arr = JSON.parse(JSON.stringify(tableData1.value))
|
|
arr.splice((currentPage1.value - 1) * pageSize1.value + e.$index, 1);
|
|
tableData1.value = arr || []
|
|
if ((currentPage1.value - 1) * pageSize1.value + e.$index === dotIndex.value) {
|
|
console.log(123)
|
|
dotIndex.value = -1
|
|
}
|
|
}
|
|
// 当前页
|
|
const currentPage1 = ref(1)
|
|
// 每页数量
|
|
const pageSize1 = ref(10)
|
|
|
|
// 打开处置模态框
|
|
const openHandleObstacleDialog = (e) => {
|
|
console.log(e.row)
|
|
dialog1.value = true
|
|
dialog1Form.value = {
|
|
dotIndex : e.index,
|
|
index: e.$index,
|
|
time: e.row.time,
|
|
longitude: e.row.Lon,
|
|
latitude: e.row.Lat,
|
|
strength: e.row.Strength,
|
|
fileList: [{url: 'https://dummyimage.com/640x360'}, {url: 'https://placehold.co/600x400/png'}]
|
|
}
|
|
}
|
|
// 确定处理
|
|
const handleObstacle = () => {
|
|
InserDealObstacleModel({
|
|
longitude: dialog1Form.value.longitude,
|
|
latitude: dialog1Form.value.latitude,
|
|
strength: dialog1Form.value.strength,
|
|
files: dialog1Form.value.fileList.map(e => e.url),
|
|
findTime: dialog1Form.value.time,
|
|
}).then(e => {
|
|
dialog1.value = false
|
|
dotIndexF(dialog1Form.value.dotIndex)
|
|
tableData1.value.splice((currentPage1.value - 1) * pageSize1.value + dialog1Form.value.index, 1);
|
|
})
|
|
}
|
|
|
|
// 批量处理
|
|
const table1Ref = ref()
|
|
const batchDisposal = () => {
|
|
dotIndex.value = -1
|
|
let datas = table1Ref.value.getSelectionRows()
|
|
datas.forEach((row) => {
|
|
InserDealObstacleModel({
|
|
longitude: row.Lon,
|
|
latitude: row.Lat,
|
|
strength: row.Strength,
|
|
files: [],
|
|
findTime: '',
|
|
})
|
|
tableData1.value = tableData1.value.filter(item => row.index !== item.index);
|
|
})
|
|
getShildTableList()
|
|
}
|
|
// 批量删除
|
|
const batchDel = () => {
|
|
dotIndex.value = -1
|
|
let datas = table1Ref.value.getSelectionRows()
|
|
datas.forEach((row) => {
|
|
tableData1.value = tableData1.value.filter(item => row.index !== item.index);
|
|
})
|
|
}
|
|
// 批量固定
|
|
const batchShild = () => {
|
|
dotIndex.value = -1
|
|
let datas = table1Ref.value.getSelectionRows()
|
|
datas.forEach((row) => {
|
|
addShildData({
|
|
lon: row.Lon,
|
|
lat: row.Lat,
|
|
strength: row.Strength,
|
|
name: Math.random().toString().split('.')[1],
|
|
deviation: shieldInfoForm.value.deviation,
|
|
shielding: shieldInfoForm.value.shielding,
|
|
})
|
|
tableData1.value = tableData1.value.filter(item => row.index !== item.index);
|
|
})
|
|
getShildTableList()
|
|
}
|
|
|
|
const date1 = ref([])
|
|
const currentPosition = ref({})
|
|
const DEG_TO_RAD = Math.PI / 180;
|
|
|
|
function lonLatToXY(lon, lat, lat0) {
|
|
const R = 111320; // 每度约等于 111.32 km
|
|
const cosLat = Math.cos(lat0 * DEG_TO_RAD);
|
|
return {x: lon * R * cosLat, y: lat * R};
|
|
}
|
|
|
|
function calcRectangleFromPoints(points) {
|
|
if (!points || points.length !== 4) throw new Error("必须传入4个点");
|
|
const p1raw = points.find(p => p.name === "点位1") || points[0];
|
|
const p2raw = points.find(p => p.name === "点位2") || points[1];
|
|
const p3raw = points.find(p => p.name === "点位3") || points[2];
|
|
const p4raw = points.find(p => p.name === "点位4") || points[3];
|
|
|
|
const lat0 = (points.reduce((s, p) => s + p.lat, 0) / points.length);
|
|
const p1 = lonLatToXY(p1raw.lon, p1raw.lat, lat0);
|
|
const p2 = lonLatToXY(p2raw.lon, p2raw.lat, lat0);
|
|
const p3 = lonLatToXY(p3raw.lon, p3raw.lat, lat0);
|
|
const p4 = lonLatToXY(p4raw.lon, p4raw.lat, lat0);
|
|
|
|
const dist = (a, b) => Math.hypot(a.x - b.x, a.y - b.y);
|
|
|
|
const top = dist(p2, p3);
|
|
const bottom = dist(p4, p1);
|
|
const left = dist(p4, p3);
|
|
const right = dist(p1, p2);
|
|
|
|
const trapezoid = Math.abs(top - bottom) / Math.max(top, bottom) > 0.05;
|
|
const width = trapezoid ? (top + bottom) / 2 : bottom;
|
|
const height = trapezoid ? (left + right) / 2 : right;
|
|
|
|
const bottomAngleRad = Math.atan2(p1.y - p4.y, p1.x - p4.x);
|
|
const angleDeg = bottomAngleRad * 180 / Math.PI;
|
|
return {
|
|
width: Number(width.toFixed(2)),
|
|
height: Number(height.toFixed(2)),
|
|
angle: Number(angleDeg.toFixed(6))
|
|
};
|
|
}
|
|
|
|
function getLocalPositionRelativeToP4(point, rectPoints, rectInfo) {
|
|
if (!rectInfo || typeof rectInfo.angle !== 'number') {
|
|
throw new Error("rectInfo.angle 必须是由 calcRectangleFromPoints 得到的角度");
|
|
}
|
|
|
|
const lat0 = rectPoints.reduce((a, b) => a + b.lat, 0) / rectPoints.length;
|
|
const p4raw = rectPoints.find(p => p.name === "点位4") || rectPoints[3];
|
|
|
|
const p4 = lonLatToXY(p4raw.lon, p4raw.lat, lat0);
|
|
const p = lonLatToXY(point.lon, point.lat, lat0);
|
|
|
|
const dx = p.x - p4.x;
|
|
const dy = p.y - p4.y;
|
|
|
|
const rad = -rectInfo.angle * DEG_TO_RAD;
|
|
const x_local = dx * Math.cos(rad) - dy * Math.sin(rad);
|
|
const y_local = dx * Math.sin(rad) + dy * Math.cos(rad);
|
|
|
|
const rotate_local = ((point.rotate ?? 0) + rectInfo.angle + 360) % 360;
|
|
|
|
return {
|
|
x: Number(x_local.toFixed(2)),
|
|
y: Number(y_local.toFixed(2)),
|
|
rotate: Number(rotate_local.toFixed(2))
|
|
};
|
|
}
|
|
|
|
// 偏差值
|
|
const deviationValue = ref({})
|
|
// 设置偏差值
|
|
const getDeviationValue = (e) => {
|
|
if (!e) return
|
|
const rectInfo = calcRectangleFromPoints(areaPoints.value);
|
|
const local = getLocalPositionRelativeToP4(e, areaPoints.value, rectInfo);
|
|
deviationValue.value = local;
|
|
}
|
|
// 修正偏差
|
|
const correctDeviation = (e) => {
|
|
let numH = deviationValue.value.y
|
|
let numW = deviationValue.value.x
|
|
let Ratio = e.y / numH
|
|
return {
|
|
...e,
|
|
x: e.x - numW * Ratio
|
|
}
|
|
}
|
|
const getPoint = (k) => {
|
|
regionLabeling.value[k].lon = currentPosition.value.lon || 0
|
|
regionLabeling.value[k].lat = currentPosition.value.lat || 0
|
|
}
|
|
const setDot = (e, re) => {
|
|
let fodInfo = e.DarDatas.filter(v => v.Lon !== 0 && v.Lat !== 0)
|
|
tableData1.value = [...tableData1.value, ...fodInfo.map((v, k) => {
|
|
return {...v, index: tableData1.value.length + k}
|
|
})]
|
|
}
|
|
|
|
const dotIndexF = (index) => {
|
|
if (dotIndex.value === index) {
|
|
dotIndex.value = -1
|
|
} else if (dotIndex.value > index) {
|
|
dotIndex.value -= 1
|
|
} else if (dotIndex.value > index) {
|
|
|
|
} else {
|
|
|
|
}
|
|
}
|
|
const table1Current = (e, v) => {
|
|
if (dotIndex.value === e.index) {
|
|
dotIndex.value = -1
|
|
} else {
|
|
dotIndex.value = e.index || 0
|
|
}
|
|
}
|
|
|
|
|
|
const options1 = ref([])
|
|
const options2 = ref([])
|
|
const getTableData = async () => {
|
|
await getAllAirPort().then(e => {
|
|
options1.value = e.data
|
|
})
|
|
}
|
|
const selectUpdate = (val) => {
|
|
form1.value.region = val
|
|
getArea({airId: val}).then(e => {
|
|
options2.value = e.data
|
|
})
|
|
}
|
|
const select2Update = (val) => {
|
|
form6.value.region = val
|
|
getArea({airId: val}).then(e => {
|
|
options2.value = e.data
|
|
})
|
|
}
|
|
const select3Update = (val) => {
|
|
form6.value.region1 = val
|
|
}
|
|
|
|
// 区域标定页面获取区域信息
|
|
const getArea2 = () => {
|
|
GetAllCData({areaId: form6.value.region1}).then((res) => {
|
|
regionLabeling.value = res.data
|
|
})
|
|
}
|
|
// 区域标定信息
|
|
const regionLabeling = ref([])
|
|
// 区域点位信息
|
|
const areaPoints = ref([])
|
|
// 保存区域信息
|
|
const saveAreaPoint = () => {
|
|
let arr = []
|
|
regionLabeling.value.forEach((v, k) => {
|
|
arr.push(UpdateCData(v))
|
|
})
|
|
Promise.all(arr).then(() => {
|
|
areaPoints.value = JSON.parse(JSON.stringify(regionLabeling.value))
|
|
ElMessage.success('保存成功')
|
|
let data = calcRectangleFromPoints(areaPoints.value)
|
|
areaData.value = data
|
|
getDeviationValue(areaPoints.value.find(e => e.name === '点位3'))
|
|
|
|
}).catch((err) => {
|
|
ElMessage.error(err)
|
|
})
|
|
}
|
|
|
|
// 主界面获取区域信息
|
|
const getAreaData = () => {
|
|
GetAllCData({areaId: form1.value.region1}).then((res) => {
|
|
areaPoints.value = res.data
|
|
let data = calcRectangleFromPoints(res.data)
|
|
areaData.value = data
|
|
getDeviationValue(res.data.find(e => e.name === '点位3'))
|
|
|
|
const sorted = res.data.sort((a, b) => {
|
|
const numA = parseInt(a.name.replace(/[^\d]/g, ''), 10);
|
|
const numB = parseInt(b.name.replace(/[^\d]/g, ''), 10);
|
|
return numA - numB;
|
|
});
|
|
const letters = ['a', 'b', 'c', 'd'];
|
|
const result = {};
|
|
sorted.forEach((item, index) => {
|
|
const key = letters[index];
|
|
result[`${key}Lon`] = item.lon;
|
|
result[`${key}Lat`] = item.lat;
|
|
});
|
|
UpdateFodScopeData(result)
|
|
})
|
|
}
|
|
|
|
|
|
const testFun = () => {
|
|
|
|
areaPoints.value = [
|
|
{
|
|
name: '点位1',
|
|
lon: 120.263824,
|
|
lat: 36.985199,
|
|
},
|
|
{
|
|
name: '点位2',
|
|
lon: 120.271545,
|
|
lat: 36.9816399,
|
|
},
|
|
{
|
|
name: '点位3',
|
|
lon: 120.271736,
|
|
lat: 36.9818535,
|
|
},
|
|
{
|
|
name: '点位4',
|
|
lon: 120.264008,
|
|
lat: 36.985424,
|
|
},
|
|
]
|
|
let data = calcRectangleFromPoints(areaPoints.value)
|
|
areaData.value = data
|
|
getDeviationValue(areaPoints.value.find(e => e.name === '点位3'))
|
|
let data1 = {
|
|
lon: 120.2716405,
|
|
lat: 36.9817467,
|
|
rotate: 303
|
|
}
|
|
const getLocal = () => {
|
|
const rectInfo = calcRectangleFromPoints(areaPoints.value);
|
|
|
|
const local = correctDeviation(getLocalPositionRelativeToP4(data1, areaPoints.value, rectInfo))
|
|
boxPos.value = local
|
|
}
|
|
getLocal()
|
|
let fx = 1
|
|
setInterval(() => {
|
|
data1.lon += fx * -0.0077245 * 0.01 / 60
|
|
data1.lat += fx * 0.0035648 * 0.01 / 60
|
|
getLocal()
|
|
}, 1000 / 60)
|
|
setInterval(() => {
|
|
fx *= -1
|
|
}, 1000 * 80)
|
|
}
|
|
const getDotsData = () => {
|
|
dots.value = [...tableData1.value.map((v, k) => {
|
|
const rectInfo = calcRectangleFromPoints(areaPoints.value);
|
|
const point = {
|
|
lon: v.Lon,
|
|
lat: v.Lat,
|
|
rotate: 0,
|
|
};
|
|
return {
|
|
...correctDeviation(getLocalPositionRelativeToP4(point, areaPoints.value, rectInfo)),
|
|
type: v.Status
|
|
}
|
|
}).filter(v => v)]
|
|
}
|
|
watch(() => [tableData1.value, areaPoints.value], (oldVal, newVal) => {
|
|
getDotsData()
|
|
}, {deep: true})
|
|
onMounted(() => {
|
|
getTableData()
|
|
getSocket()
|
|
getOption()
|
|
getShildTableList()
|
|
// testFun()
|
|
getDotsData()
|
|
})
|
|
|
|
|
|
</script>
|
|
<style>
|
|
.app-container {
|
|
width: 1080px;
|
|
min-width: 1080px;
|
|
height: 100vh;
|
|
margin: 0 auto;
|
|
}
|
|
|
|
.top {
|
|
width: 100%;
|
|
height: 80px;
|
|
font-size: 24px;
|
|
background-color: #202227;
|
|
position: relative;
|
|
}
|
|
|
|
.items {
|
|
position: absolute;
|
|
top: 15%;
|
|
left: 50%;
|
|
transform: translateX(-50%);
|
|
white-space: nowrap;
|
|
}
|
|
|
|
.item {
|
|
display: inline-block;
|
|
width: 120px;
|
|
height: 100%;
|
|
color: #8C9A9C;
|
|
line-height: 56px;
|
|
cursor: pointer;
|
|
font-size: 18px;
|
|
font-weight: 600;
|
|
letter-spacing: 2px;
|
|
}
|
|
|
|
.item span {
|
|
display: inline-block;
|
|
height: 100%;
|
|
}
|
|
|
|
.item.click span,
|
|
.item:hover span {
|
|
color: #0D52BF;
|
|
border-bottom: 2px solid #0D52BF;
|
|
}
|
|
|
|
.title {
|
|
line-height: 80px;
|
|
color: #fff;
|
|
margin-left: 20px;
|
|
display: inline-block;
|
|
text-align: left;
|
|
}
|
|
|
|
.left {
|
|
display: inline-block;
|
|
height: calc(100vh - 80px);
|
|
width: 34%;
|
|
vertical-align: top;
|
|
}
|
|
|
|
.right {
|
|
padding: 20px 20px 0 20px;
|
|
display: inline-block;
|
|
height: calc(100vh - 80px - 20px);
|
|
width: calc(66% - 40px);
|
|
background-color: #F6F7FB;
|
|
overflow: auto;
|
|
}
|
|
</style> |