运营区设置页面

This commit is contained in:
tx 2024-12-21 15:41:58 +08:00
parent c4ab198423
commit 772bb2eab1
13 changed files with 2205 additions and 196 deletions

View File

@ -5,11 +5,11 @@ const install = (Vue, vm) => {
Vue.prototype.$u.http.setConfig({
// baseUrl: 'http://61.174.243.28:15861',
// baseUrl: 'http://192.168.2.46:8080',
// baseUrl: 'https://che.chuangtewl.com/prod-api',
baseUrl: 'https://che.chuangtewl.com/prod-api',
// 测试环境
// baseUrl: 'https://dianche.chuantewulian.cn/prod-api',
// 俞山岛
baseUrl: 'https://dche.ccttiot.com/prod-api',
// baseUrl: 'https://dche.ccttiot.com/prod-api',
// 创特
loadingText: '努力加载中~',
loadingTime: 10000,

View File

@ -5,13 +5,15 @@
back-icon-color='#fff' height='45'></u-navbar>
</view>
<image class="backimg" src="https://lxnapi.ccttiot.com/bike/img/static/uJVTiExwVDJJzYywmoLc" mode=""
v-if="deptId == 101"></image>
<image class="backimg" src="https://lxnapi.ccttiot.com/bike/img/static/uvRt04OhPwHf2MRkU6mk" mode=""
v-if="deptId == 100"></image>
<view class="backimg-container">
<image class="backimg" src="https://lxnapi.ccttiot.com/bike/img/static/uJVTiExwVDJJzYywmoLc"
mode="aspectFit" v-if="deptId == 101"></image>
<image class="backimg" src="https://lxnapi.ccttiot.com/bike/img/static/uvRt04OhPwHf2MRkU6mk"
mode="aspectFit" v-if="deptId == 100"></image>
</view>
<view class="statusBar" :style="{ paddingTop: statusBarHeight + 'px' }"></view>
<view class="statusBars" :style="{ height: navBarHeight + 'px' }"></view>
<view class="topbg">
<view class="topbg" style="z-index: 10;position: relative;">
<view class="topbox">
<view class="info">
<image v-if="info.avatar != ''" :src='info.avatar' mode=""></image>
@ -687,7 +689,7 @@ export default {
}
</script>
<
<style lang="scss">
/deep/ .uni-navbar {
background: transparent !important; // 使 !important

View File

@ -360,6 +360,42 @@
"navigationStyle": "custom",
"enablePullDownRefresh" : false
}
},
{
"path" : "map_set",
"style" :
{
"navigationBarTitleText" : "",
"navigationStyle": "custom",
"enablePullDownRefresh" : false
}
},
{
"path" : "ParkIngList",
"style" :
{
"navigationBarTitleText" : "",
"navigationStyle": "custom",
"enablePullDownRefresh" : false
}
},
{
"path" : "Parking_set",
"style" :
{
"navigationBarTitleText" : "",
"navigationStyle": "custom",
"enablePullDownRefresh" : false
}
},
{
"path" : "park_map",
"style" :
{
"navigationBarTitleText" : "",
"navigationStyle": "custom",
"enablePullDownRefresh" : false
}
}

View File

@ -61,7 +61,7 @@
<!-- <u-navbar :is-back="false" title="共享电动车" :border-bottom="false" :background="bgc" title-color='#2E4975'
title-size='36' height='36'></u-navbar> -->
<map class="map" id="map" ref="map" :scale="zoomSize" show-location :key="mapKey" :latitude="latitude"
<map class="map" id="map" ref="map" :scale="zoomSize" show-location :key="mapKey" :latitude="latitude"
:longitude="longitude" :show-location="true" :markers="markers" :polygons="polyline" :polyline="mappolyline"
@markertap="onMarkerTap" @tap="onMapTap" @regionchange="onMapRegionChange">
@ -485,7 +485,7 @@
</view>
</view>
<view class="bottom_more" v-if="showindex == 1">
<view class="close" @click="showindex = 0">
<image src="https://api.ccttiot.com/smartmeter/img/static/uM76uO46a5cZOkFlffnX" mode=""></image>
@ -1089,8 +1089,8 @@ export default {
},
tozf: false,
islogin: true,
mapKey:0,
mapKey: 0,
}
},
watch: {
@ -1241,8 +1241,8 @@ export default {
that.gps.latitude = lb.latitude;
that.gps.longitude = lb.longitude;
that.latitude = parseFloat(lb.latitude) ;
that.longitude = parseFloat(lb.longitude);
that.latitude = parseFloat(lb.latitude);
that.longitude = parseFloat(lb.longitude);
that.deviceGps.latitude = lb.latitude;
that.deviceGps.longitude = lb.longitude;
// console.log(that.areaInfo, 'that.areaInfo');
@ -1383,129 +1383,129 @@ export default {
})
},
getArea() {
// this.showmap = false
//
this.areaInfo = {}
// console.log(this.qParam, 'qParamqParamqParamqParam');
this.polyline = []
if (/^\d{7}$/.test(this.qParam)) {
this.gps.sn = this.qParam
}
if (this.orderinfo.sn != '') {
this.gps.sn = this.orderinfo.sn
}
if (this.deviceInfos.sn) {
this.gps.sn = this.deviceInfos.sn
}
if (this.qParam != '' && this.qParam != null) {
this.sn = this.getSNFromQRCode(this.qParam);
if (this.sn != 0 && this.sn != '' && this.sn != null) {
this.gps.sn = this.sn
// this.showmap = false
//
this.areaInfo = {}
// console.log(this.qParam, 'qParamqParamqParamqParam');
this.polyline = []
if (/^\d{7}$/.test(this.qParam)) {
this.gps.sn = this.qParam
}
if (this.orderinfo.sn != '') {
this.gps.sn = this.orderinfo.sn
}
if (this.deviceInfos.sn) {
this.gps.sn = this.deviceInfos.sn
}
if (this.qParam != '' && this.qParam != null) {
this.sn = this.getSNFromQRCode(this.qParam);
if (this.sn != 0 && this.sn != '' && this.sn != null) {
this.gps.sn = this.sn
}
}
this.$u.get('/app/area/info?', this.gps).then((res) => {
if (res.code === 200) {
// 线
const polylines = this.convertBoundaryToPolyline(res.data.boundaryStr)
this.mapKey += 1
// 线
this.areaInfo = res.data
setTimeout(() => {
let abb = this.cleanedText()
//
this.duration = abb.length * 0.3; // 50 px/s
}, 200)
this.polyline.push(polylines)
uni.setStorageSync('areaId', res.data.areaId);
this.getinfo()
this.getmarks()
// this.getlist()
this.getParking()
if (uni.getStorageSync('token') && this.islogin) {
this.getisInOrder()
}
}
this.$u.get('/app/area/info?', this.gps).then((res) => {
if (res.code === 200) {
const hasShownPopup = uni.getStorageSync('hasShownPopup');
// 线
const polylines = this.convertBoundaryToPolyline(res.data.boundaryStr)
this.mapKey += 1
// 线
this.areaInfo = res.data
if (hasShownPopup === '' || hasShownPopup === null) {
this.showTips = true;
this.startCountdown();
uni.setStorageSync('hasShownPopup', true);
} else {
this.showTips = uni.getStorageSync('hasShownPopup');
;
this.startCountdown();
}
if (this.gps.sn && /^\d{7}$/.test(this.gps.sn)) {
setTimeout(() => {
let abb = this.cleanedText()
if (this.orderinfo.status) {
// this.showtcs = true
} else {
// if(this.showdevice==false&&){
// console.log('1');
// this.deviceInfo(1)
// this.qParam = null
// }
if (this.deviceInfos.sn) {
//
this.duration = abb.length * 0.3; // 50 px/s
}, 200)
this.polyline.push(polylines)
uni.setStorageSync('areaId', res.data.areaId);
this.getinfo()
this.getmarks()
// this.getlist()
this.getParking()
if (uni.getStorageSync('token')&&this.islogin) {
this.getisInOrder()
}
const hasShownPopup = uni.getStorageSync('hasShownPopup');
if (hasShownPopup === '' || hasShownPopup === null) {
this.showTips = true;
this.startCountdown();
uni.setStorageSync('hasShownPopup', true);
} else {
this.showTips = uni.getStorageSync('hasShownPopup');
;
this.startCountdown();
}
if (this.gps.sn && /^\d{7}$/.test(this.gps.sn)) {
setTimeout(() => {
if (this.orderinfo.status) {
// this.showtcs = true
} else {
// if(this.showdevice==false&&){
// console.log('1');
// this.deviceInfo(1)
// this.qParam = null
// }
if (this.deviceInfos.sn) {
if (this.qParam) {
console.log('进入的判断2');
this.deviceInfo(1)
this.qParam = null
} else {
if (this.qParam) {
console.log('进入的判断2');
this.deviceInfo(1)
this.qParam = null
} else {
console.log('进入的判断3');
// if(this.de)
this.deviceInfo(0)
this.qParam = null
}
console.log('进入的判断3');
// if(this.de)
this.deviceInfo(0)
this.qParam = null
}
}
}, 200)
}
} else {
uni.showToast({
title: res.msg,
icon: 'none',
duration: 2000
});
}
}, 200)
}
}).catch(error => {
});
} else {
uni.showToast({
title: res.msg,
icon: 'none',
duration: 2000
});
}
}).catch(error => {
});
},
deviceInfo(num) {
if (!this.sn) return;
//
if(this.orderinfo.sn){
if (this.orderinfo.sn!=this.sn) {
if (this.orderinfo.sn) {
if (this.orderinfo.sn != this.sn) {
uni.showToast({
title: '您有正在进行的订单,请先完成当前订单',
icon: 'none',
@ -1513,9 +1513,9 @@ export default {
});
return;
}
}
this.$u.get('/app/device/info?sn=' + this.sn).then((res) => {
if (res.code === 200) {
this.isqrcode = true
@ -1528,10 +1528,10 @@ export default {
//
const areaChanged = this.areaInfo.areaId != res.data.areaId;
console.log( this.areaInfo.areaId,res.data.areaId,'aaaaaaaaaaaaaaaaaaaaaa');
console.log(this.areaInfo.areaId, res.data.areaId, 'aaaaaaaaaaaaaaaaaaaaaa');
if (num == 0) {
if (areaChanged) {
//
uni.setStorageSync('areaId', res.data.areaId);
this.gps.sn = this.sn;
@ -1594,7 +1594,7 @@ export default {
},
mapFun() {
uni.openLocation({
latitude: parseFloat(this.parkinfo.latitude),
// - /
@ -1802,93 +1802,94 @@ export default {
}
},
onMarkerTap(e) {
this.showdevice = false;
this.deviceIndex = 0;
this.type = 0;
// this.freeInfo = {};
// this.freeListIndex = 0;
this.parkinfo = {}
this.showparkinfo = false
if (e.type === 'markertap') {
console.log('Clicked marker:', e.markerId);
//
let parkmark = this.parkingList.some(item => item.parkingId == e.markerId);
if (parkmark) {
const matchingItem = this.parkingList.find(item => item.parkingId == e.markerId);
this.showparkinfo = true
this.parkinfo = matchingItem
console.log(matchingItem, 'parkmarkparkmark');
this.showparkinfo = true;
this.parkinfo = matchingItem;
e.stopPropagation();
return;
}
//
const markerExists = this.listData.some(item => item.sn == e.markerId);
if (this.orderinfo.sn) {
uni.showToast({
title: '有进行中的订单,无法查看',
icon: 'none'
});
} else {
if (markerExists) {
this.sn = e.markerId;
this.deviceInfo(0);
if (markerExists) {
//
if (this.orderinfo.sn) {
uni.showToast({
title: '有进行中的订单,无法查看',
icon: 'none'
});
return;
}
// Revert the last clicked marker to the default image and zIndex
if (this.lastClickedMarkerId !== null) {
this.markers = this.markers.map(marker => {
if (marker.id === this.lastClickedMarkerId) {
return {
...marker,
iconPath: this.defaultMarkerIconPath,
width: 40,
height: 28,
// zIndex: 1 // Reset to default zIndex
};
}
return marker;
});
}
this.lastClickedMarkerId = e.markerId;
// Update the clicked marker's image and zIndex
// UI
this.showdevice = true;
this.deviceIndex = 1;
this.type = 1;
this.sn = e.markerId;
//
if (this.lastClickedMarkerId !== null) {
this.markers = this.markers.map(marker => {
if (marker.id === e.markerId) {
if (marker.id === this.lastClickedMarkerId) {
return {
...marker,
iconPath: this.clickedMarkerIconPath,
width: 50,
height: 35
// zIndex: 999 // Higher zIndex to bring it to the front
iconPath: this.defaultMarkerIconPath,
width: 40,
height: 28
};
}
return marker;
});
}
// Store the ID of the currently clicked marker
// Get the latitude and longitude of the clicked marker
const clickedMarker = this.markers.find(marker => marker.id === e.markerId);
if (clickedMarker) {
const {
latitude,
longitude
} = clickedMarker;
console.log(`Clicked marker location - Latitude: ${latitude}, Longitude: ${longitude}`);
this.routePlanning(latitude, longitude)
// Find nearby markers
// this.getNearbyMarkers(latitude, longitude);
//
this.markers = this.markers.map(marker => {
if (marker.id === e.markerId) {
return {
...marker,
iconPath: this.clickedMarkerIconPath,
width: 50,
height: 35
};
}
return marker;
});
// console.log(this.markers, 'Updated markers');
} else {
console.log('Marker ID does not exist in the list data');
this.lastClickedMarkerId = e.markerId;
//
this.deviceInfo(1);
// 线
const clickedMarker = this.markers.find(marker => marker.id === e.markerId);
if (clickedMarker) {
const { latitude, longitude } = clickedMarker;
this.routePlanning(latitude, longitude);
}
}
// Prevent event bubbling
e.stopPropagation();
}
},
// showDeviceInfo
showDeviceInfo(device) {
if (this.orderinfo.status) {
this.showtcs = true;
return;
}
this.deviceIndex = 1;
this.type = 1;
this.showdevice = true;
//
this.deviceInfo(0);
},
getNearbyMarkers(clickedLat, clickedLon) {
const nearbyMarkers = this.listData.filter(item => {
if (item.latitude && item.longitude) {
@ -2885,7 +2886,7 @@ export default {
// 使this.$setVue
this.$set(this, 'markers', this.markers);
}
console.log(this.markers,'aaa');
console.log(this.markers, 'aaa');
}
}).catch(error => {
console.error("Error fetching device data:", error);

View File

@ -19,15 +19,10 @@
</view>
</view>
<map class="map" id="map" ref="map" :scale="zoomSize" :latitude="latitude" :longitude="longitude"
:circles="circles" :show-location="true" :markers="markers" :polygons="polyline" @markertap="onMarkerTap"
@regionchange="onMapRegionChange">
<map class="map" id="map" ref="map" :scale="zoomSize" :latitude="latitude" :longitude="longitude" :circles="circles" :show-location="true" :markers="markers" :polygons="polyline" @markertap="onMarkerTap" @regionchange="onMapRegionChange">
<view class="center-marker">
<!-- https://lxnapi.ccttiot.com/bike/img/static/uZdgdHaduvMTKOpMKYxZ -->
<image src="https://lxnapi.ccttiot.com/bike/img/static/uZdgdHaduvMTKOpMKYxZ" style="width: 42rpx; height: 78rpx;" v-if="showdevList" />
<!-- <image src="https://lxnapi.ccttiot.com/bike/img/static/uAwnNprackI9p9dXFg8C" style="width: 80rpx; height: 80rpx;" /> -->
</view>
<image src="https://lxnapi.ccttiot.com/bike/img/static/uZdgdHaduvMTKOpMKYxZ" style="width: 42rpx; height: 78rpx;" v-if="showdevList" />
</view>
</map>
<view class="park" @click="toggleIconAndCallout">
<image src="https://lxnapi.ccttiot.com/bike/img/static/uRiYQZQEb3l2LsltEsyW" mode=""></image>

View File

@ -33,7 +33,9 @@
</view>
<view class="cont" @click="topage(4)" v-if="hasMenuPermission('用户页面')">
<image src="https://lxnapi.ccttiot.com/bike/img/static/uM4rBBaXc7b3TmsqQTvz" mode=""></image>
<!-- <image src="https://lxnapi.ccttiot.com/bike/img/static/uM4rBBaXc7b3TmsqQTvz" mode=""></image> -->
<image src="https://lxnapi.ccttiot.com/bike/img/static/ui9GMTP9k1K0YC9jlWHu" mode=""></image>
<view class="text">
用户页面
</view>
@ -74,7 +76,14 @@
style="width: 96rpx;height: 96rpx;"></image>
</view>
<view class="cont" @click="topage(12)" v-if="hasMenuPermission('区域设置')">
<image src="https://lxnapi.ccttiot.com/bike/img/static/uM4rBBaXc7b3TmsqQTvz" mode=""></image>
<!-- <image src="https://lxnapi.ccttiot.com/bike/img/static/ui9GMTP9k1K0YC9jlWHu" mode=""></image> -->
<view class="text">
区域设置
</view>
</view>
</view>
</view>
<u-select v-model="show" :list="list" title='选择运营区' @confirm="confirm" v-if="list.length <= 5"></u-select>
@ -308,6 +317,10 @@ export default {
uni.navigateTo({
url: '/pages_adminSet/sysSet'
})
} else if (num == 12) {
uni.navigateTo({
url: '/pages_adminSet/ParkIngList'
})
}
}

View File

@ -0,0 +1,314 @@
<template>
<view class="page">
<u-navbar title="地图绘制" :border-bottom="false" :background="bgc" title-color='#000' title-size='36' height='45'>
</u-navbar>
<!-- Tab切换 -->
<u-tabs-swiper :list="tabList" :current="current" @change="tabChange" :is-scroll="false" active-color="#007AFF"
bar-width="60" bar-height="6" item-width="250" :show-bar="true"></u-tabs-swiper>
<!-- 列表内容 -->
<view class="content">
<!-- 新增按钮 -->
<view class="add-btn" @tap="goAdd">
<u-icon name="plus" color="#007AFF" size="28"></u-icon>
<text>新增{{ currentTabName }}</text>
</view>
<!-- 列表 -->
<view class="list">
<u-swipe-action v-for="(item, index) in parkingList " :key="index" :show="item.show" :index="index"
@click="handleSwipeClick" :options="swipeOptions">
<view class="list-item">
<view class="item-left">
<view class="item-name">{{ item.parkingName }}</view>
<view class="item-info">
<text>创建时间: {{ item.createTime }}</text>
<text>误差距离: {{ item.error !== null ? item.error : 0 }}</text>
</view>
</view>
<view class="item-right">
<view class="custom-switch" @click="handleStatusChange(item)">
<view class="switch-track" :class="{ 'switch-active': item.status == 0 }">
<view class="switch-thumb"></view>
</view>
</view>
<u-button type="primary" size="mini" @click="goEdit(item)">修改</u-button>
</view>
</view>
</u-swipe-action>
</view>
<!-- 空状态 -->
<u-empty v-if="parkingList.length === 0" text="暂无数据" mode="list"></u-empty>
</view>
</view>
</template>
<script>
export default {
data() {
return {
bgc: {
backgroundColor: "#fff",
},
areaId: '',
current: 0,
tabList: [
{ name: '停车区' },
{ name: '禁停区' },
{ name: '禁行区' }
],
parkingList: [],
swipeOptions: [{
text: '删除',
style: {
backgroundColor: '#dd524d'
}
}]
}
},
computed: {
currentTabName() {
return this.tabList[this.current]?.name || ''
},
currentType() {
return String(this.current + 1)
}
},
onLoad(e) {
this.areaId = uni.getStorageSync('adminAreaid')
this.getParking()
},
methods: {
// Tab
tabChange(index) {
this.current = index
this.getParking()
},
//
getTypeName(type) {
const typeMap = {
'1': '停车区',
'2': '禁停区',
'3': '禁行区'
}
return typeMap[type] || '未知'
},
//
getParking() {
let data = {
areaId: this.areaId,
type: this.currentType
}
this.$u.get('/appVerify/parking/list', data).then((res) => {
if (res.code === 200) {
this.parkingList = res.data || []
console.log(this.parkingList[0])
} else {
uni.showToast({
title: res.msg || '获取数据失败',
icon: 'none'
})
}
})
},
//
goAdd() {
uni.navigateTo({
url: `/pages_adminSet/Parking_set?type=${this.currentType}`
})
},
//
goEdit(item) {
uni.navigateTo({
url: `/pages_adminSet/Parking_set?type=${item.type}&parkingId=${item.parkingId}`
})
},
//
handleStatusChange(item) {
const newStatus = item.status === 1 ? 0 : 1
let data = {
parkingId: item.parkingId,
status: newStatus
}
this.$u.put('/appVerify/parking/changeParkingStatus', data).then(res => {
if (res.code === 200) {
item.status = newStatus
uni.showToast({
title: '状态更新成功',
icon: 'success'
})
} else {
uni.showToast({
title: res.msg || '状态更新失败',
icon: 'none'
})
}
})
},
//
handleSwipeClick(e) {
console.log(e,'e');
// const { index } = e;
const item = this.parkingList[e];
console.log(item,'item');
uni.showModal({
title: '提示',
content: '确定要删除该记录吗?',
success: (res) => {
if (res.confirm) {
this.deleteParking(item.parkingId);
}
}
});
},
//
deleteParking(parkingId) {
this.$u.delete(`/appVerify/parking/${parkingId}`).then(res => {
if (res.code === 200) {
uni.showToast({
title: '删除成功',
icon: 'success'
});
this.getParking(); //
} else {
uni.showToast({
title: res.msg || '删除失败',
icon: 'none'
});
}
});
}
}
}
</script>
<style lang="scss">
page {
width: 100%;
height: 100%;
background-color: #f5f5f5;
}
.page {
width: 100%;
height: 100%;
}
.content {
padding: 20rpx;
}
.add-btn {
position: fixed;
bottom: 100rpx;
left: 60rpx;
display: flex;
width: 650rpx;
align-items: center;
justify-content: center;
height: 80rpx;
background-color: #fff;
border-radius: 8rpx;
margin-bottom: 20rpx;
text {
margin-left: 10rpx;
color: #007AFF;
font-size: 28rpx;
}
}
.content {
padding: 20rpx;
background-color: #f5f5f5;
}
.list {
.list-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 20rpx;
background-color: #fff;
border-radius: 8rpx;
// margin-bottom: 20rpx;
//
// border-bottom: 1px solid #f0f0f0;
background-color: #fafafa;
.item-left {
flex: 1;
.item-name {
font-size: 32rpx;
font-weight: bold;
color: #333;
margin-bottom: 10rpx;
}
.item-info {
font-size: 24rpx;
color: #999;
display: flex;
flex-direction: column;
text {
margin-bottom: 4rpx;
}
}
}
.item-right {
display: flex;
align-items: center;
gap: 20rpx;
}
}
}
.custom-switch {
display: inline-block;
.switch-track {
position: relative;
width: 100rpx;
height: 60rpx;
background-color: #e0e0e0;
border-radius: 30rpx;
transition: all 0.3s;
cursor: pointer;
.switch-thumb {
position: absolute;
left: 4rpx;
top: 4rpx;
width: 52rpx;
height: 52rpx;
background-color: #fff;
border-radius: 50%;
transition: all 0.3s;
box-shadow: 0 2rpx 4rpx rgba(0, 0, 0, 0.2);
}
&.switch-active {
background-color: #007AFF;
.switch-thumb {
transform: translateX(40rpx);
}
}
}
}
</style>

View File

@ -0,0 +1,495 @@
<template>
<view class="page">
<u-navbar :title="getTitle" :border-bottom="false" :background="bgc" title-color='#000' title-size='36'
height='45'>
</u-navbar>
<view class="content">
<view class="form-item">
<text class="label">区域名称</text>
<u-input v-model="formData.parkingName" placeholder="请输入区域名称" />
</view>
<view class="form-item">
<text class="label">区域类型</text>
<picker :range="typeList" range-key="text" @change="handleTypeChange">
<view class="type-selector">
<text>{{ getTypeName(formData.type) }}</text>
<u-icon name="arrow-right"></u-icon>
</view>
</picker>
</view>
<view class="form-item">
<text class="label">误差范围()</text>
<u-input v-model="formData.error" type="number" placeholder="请输入误差范围" />
</view>
<view class="form-item">
<text class="label">区域图片</text>
<view class="image-upload">
<image v-if="formData.picture" :src="formData.picture" mode="aspectFill" class="preview-image"
@tap="previewImage" />
<view v-else class="upload-btn" @tap="chooseImage">
<u-icon name="plus" size="40"></u-icon>
</view>
<view v-if="formData.picture" class="delete-btn" @tap="deletePic">
<u-icon name="close" color="#fff" size="20"></u-icon>
</view>
</view>
</view>
<view class="form-item">
<text class="label">区域坐标</text>
<view class="map-container">
<map id="map" :latitude="mapCenter.latitude" :longitude="mapCenter.longitude" :polygons="polygons"
scale="19"></map>
</view>
<!-- <view class="map-tips">点击地图添加坐标点至少需要3个点形成区域</view> -->
<view class="clear-btn" @tap="toSetmap">
修改区域
</view>
</view>
<view class="form-item">
<text class="label">备注</text>
<u-input v-model="formData.remark" type="textarea" placeholder="请输入备注信息" />
</view>
<view class="btn-group">
<u-button type="primary" @click="handleSubmit">保存</u-button>
<u-button @click="goBack">取消</u-button>
</view>
</view>
</view>
</template>
<script>
export default {
data() {
return {
bgc: {
backgroundColor: "#fff",
},
areaId: '',
parkingId: '',
typeList: [
{ text: '停车区', value: '1' },
{ text: '禁停区', value: '2' },
{ text: '禁行区', value: '3' }
],
formData: {
parkingName: '',
type: '',
error: '',
picture: '',
remark: '',
boundary: '',
boundaryStr: '',
longitude: '',
latitude: '',
},
mapCenter: {
latitude: 27.105722,
longitude: 120.25721
},
areaInfo: {},
polygons: [],
coordinates: [],
token: '', // token
upurl: '', //
}
},
computed: {
getTitle() {
const action = this.parkingId ? '修改' : '新增'
return `${action}${this.getTypeName(this.formData.type)}`
}
},
onLoad(e) {
this.parkingId = e.parkingId
this.formData.type = e.type
// token
this.getQiniuToken()
this.areaId = uni.getStorageSync('adminAreaid')
this.getArea()
if (this.parkingId) {
this.getParkingDetail()
}
},
methods: {
getArea() {
let id = this.areaId
this.$u.get("/app/area/" + id).then((res) => {
if (res.code == 200) {
this.areaInfo = res.data
} else {
uni.showToast({
title: res.msg,
icon: 'none',
duration: 2000
});
}
});
},
//
getTypeName(type) {
const typeMap = {
'1': '停车区',
'2': '禁停区',
'3': '禁行区'
}
return typeMap[type] || '请选择区域类型'
},
//
handleTypeChange(e) {
const index = e.detail.value
this.formData.type = this.typeList[index].value
},
//
getParkingDetail() {
this.$u.get(`/app/parking/${this.parkingId}`).then(res => {
if (res.code === 200) {
const data = res.data
this.formData = {
parkingName: data.parkingName,
type: data.type,
error: data.error,
picture: data.picture,
remark: data.remark,
boundary: data.boundary,
boundaryStr: data.boundaryStr,
longitude: data.longitude,
latitude: data.latitude
}
//
if (data.boundaryStr) {
this.coordinates = JSON.parse(data.boundaryStr)
this.updatePolygons()
}
//
if (data.latitude && data.longitude) {
this.mapCenter = {
latitude: Number(data.latitude),
longitude: Number(data.longitude)
}
}
}
})
},
//
chooseImage() {
uni.chooseImage({
count: 1,
sizeType: ['compressed'],
sourceType: ['album', 'camera'],
success: (res) => {
this.uploadImage(res.tempFilePaths[0])
}
})
},
// token
getQiniuToken() {
this.$u.get("/common/qiniu/uploadInfo").then((res) => {
if (res.code == 200) {
this.token = res.token
this.upurl = res.domain
}
});
},
//
uploadImage(filePath) {
uni.showLoading({
title: '上传中...'
})
const math = 'static/' + this.$u.guid(20)
uni.uploadFile({
url: 'https://up-z2.qiniup.com',
filePath: filePath,
name: 'file',
formData: {
token: this.token,
key: 'bike/img/' + math
},
success: (res) => {
const response = JSON.parse(res.data)
if (response.key) {
this.formData.picture = this.upurl + '/' + response.key
uni.showToast({
title: '上传成功',
icon: 'success'
})
} else {
uni.showToast({
title: '上传失败',
icon: 'none'
})
}
},
fail: () => {
uni.showToast({
title: '上传失败',
icon: 'none'
})
},
complete: () => {
uni.hideLoading()
}
})
},
//
previewImage() {
if (this.formData.picture) {
uni.previewImage({
urls: [this.formData.picture]
})
}
},
//
deletePic() {
this.formData.picture = ''
},
toSetmap() {
if (this.formData.longitude && this.formData.latitude) {
uni.navigateTo({
url: `/pages_adminSet/park_map?type=${this.formData.type}&boundaryStr=${encodeURIComponent(this.formData.boundaryStr || '')}&longitude=${this.formData.longitude}&latitude=${this.formData.latitude}`
})
} else {
uni.navigateTo({
url: `/pages_adminSet/park_map?type=${this.formData.type}&boundaryStr=${encodeURIComponent(this.formData.boundaryStr || '')}&longitude=${this.areaInfo.longitude}&latitude=${this.areaInfo.latitude}`
})
}
},
//
handleMapDataReturn(boundaryStr) {
this.formData.boundaryStr = boundaryStr
const coordinates = JSON.parse(boundaryStr)
// boundary
// this.formData.boundary = `POLYGON((${coordinates.map(coord => coord.join(' ')).join(',')}))`
//
// const center = this.calculateCenter(coordinates)
// this.formData.latitude = String(center.latitude)
// this.formData.longitude = String(center.longitude)
//
this.coordinates = coordinates
this.updatePolygons()
},
//
updatePolygons() {
if (this.coordinates.length >= 3) {
this.polygons = [{
points: this.coordinates.map(coord => ({
latitude: coord[1],
longitude: coord[0]
})),
strokeWidth: 2,
strokeColor: '#007AFF',
fillColor: '#007AFF20'
}]
//
this.formData.boundaryStr = JSON.stringify(this.coordinates)
this.formData.boundary = `POLYGON((${this.coordinates.map(coord => coord.join(' ')).join(',')}))`
//
const center = this.calculateCenter()
this.formData.latitude = String(center.latitude)
this.formData.longitude = String(center.longitude)
}
},
//
calculateCenter() {
const lats = this.coordinates.map(coord => coord[1])
const lngs = this.coordinates.map(coord => coord[0])
return {
latitude: (Math.max(...lats) + Math.min(...lats)) / 2,
longitude: (Math.max(...lngs) + Math.min(...lngs)) / 2
}
},
//
handleSubmit() {
if (!this.formData.parkingName) {
return uni.showToast({
title: '请输入区域名称',
icon: 'none'
})
}
if (!this.formData.type) {
return uni.showToast({
title: '请选择区域类型',
icon: 'none'
})
}
if (!this.formData.boundary) {
return uni.showToast({
title: '请在地图上绘制区域',
icon: 'none'
})
}
const data = {
...this.formData,
areaId: this.areaId
}
const request = this.parkingId ?
this.$u.put('/appVerify/parking', { ...data, parkingId: this.parkingId }) :
this.$u.post('/appVerify/parking', data)
request.then(res => {
if (res.code === 200) {
uni.showToast({
title: '保存成功',
icon: 'success'
})
setTimeout(() => {
this.goBack()
}, 1500)
} else {
uni.showToast({
title: res.msg || '保存失败',
icon: 'none'
})
}
})
},
//
goBack() {
uni.navigateBack()
}
}
}
</script>
<style lang="scss">
page {
width: 100%;
height: 100%;
background-color: #f5f5f5;
}
.page {
width: 100%;
height: 100%;
}
.content {
padding: 20rpx;
}
.form-item {
background-color: #fff;
padding: 20rpx;
margin-bottom: 20rpx;
border-radius: 8rpx;
.label {
font-size: 28rpx;
color: #333;
margin-bottom: 10rpx;
display: block;
}
}
.type-selector {
display: flex;
justify-content: space-between;
align-items: center;
padding: 20rpx;
background-color: #f8f8f8;
border-radius: 8rpx;
}
.image-upload {
position: relative;
display: inline-block;
.preview-image {
width: 200rpx;
height: 200rpx;
border-radius: 8rpx;
}
.upload-btn {
width: 200rpx;
height: 200rpx;
display: flex;
align-items: center;
justify-content: center;
background-color: #f8f8f8;
border-radius: 8rpx;
}
.delete-btn {
position: absolute;
top: -20rpx;
right: -20rpx;
width: 40rpx;
height: 40rpx;
background-color: rgba(0, 0, 0, 0.5);
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
}
}
.map-container {
width: 100%;
height: 500rpx;
map {
width: 100%;
height: 100%;
}
}
.map-tips {
font-size: 24rpx;
color: #999;
margin-top: 10rpx;
}
.clear-btn {
margin-top: 20rpx;
text-align: center;
color: #007AFF;
font-size: 28rpx;
}
.btn-group {
margin-top: 40rpx;
display: flex;
gap: 20rpx;
.u-button {
flex: 1;
}
}
</style>

View File

@ -0,0 +1,523 @@
<template>
<view class="map-container">
<u-notice-bar mode="horizontal" :list="list"></u-notice-bar>
<map id="map" class="map" :latitude="latitude" :longitude="longitude" :polygons="polygons" :markers="markers"
:show-location="true" @tap="handleMapTap" @markertap="handleMarkerTap" @markerdragend="handleMarkerDrag"
scale="14">
</map>
<!-- 定位按钮 -->
<cover-image class="location-btn" src="https://lxnapi.ccttiot.com/bike/img/static/uoxanRjBrBrtcYwRGXKa"
@tap="moveToLocation"></cover-image>
<!-- 提示文本 -->
<cover-view class="tip-text" v-if="isDrawing">
{{ editMode ? '点击锚点进行删除' : '点击地图添加锚点,拖动锚点可调整位置' }}
</cover-view>
<!-- 控制按钮组 -->
<cover-view class="controls">
<!-- 未开始绘制时显示开始按钮 -->
<cover-view class="control-btn" @tap="startDrawing" v-if="!isDrawing">开始绘制</cover-view>
<!-- 绘制过程中显示的按钮组 -->
<block v-else>
<cover-view class="control-btn-group">
<!-- 绘制锚点按钮 -->
<cover-view class="control-btn control-btn-small" :class="{ 'control-btn-active': !editMode }"
@tap="toggleEditMode('draw')">绘制锚点</cover-view>
<!-- 删除锚点按钮 -->
<cover-view class="control-btn control-btn-small" :class="{ 'control-btn-active': editMode }"
@tap="toggleEditMode('delete')">删除锚点</cover-view>
</cover-view>
<cover-view class="control-btn-group">
<!-- 完成按钮 -->
<cover-view class="control-btn" @tap="finishDrawing" :class="{ 'btn-disabled': points.length < 3 }">完成绘制
</cover-view>
<!-- 清除按钮 -->
<cover-view class="control-btn control-btn-danger" @tap="clearPolygon">清除</cover-view>
</cover-view>
</block>
</cover-view>
</view>
</template>
<script>
export default {
name: 'CustomMap',
props: {
areaInfo: {
type: Object,
default: () => ({})
}
},
data() {
return {
latitude: 39.909,
longitude: 116.397,
points: [],
polygons: [],
markers: [],
isDrawing: false,
editMode: false,
mapContext: null,
initialCenter: {
latitude: 39.909,
longitude: 116.397
},
list: [
'提示:点击"开始绘制"按钮开始绘制区域',
'点击地图添加锚点',
'点击锚点可删除',
'至少需要3个锚点才能形成有效区域'
],
// noticeText: '""3',
}
},
mounted() {
this.mapContext = uni.createMapContext('map', this)
},
watch: {
areaInfo: {
handler(newVal) {
if (newVal.boundary && newVal.latitude && newVal.longitude) {
this.initBoundary(newVal)
}
},
immediate: true
}
},
methods: {
//
initBoundary(info) {
this.initialCenter = {
latitude: info.latitude,
longitude: info.longitude
}
this.latitude = info.latitude
this.longitude = info.longitude
//
const points = info.boundaryStr.split('],[').map(pointStr => {
//
const cleanStr = pointStr.replace(/^\[|\]$/g, '')
//
const [longitude, latitude] = cleanStr.split(',')
return {
longitude: Number(longitude),
latitude: Number(latitude)
}
})
console.log('解析后的点:', points)
//
this.points = points.filter(point =>
!isNaN(point.latitude) && !isNaN(point.longitude)
)
this.updateMarkers()
this.updatePolygonWithNearestNeighbor()
},
//
updateMarkers() {
this.markers = this.points.map((point, index) => ({
id: index,
latitude: point.latitude,
longitude: point.longitude,
width: 20,
height: 20,
// callout: {
// // content: `${index + 1}`,
// color: '#ffffff',
// fontSize: 14,
// borderRadius: 4,
// bgColor: '#007AFF',
// padding: 4,
// display: 'ALWAYS'
// },
draggable: true
}))
},
//
async moveToLocation() {
try {
const [err, res] = await uni.getLocation({
type: 'gcj02'
});
if (err) {
throw new Error('获取位置失败');
}
this.latitude = res.latitude;
this.longitude = res.longitude;
await new Promise(resolve => setTimeout(resolve, 100));
this.mapContext.moveToLocation({
latitude: res.latitude,
longitude: res.longitude,
success: () => {
setTimeout(() => {
this.mapContext.getScale({
success: () => { }
});
}, 500);
}
});
} catch (error) {
uni.showToast({
title: '定位失败,请检查定位权限',
icon: 'none'
});
}
},
//
toggleEditMode(mode) {
this.editMode = mode === 'delete'
uni.showToast({
title: this.editMode ? '请点击锚点进行删除' : '请点击地图添加锚点',
icon: 'none'
})
},
//
startDrawing() {
//
const currentCenter = {
latitude: this.latitude,
longitude: this.longitude
}
// this.points = []
// this.markers = []
// this.polygons = []
this.isDrawing = true
this.editMode = false
//
this.$nextTick(() => {
this.latitude = currentCenter.latitude
this.longitude = currentCenter.longitude
})
uni.showToast({
title: '开始绘制区域',
icon: 'none'
})
},
//
isPointTooClose(point1, point2, threshold = 0.0001) {
const distance = Math.sqrt(
Math.pow(point1.latitude - point2.latitude, 2) +
Math.pow(point1.longitude - point2.longitude, 2)
)
return distance < threshold
},
//
cleanNearbyPoints() {
if (this.points.length < 2) return
let cleanedPoints = []
let pointsToKeep = new Set()
//
for (let i = 0; i < this.points.length; i++) {
let shouldKeep = true
for (let j = 0; j < i; j++) {
if (pointsToKeep.has(j) && this.isPointTooClose(this.points[i], this.points[j])) {
shouldKeep = false
break
}
}
if (shouldKeep) {
pointsToKeep.add(i)
}
}
//
pointsToKeep.forEach(index => {
cleanedPoints.push(this.points[index])
})
//
this.points = cleanedPoints
this.updateMarkers()
},
//
handleMapTap(e) {
if (!this.isDrawing || this.editMode) return
const { latitude, longitude } = e.detail
//
const newPoint = { latitude, longitude }
for (let point of this.points) {
if (this.isPointTooClose(newPoint, point)) {
uni.showToast({
title: '锚点位置过近',
icon: 'none'
})
return
}
}
//
this.points.push(newPoint)
this.updateMarkers()
//
this.updatePolygonWithNearestNeighbor()
},
//
handleMarkerTap(e) {
if (!this.editMode) return
const markerId = e.detail.markerId
//
this.points = this.points.filter((_, index) => index !== markerId)
this.updateMarkers()
//
this.updatePolygonWithNearestNeighbor()
},
//
handleMarkerDrag(e) {
const { markerId, latitude, longitude } = e.detail
//
this.points[markerId] = { latitude, longitude }
this.updateMarkers()
//
this.updatePolygonWithNearestNeighbor()
},
//
updatePolygonWithNearestNeighbor() {
if (this.points.length < 2) {
this.polygons = []
return
}
//
this.cleanNearbyPoints()
//
if (this.points.length < 3) {
this.polygons = [{
points: [...this.points],
strokeWidth: 2,
strokeColor: '#FF0000',
fillColor: '#FF000050',
zIndex: 1
}]
return
}
//
let centerLat = this.points.reduce((sum, p) => sum + p.latitude, 0) / this.points.length
let centerLng = this.points.reduce((sum, p) => sum + p.longitude, 0) / this.points.length
//
let sortedPoints = [...this.points].sort((a, b) => {
let angleA = Math.atan2(a.latitude - centerLat, a.longitude - centerLng)
let angleB = Math.atan2(b.latitude - centerLat, b.longitude - centerLng)
return angleA - angleB
})
//
sortedPoints.push(sortedPoints[0])
//
this.polygons = [{
points: sortedPoints,
strokeWidth: 2,
strokeColor: '#FF0000',
fillColor: '#FF000050',
zIndex: 1
}]
},
//
finishDrawing() {
if (this.points.length < 3) {
uni.showToast({
title: '至少需要3个锚点才能形成区域',
icon: 'none'
})
return
}
//
this.cleanNearbyPoints()
//
const centerLat = this.points.reduce((sum, p) => sum + Number(p.latitude), 0) / this.points.length
const centerLng = this.points.reduce((sum, p) => sum + Number(p.longitude), 0) / this.points.length
//
let sortedPoints = [...this.points].sort((a, b) => {
let angleA = Math.atan2(a.latitude - centerLat, a.longitude - centerLng)
let angleB = Math.atan2(b.latitude - centerLat, b.longitude - centerLng)
return angleA - angleB
})
// 5
const formattedPoints = sortedPoints.map(point => ({
latitude: Number(Number(point.latitude).toFixed(5)),
longitude: Number(Number(point.longitude).toFixed(5))
}))
//
this.isDrawing = false
this.editMode = false
//
this.polygons = [{
points: formattedPoints,
strokeWidth: 2,
strokeColor: '#FF0000',
fillColor: '#FF000050',
zIndex: 1
}]
//
this.$emit('on-polygon-complete', {
points: formattedPoints,
polygon: this.polygons[0]
})
},
//
clearPolygon() {
const currentCenter = {
latitude: this.latitude,
longitude: this.longitude
}
this.points = []
this.markers = []
this.polygons = []
this.isDrawing = false
this.editMode = false
this.$nextTick(() => {
this.latitude = currentCenter.latitude
this.longitude = currentCenter.longitude
})
uni.showToast({
title: '已清除绘制',
icon: 'none'
})
}
}
}
</script>
<style>
.map-container {
width: 100%;
height: 100vh;
position: relative;
}
.map {
width: 100%;
height: 100%;
}
.location-btn {
position: fixed;
right: 30rpx;
bottom: 300rpx;
width: 80rpx;
height: 80rpx;
z-index: 100;
}
.tip-text {
position: fixed;
top: 40rpx;
left: 50%;
transform: translateX(-50%);
background-color: rgba(0, 0, 0, 0.7);
color: #ffffff;
padding: 16rpx 30rpx;
border-radius: 30rpx;
font-size: 28rpx;
z-index: 100;
}
.control-btn-group {
display: flex;
gap: 20rpx;
margin-bottom: 20rpx;
}
.control-btn-small {
min-width: 120rpx;
height: 60rpx;
line-height: 60rpx;
font-size: 24rpx;
padding: 0 20rpx;
}
.controls {
position: fixed;
bottom: 60rpx;
left: 0;
right: 0;
display: flex;
flex-direction: column;
align-items: center;
padding: 0 30rpx;
z-index: 100;
}
.control-btn {
pointer-events: auto;
min-width: 200rpx;
height: 80rpx;
line-height: 80rpx;
background-color: #ffffff;
border-radius: 10rpx;
font-size: 28rpx;
text-align: center;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.1);
margin: 0 10rpx;
padding: 0 40rpx;
}
.control-btn-danger {
background-color: #ff4444;
color: #ffffff;
}
.control-btn-active {
background-color: #007AFF;
color: #ffffff;
}
.btn-disabled {
opacity: 0.5;
background-color: #cccccc;
pointer-events: none;
}
</style>

View File

@ -0,0 +1,394 @@
<template>
<view class="map-container">
<u-notice-bar mode="horizontal" :list="tipsList"></u-notice-bar>
<map
id="editMap"
class="map"
:latitude="latitude"
:longitude="longitude"
:polygons="polygons"
:markers="markers"
:show-location="true"
@tap="handleMapTap"
@markertap="handleMarkerTap"
@markerdragend="handleMarkerDrag"
scale="19"
></map>
<!-- 定位按钮 -->
<cover-image
class="location-btn"
src="https://lxnapi.ccttiot.com/bike/img/static/uoxanRjBrBrtcYwRGXKa"
@tap="moveToLocation"
></cover-image>
<!-- 提示文本 -->
<cover-view class="tip-text" v-if="isDrawing">
{{ editMode ? '点击锚点进行删除' : '点击地图添加锚点,拖动锚点可调整位置' }}
</cover-view>
<!-- 控制按钮组 -->
<cover-view class="controls">
<!-- 未开始绘制时显示开始按钮 -->
<cover-view class="control-btn" @tap="startDrawing" v-if="!isDrawing">
开始绘制
</cover-view>
<!-- 绘制过程中显示的按钮组 -->
<block v-else>
<cover-view class="control-btn-group">
<cover-view
class="control-btn"
:class="{'control-btn-active': !editMode}"
@tap="toggleEditMode('draw')"
>
绘制锚点
</cover-view>
<cover-view
class="control-btn"
:class="{'control-btn-active': editMode}"
@tap="toggleEditMode('delete')"
>
删除锚点
</cover-view>
</cover-view>
<cover-view class="control-btn-group">
<cover-view
class="control-btn"
@tap="finishDrawing"
:class="{'btn-disabled': points.length < 3}"
>
完成绘制
</cover-view>
<cover-view
class="control-btn control-btn-danger"
@tap="clearPolygon"
>
清除
</cover-view>
</cover-view>
</block>
</cover-view>
</view>
</template>
<script>
export default {
name: 'EditMap',
props: {
boundaryStr: {
type: String,
default: ''
},
longitude: {
type: Number,
default: 0
},
latitude: {
type: Number,
default: 0
}
},
data() {
return {
// latitude: parseFloat(this.latitude),
// longitude: parseFloat(this.longitude),
points: [],
polygons: [],
markers: [],
isDrawing: false,
editMode: false,
mapContext: null,
tipsList: [
'提示:点击"开始绘制"按钮开始绘制区域',
'点击地图添加锚点',
'点击锚点可删除',
'至少需要3个锚点才能形成有效区域'
]
}
},
mounted() {
this.mapContext = uni.createMapContext('editMap', this)
this.initBoundary()
},
watch: {
boundaryStr: {
handler() {
this.initBoundary()
},
immediate: true
}
},
methods: {
//
initBoundary() {
if (!this.boundaryStr) return
try {
const points = JSON.parse(this.boundaryStr)
if (Array.isArray(points) && points.length > 0) {
this.points = points.map(point => ({
longitude: Number(point[0]),
latitude: Number(point[1])
})).filter(point =>
!isNaN(point.latitude) && !isNaN(point.longitude)
)
//
this.updateMarkers()
this.updatePolygonWithNearestNeighbor()
}
} catch (error) {
console.error('解析边界数据失败:', error)
}
},
//
updateMarkers() {
this.markers = this.points.map((point, index) => ({
id: index,
latitude: point.latitude,
longitude: point.longitude,
width: 20,
height: 20,
draggable: true
}))
},
//
async moveToLocation() {
try {
const [err, res] = await uni.getLocation({ type: 'gcj02' })
if (err) throw new Error('获取位置失败')
this.latitude = res.latitude
this.longitude = res.longitude
await new Promise(resolve => setTimeout(resolve, 100))
this.mapContext.moveToLocation({
latitude: res.latitude,
longitude: res.longitude
})
} catch (error) {
uni.showToast({
title: '定位失败,请检查定位权限',
icon: 'none'
})
}
},
//
startDrawing() {
this.isDrawing = true
this.editMode = false
if (this.points.length === 0) {
uni.showToast({
title: '请点击地图添加锚点',
icon: 'none'
})
}
},
//
toggleEditMode(mode) {
this.editMode = mode === 'delete'
uni.showToast({
title: this.editMode ? '请点击锚点进行删除' : '请点击地图添加锚点',
icon: 'none'
})
},
//
handleMapTap(e) {
if (!this.isDrawing || this.editMode) return
const { latitude, longitude } = e.detail
this.points.push({ latitude, longitude })
this.updateMarkers()
this.updatePolygonWithNearestNeighbor()
},
//
handleMarkerTap(e) {
if (!this.editMode) return
const markerId = e.detail.markerId
this.points = this.points.filter((_, index) => index !== markerId)
this.updateMarkers()
this.updatePolygonWithNearestNeighbor()
},
//
handleMarkerDrag(e) {
const { markerId, latitude, longitude } = e.detail
this.points[markerId] = { latitude, longitude }
this.updateMarkers()
this.updatePolygonWithNearestNeighbor()
},
//
updatePolygonWithNearestNeighbor() {
if (this.points.length < 3) {
this.polygons = []
return
}
//
const centerLat = this.points.reduce((sum, p) => sum + p.latitude, 0) / this.points.length
const centerLng = this.points.reduce((sum, p) => sum + p.longitude, 0) / this.points.length
//
let sortedPoints = [...this.points].sort((a, b) => {
let angleA = Math.atan2(a.latitude - centerLat, a.longitude - centerLng)
let angleB = Math.atan2(b.latitude - centerLat, b.longitude - centerLng)
return angleA - angleB
})
//
sortedPoints.push(sortedPoints[0])
this.polygons = [{
points: sortedPoints,
strokeWidth: 2,
strokeColor: '#FF0000',
fillColor: '#FF000050',
zIndex: 1
}]
},
//
finishDrawing() {
if (this.points.length < 3) {
uni.showToast({
title: '至少需要3个锚点才能形成区域',
icon: 'none'
})
return
}
//
const centerLat = this.points.reduce((sum, p) => sum + p.latitude, 0) / this.points.length
const centerLng = this.points.reduce((sum, p) => sum + p.longitude, 0) / this.points.length
//
let sortedPoints = [...this.points].sort((a, b) => {
let angleA = Math.atan2(a.latitude - centerLat, a.longitude - centerLng)
let angleB = Math.atan2(b.latitude - centerLat, b.longitude - centerLng)
return angleA - angleB
})
//
const coordinates = sortedPoints.map(point => [
point.longitude,
point.latitude
])
this.$emit('on-complete', JSON.stringify(coordinates))
//
this.isDrawing = false
this.editMode = false
},
//
clearPolygon() {
this.points = []
this.markers = []
this.polygons = []
this.isDrawing = false
this.editMode = false
uni.showToast({
title: '已清除绘制',
icon: 'none'
})
}
}
}
</script>
<style lang="scss">
.map-container {
width: 100%;
height: 100vh;
position: relative;
}
.map {
width: 100%;
height: 100%;
}
.location-btn {
position: fixed;
right: 30rpx;
bottom: 300rpx;
width: 80rpx;
height: 80rpx;
z-index: 100;
}
.tip-text {
position: fixed;
top: 40rpx;
left: 50%;
transform: translateX(-50%);
background-color: rgba(0, 0, 0, 0.7);
color: #ffffff;
padding: 16rpx 30rpx;
border-radius: 30rpx;
font-size: 28rpx;
z-index: 100;
}
.controls {
position: fixed;
bottom: 60rpx;
left: 0;
right: 0;
display: flex;
flex-direction: column;
align-items: center;
padding: 0 30rpx;
z-index: 100;
}
.control-btn-group {
display: flex;
gap: 20rpx;
margin-bottom: 20rpx;
}
.control-btn {
min-width: 200rpx;
height: 80rpx;
line-height: 80rpx;
background-color: #ffffff;
border-radius: 10rpx;
font-size: 28rpx;
text-align: center;
box-shadow: 0 2rpx 10rpx rgba(0, 0, 0, 0.1);
margin: 0 10rpx;
}
.control-btn-active {
background-color: #007AFF;
color: #ffffff;
}
.control-btn-danger {
background-color: #ff4444;
color: #ffffff;
}
.btn-disabled {
opacity: 0.5;
background-color: #cccccc;
pointer-events: none;
}
</style>

View File

@ -0,0 +1,97 @@
<template>
<view class="page">
<u-navbar title="停车区列表" :border-bottom="false" :background="bgc" title-color='#000' title-size='36'
height='45'></u-navbar>
<custom-map :areaInfo="areaInfo" @on-polygon-complete="handlePolygonComplete"></custom-map>
</view>
</template>
<script>
import CustomMap from './components/CustomMap.vue'
export default {
components: {
CustomMap
},
data() {
return {
bgc: {
backgroundColor: "#fff",
},
areaId: '',
areaInfo: {}
}
},
onLoad(e) {
if (e.areaId) {
this.areaId = e.areaId
this.getArea()
}
},
methods: {
getArea() {
this.$u.get(`/app/area/` + this.areaId).then((res) => {
if (res.code == 200) {
this.areaInfo = res.data
} else {
uni.showToast({
title: res.msg || '获取区域信息失败',
icon: 'none'
})
}
}).catch(error => {
uni.showToast({
title: '获取区域信息失败',
icon: 'none'
})
});
},
handlePolygonComplete(data) {
//
const boundaryStr = data.points.map(point =>
`[${point.longitude},${point.latitude}]`
).join(',')
//
const params = {
...this.areaInfo,
boundaryStr: `[${boundaryStr}]` // 使 boundaryStr
}
//
this.$u.put('/appVerify/areaEdit', params).then(res => {
if (res.code === 200) {
uni.showToast({
title: '保存成功',
icon: 'success'
})
setTimeout(() => {
uni.navigateBack()
}, 1500)
} else {
uni.showToast({
title: res.msg || '保存失败',
icon: 'none'
})
}
}).catch(() => {
uni.showToast({
title: '保存失败',
icon: 'none'
})
})
}
}
}
</script>
<style lang="scss">
page {
width: 100%;
height: 100%;
}
.page {
width: 100%;
height: 100%;
}
</style>

118
pages_adminSet/park_map.vue Normal file
View File

@ -0,0 +1,118 @@
<template>
<view class="page">
<u-navbar :title="getTitle" :border-bottom="false" :background="bgc" title-color='#000' title-size='36' height='45'>
</u-navbar>
<edit-map
:boundary-str="boundaryStr"
:longitude="longitude"
:latitude="latitude"
@on-complete="handlePolygonComplete"
></edit-map>
<view class="bottom-btn">
<u-button type="primary" @click="confirmEdit">确认修改</u-button>
</view>
</view>
</template>
<script>
import EditMap from './components/EditMap.vue'
export default {
components: {
EditMap
},
data() {
return {
bgc: {
backgroundColor: "#fff",
},
type: '',
boundaryStr: '',
newBoundaryStr: null,
longitude: '',
latitude: ''
}
},
computed: {
getTitle() {
const typeMap = {
'1': '停车区',
'2': '禁停区',
'3': '禁行区'
}
return `编辑${typeMap[this.type] || '区域'}坐标`
}
},
onLoad(e) {
this.type = e.type
//
if (e.boundaryStr) {
try {
this.boundaryStr = decodeURIComponent(e.boundaryStr)
} catch (error) {
console.error('解析边界数据失败:', error)
}
}
this.longitude = parseFloat(e.longitude)
this.latitude = parseFloat(e.latitude)
},
methods: {
//
handlePolygonComplete(boundaryStr) {
this.newBoundaryStr = boundaryStr
//
this.confirmEdit()
},
//
confirmEdit() {
if (!this.newBoundaryStr) {
return uni.showToast({
title: '请完成区域绘制',
icon: 'none'
})
}
//
const pages = getCurrentPages()
const prevPage = pages[pages.length - 2]
if (prevPage && prevPage.$vm) {
prevPage.$vm.handleMapDataReturn(this.newBoundaryStr)
//
setTimeout(() => {
uni.navigateBack()
}, 100)
} else {
uni.navigateBack()
}
}
}
}
</script>
<style lang="scss">
page {
width: 100%;
height: 100%;
}
.page {
width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.bottom-btn {
padding: 20rpx;
background-color: #fff;
.u-button {
width: 100%;
}
}
</style>

View File

@ -297,6 +297,22 @@
设置车辆电量低于多少时自动生成的换电维护工单
</view>
</view>
<view class="card_li" @click="toMap()">
<view class="tops">
<view class="card_left">
电子围栏设置
</view>
<view class="card_right">
<view class="iconfont icon-xiangyou1">
</view>
</view>
</view>
<view class="tips" style="text-align: left;">
设置车辆可以正常行驶的区域
</view>
</view>
</view>
@ -343,6 +359,11 @@
},
methods: {
toMap(){
uni.navigateTo({
url:'/pages_adminSet/map_set?areaId='+this.areaId
})
},
addService() {
if (this.customServices.length < 3) {
this.customServices.push({