1634 lines
57 KiB
Vue
1634 lines
57 KiB
Vue
|
<template>
|
|||
|
<view>
|
|||
|
<view class="bottom">
|
|||
|
<div class="info" v-if="showInfo">
|
|||
|
<view class="flex-col section_2">
|
|||
|
<view class="justify-between position-relative">
|
|||
|
<view class="flex-row group_5">
|
|||
|
<image @click="toDetil(detailInfo.userId)" :src="$u.http.config.imgUrl + detailInfo.imageUrl"
|
|||
|
class="image_5"/>
|
|||
|
<view class="name-work">
|
|||
|
<view class="flex-row items-center h100">
|
|||
|
<text class="text text_1">{{ detailInfo.name }}</text>
|
|||
|
<text class="text text_3" v-if="careerList[detailInfo.workFieldName]">
|
|||
|
<img :src="careerList[detailInfo.workFieldName].icon"
|
|||
|
alt="">{{ detailInfo.workLableName || detailInfo.workFieldName }}
|
|||
|
</text>
|
|||
|
<text class="text text_3" v-else>
|
|||
|
{{ detailInfo.workLableName }}
|
|||
|
</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view class="flex-row section_3">
|
|||
|
<image src="/static/common/img/chats.png" class="image_6"/>
|
|||
|
<text class="text text_2" @click="GoChat(detailInfo.userId)">打招呼</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view class="flex-row tags">
|
|||
|
<u-tag :text="detailInfo.college" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
<u-tag :text="detailInfo.major" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
<u-tag :text="detailInfo.startYear" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
</view>
|
|||
|
<view>
|
|||
|
<text v-if="detailInfo.aiBlurb" class="text AIdescribe">{{ detailInfo.aiBlurb }}</text>
|
|||
|
</view>
|
|||
|
<view class="flex-row group_6">
|
|||
|
<image src="/static/common/img/location.svg" style="width: 32rpx;height: 32rpx;"></image>
|
|||
|
<text class="text text_5">{{
|
|||
|
detailInfo.province +
|
|||
|
" " +
|
|||
|
detailInfo.city +
|
|||
|
" 距你 " +
|
|||
|
(detailInfo.distance > 1000
|
|||
|
? (detailInfo.distance / 1000).toFixed(1) + "KM"
|
|||
|
: detailInfo.distance + "M")
|
|||
|
}}
|
|||
|
</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</div>
|
|||
|
</view>
|
|||
|
<view class="search-box" @click="toSearch">
|
|||
|
|
|||
|
<image
|
|||
|
src="/static/common/img/search.svg"
|
|||
|
style="width: 40rpx; height: 40rpx"
|
|||
|
></image>
|
|||
|
</view>
|
|||
|
<!-- :scale="map.scale" 缩放max:18 min:5-->
|
|||
|
<map
|
|||
|
id="map"
|
|||
|
ref="map"
|
|||
|
:style="'width: 100vw; height: calc(100vh - .5rem );'"
|
|||
|
:scale="mapScale"
|
|||
|
:layer-style="'style1'"
|
|||
|
@regionchange="regionchange"
|
|||
|
@updated="updated"
|
|||
|
:latitude="map.latitude"
|
|||
|
:longitude="map.longitude"
|
|||
|
:markers="covers"
|
|||
|
@callouttap="de"
|
|||
|
@markertap="getUserDetail"
|
|||
|
@click="tap"
|
|||
|
>
|
|||
|
</map>
|
|||
|
<!-- 回到本人定位位置 -->
|
|||
|
<view class="back-to-location" @click="backToLocation"
|
|||
|
v-if="latitude!=vuex_userLocation.latitude||longitude!=vuex_userLocation.longitude">
|
|||
|
<image src="/static/common/img/back-to-location.png" style="width: 120rpx; height: 120rpx"></image>
|
|||
|
</view>
|
|||
|
<u-tabbar
|
|||
|
:list="vuex_tabbar"
|
|||
|
:class="{ phone: vuex_iPhone }"
|
|||
|
:active-color="vuex_tabbar_config.activeColor"
|
|||
|
:inactive-color="vuex_tabbar_config.inactiveColor"
|
|||
|
></u-tabbar>
|
|||
|
<u-toast ref="uToast"/>
|
|||
|
<!-- 弹出列表 -->
|
|||
|
<view class="popup-box" v-if="showPopup" :style="{
|
|||
|
maxHeight: isExpanded,height: isExpanded,
|
|||
|
}">
|
|||
|
<view class="popup-top"
|
|||
|
@click.stop="togglePopup"
|
|||
|
@touchstart.stop.prevent="touchStart"
|
|||
|
@touchmove.stop.prevent="touchMoveFn"
|
|||
|
@touchend.stop.prevent="touchEnd">
|
|||
|
<view class="pop-top"></view>
|
|||
|
<view class="pop-title">附近校友</view>
|
|||
|
</view>
|
|||
|
<view class="content">
|
|||
|
<template v-for="(item, index) in detailInfoList">
|
|||
|
<view
|
|||
|
v-if="item.userId == 0"
|
|||
|
class="info"
|
|||
|
style="
|
|||
|
position: initial;
|
|||
|
transform: initial;
|
|||
|
margin-bottom: 0.1rem;
|
|||
|
"
|
|||
|
:key="index"
|
|||
|
>
|
|||
|
<view class="flex-col section_2">
|
|||
|
<view class="justify-between position-relative">
|
|||
|
<view class="flex-row group_5">
|
|||
|
<image
|
|||
|
:src="$u.http.config.imgUrl + item.imageUrl"
|
|||
|
class="image_5"
|
|||
|
/>
|
|||
|
<view class="name-work">
|
|||
|
<view class="flex-row items-center h100">
|
|||
|
<text class="text text_1">{{ item.name }}</text>
|
|||
|
<text class="text text_3" v-if="careerList[item.workFieldName]">
|
|||
|
<img :src="careerList[item.workFieldName].icon"
|
|||
|
alt="">{{ item.workLableName || item.workFieldName }}
|
|||
|
</text>
|
|||
|
<text class="text text_3" v-else>
|
|||
|
{{ item.workFieldName }}
|
|||
|
</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view v-if="item.college||item.major||item.startYear" class="flex-row tags">
|
|||
|
<u-tag :text="item.college" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
<u-tag :text="item.major" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
<u-tag :text="item.startYear" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
</view>
|
|||
|
<view>
|
|||
|
<text v-if="item.aiBlurb" class="text AIdescribe">{{ item.aiBlurb }}</text>
|
|||
|
</view>
|
|||
|
<view class="flex-row group_6">
|
|||
|
<image
|
|||
|
src="/static/common/img/location.svg"
|
|||
|
style="width: 32rpx; height: 32rpx"
|
|||
|
></image>
|
|||
|
<text class="text text_5"
|
|||
|
>{{
|
|||
|
item.province +
|
|||
|
" " +
|
|||
|
item.city +
|
|||
|
" 距你 " +
|
|||
|
(item.distance > 1000
|
|||
|
? (item.distance / 1000).toFixed(1) + "KM"
|
|||
|
: item.distance + "M")
|
|||
|
}}
|
|||
|
</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view
|
|||
|
v-else
|
|||
|
class="info"
|
|||
|
style="
|
|||
|
position: initial;
|
|||
|
transform: initial;
|
|||
|
margin-bottom: 0.1rem;
|
|||
|
"
|
|||
|
:key="index"
|
|||
|
>
|
|||
|
<view class="flex-col section_2">
|
|||
|
<view class="justify-between position-relative">
|
|||
|
<view class="flex-row group_5">
|
|||
|
<u-avatar
|
|||
|
@click="toDetil(item.userId)"
|
|||
|
size="0.5rem"
|
|||
|
:src="$u.http.config.imgUrl + item.imageUrl"
|
|||
|
class="image_5"
|
|||
|
></u-avatar>
|
|||
|
<view class="name-work">
|
|||
|
<view class="flex-row items-center h100">
|
|||
|
<text class="text text_1">{{ item.name }}</text>
|
|||
|
<text class="text text_3" v-if="careerList[item.workFieldName]">
|
|||
|
<img :src="careerList[item.workFieldName].icon"
|
|||
|
alt="">{{ item.workLableName || item.workFieldName }}
|
|||
|
</text>
|
|||
|
<text class="text text_3" v-else>
|
|||
|
{{ item.workFieldName }}
|
|||
|
</text>
|
|||
|
</view>
|
|||
|
<view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view @click="GoChat(item.userId)" class="flex-row section_3">
|
|||
|
<image src="/static/common/img/chats.png" class="image_6"/>
|
|||
|
<text class="text text_2">打招呼</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
<view class="flex-row tags">
|
|||
|
<u-tag :text="item.college" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
<u-tag :text="item.major" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
<u-tag :text="item.startYear" bg-color="#F6F8F9" color="#000000" mode="dark" size="mini"
|
|||
|
type="info"></u-tag>
|
|||
|
</view>
|
|||
|
<view>
|
|||
|
<text v-if="item.aiBlurb" class="text AIdescribe">{{ item.aiBlurb }}</text>
|
|||
|
</view>
|
|||
|
<view class="flex-row group_6">
|
|||
|
<image
|
|||
|
src="/static/common/img/location.svg"
|
|||
|
style="width: 32rpx; height: 32rpx"
|
|||
|
></image>
|
|||
|
<text class="text text_5"
|
|||
|
>{{
|
|||
|
item.province +
|
|||
|
" " +
|
|||
|
item.city +
|
|||
|
" 距你 " +
|
|||
|
(item.distance > 1000
|
|||
|
? (item.distance / 1000).toFixed(1) + "KM"
|
|||
|
: item.distance + "M")
|
|||
|
}}
|
|||
|
</text>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
// var jweixin = require('./../../../static/common/js/jweixin.js')
|
|||
|
var jweixin = require("jweixin-module");
|
|||
|
var map;
|
|||
|
var qqmapsdk;
|
|||
|
export default {
|
|||
|
data() {
|
|||
|
return {
|
|||
|
latitude: 0, //纬度
|
|||
|
longitude: 0,
|
|||
|
mapScale: 16,
|
|||
|
showInfo: false,
|
|||
|
height: "calc(100vh - .5rem )",
|
|||
|
width: "100vw",
|
|||
|
// showList: true,
|
|||
|
// viewHeight: 0,
|
|||
|
map: {
|
|||
|
scale: 14,
|
|||
|
latitude: 39.909,
|
|||
|
longitude: 116.39742,
|
|||
|
},
|
|||
|
// 标记点
|
|||
|
covers: [],
|
|||
|
pointDataList: "",
|
|||
|
detailInfoList: [],
|
|||
|
// 中心点
|
|||
|
center: {
|
|||
|
userId: 0,
|
|||
|
latitude: 39.909, //纬度
|
|||
|
longitude: 116.39742, //经度
|
|||
|
iconPath: "", //显示的图标
|
|||
|
rotate: 0, // 旋转度数
|
|||
|
width: 50, //宽
|
|||
|
imageUrl: "",
|
|||
|
height: 50, //高
|
|||
|
name: "我",
|
|||
|
province: "",
|
|||
|
city: "",
|
|||
|
distance: "0",
|
|||
|
workFieldName: "",
|
|||
|
title: "我的位置", //标注点名
|
|||
|
alpha: 0.8, //透明度
|
|||
|
},
|
|||
|
|
|||
|
callout: {
|
|||
|
content: '',//String 文本
|
|||
|
color: '',//String 文本颜色
|
|||
|
fontSize: '',// Number 文字大小
|
|||
|
x: 3,// label的坐标,原点是 marker 对应的经纬度
|
|||
|
y: -20,// label的坐标,原点是 marker 对应的经纬度
|
|||
|
borderWidth: 0,//Number边框宽度
|
|||
|
borderColor: '',//String 边框颜色
|
|||
|
borderRadius: 0,//Number callout边框圆角
|
|||
|
bgColor: '',//String 背景色
|
|||
|
padding: '',//Number 文本边缘留白
|
|||
|
display: 'ALWAYS',//String 'BYCLICK':点击显示; 'ALWAYS':常显
|
|||
|
},
|
|||
|
careerList: {
|
|||
|
'生物/制药/化工/医疗': {
|
|||
|
icon: '/static/common/icon/icon-1.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-1.png',
|
|||
|
},
|
|||
|
'政府机构/翻译/其他': {
|
|||
|
icon: '/static/common/icon/icon-2.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-2.png',
|
|||
|
},
|
|||
|
'人事/行政/高级管理': {
|
|||
|
icon: '/static/common/icon/icon-3.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-3.png',
|
|||
|
},
|
|||
|
'设计/市场/媒体/广告': {
|
|||
|
icon: '/static/common/icon/icon-4.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-4.png',
|
|||
|
},
|
|||
|
'闽建筑/房地产': {
|
|||
|
icon: '/static/common/icon/icon-5.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-5.png',
|
|||
|
},
|
|||
|
'计算机/互联网/通信/电子': {
|
|||
|
icon: '/static/common/icon/icon-6.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-6.png',
|
|||
|
},
|
|||
|
'服务业': {
|
|||
|
icon: '/static/common/icon/icon-7.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-7.png',
|
|||
|
},
|
|||
|
'教育/培训': {
|
|||
|
icon: '/static/common/icon/icon-8.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-8.png',
|
|||
|
},
|
|||
|
'销售/客服': {
|
|||
|
icon: '/static/common/icon/icon-9.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-9.png',
|
|||
|
},
|
|||
|
'资讯/法律/科研': {
|
|||
|
icon: '/static/common/icon/icon-10.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-10.png',
|
|||
|
},
|
|||
|
'会计/金融/银行/保险': {
|
|||
|
icon: '/static/common/icon/icon-11.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-11.png',
|
|||
|
},
|
|||
|
'生产/营运/采购/物流': {
|
|||
|
icon: '/static/common/icon/icon-12.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-12.png',
|
|||
|
},
|
|||
|
'学生': {
|
|||
|
icon: '/static/common/icon/icon-13.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-13.png',
|
|||
|
},
|
|||
|
'教师': {
|
|||
|
icon: '/static/common/icon/icon-14.png',
|
|||
|
minIcon: '/static/common/icon/min-icon-14.png',
|
|||
|
},
|
|||
|
},
|
|||
|
list: [],
|
|||
|
load: true,
|
|||
|
detailInfo: "",
|
|||
|
Showip: "",
|
|||
|
isExpanded: '0vh',
|
|||
|
touchStartY: 0,
|
|||
|
startScrollTop: 0,
|
|||
|
showPopup: false,
|
|||
|
timeer: null
|
|||
|
};
|
|||
|
},
|
|||
|
onLoad(e) {
|
|||
|
// if(!this.vuex_user.isCard){
|
|||
|
// uni.navigateTo({
|
|||
|
// url: '/pages/Face/index/index'
|
|||
|
// });
|
|||
|
// return
|
|||
|
// }
|
|||
|
// if (!this.vuex_user.isFill) {
|
|||
|
// uni.navigateTo({
|
|||
|
// url: "/pages/login/perfect/perfect",
|
|||
|
// });
|
|||
|
// return
|
|||
|
// } else {
|
|||
|
// this.$u.api.getUser()
|
|||
|
// }
|
|||
|
|
|||
|
console.info("🚀 ~ file:index method:onLoad line:374 -----", this.vuex_user)
|
|||
|
|
|||
|
this.center.imageUrl = this.vuex_user.head;
|
|||
|
this.center.workField = this.vuex_user?.workField?.workFieldName || "暂无";
|
|||
|
this.center.workFieldName = this.vuex_user?.workFieldName || "暂无";
|
|||
|
this.center.aiBlurb = this.vuex_user?.aiBlurb || "暂无";
|
|||
|
this.center.aiLabel = this.vuex_user?.aiLabel || "";
|
|||
|
this.center.college = this.vuex_user?.college || "";
|
|||
|
this.center.major = this.vuex_user?.major || "";
|
|||
|
this.center.startYear = this.vuex_user?.startYear || "";
|
|||
|
|
|||
|
// 创建地图上下文
|
|||
|
map = uni.createMapContext("map", this);
|
|||
|
uni.getSystemInfo({
|
|||
|
success: (res) => {
|
|||
|
this.height = res.windowHeight + "px";
|
|||
|
this.width = res.windowWidth + "px";
|
|||
|
},
|
|||
|
});
|
|||
|
// this.location();
|
|||
|
// qqmapsdk = new QQMapWX({
|
|||
|
// key: '2OLBZ-OOSRQ-RYZ5A-GMCM2-DJ43O-3QFLS', //开发者密钥
|
|||
|
// })
|
|||
|
// console.log(this.vuex_user.isFill)
|
|||
|
},
|
|||
|
onShow() {
|
|||
|
/* uni.getSystemInfo({
|
|||
|
success: (res) => {
|
|||
|
this.viewHeight = res.windowHeight;
|
|||
|
},
|
|||
|
}); */
|
|||
|
|
|||
|
this.map.latitude = this.vuex_userLocation.latitude;
|
|||
|
this.map.longitude = this.vuex_userLocation.longitude;
|
|||
|
this.center.latitude = this.vuex_userLocation.latitude;
|
|||
|
this.center.longitude = this.vuex_userLocation.longitude;
|
|||
|
setTimeout(() => {
|
|||
|
if (!this.map.latitude || !this.map.longitude) {
|
|||
|
this.location(true);
|
|||
|
}
|
|||
|
this.test();
|
|||
|
}, 500);
|
|||
|
if (!this.vuex_user.isFill) {
|
|||
|
uni.navigateTo({
|
|||
|
url: "/pages/login/perfect/perfect",
|
|||
|
});
|
|||
|
return;
|
|||
|
} else {
|
|||
|
this.$u.api.getUser();
|
|||
|
}
|
|||
|
},
|
|||
|
methods: {
|
|||
|
// 回到当前位置
|
|||
|
backToLocation() {
|
|||
|
this.location(true);
|
|||
|
map.moveToLocation({
|
|||
|
latitude: this.vuex_userLocation.latitude || this.center.latitude,
|
|||
|
longitude: this.vuex_userLocation.longitude || this.center.longitude,
|
|||
|
});
|
|||
|
|
|||
|
this.latitude = this.vuex_userLocation.latitude || this.center.latitude;
|
|||
|
this.longitude = this.vuex_userLocation.longitude || this.center.longitude;
|
|||
|
},
|
|||
|
// 查看详情
|
|||
|
toDetil(id) {
|
|||
|
this.$u.route({
|
|||
|
url: "/pages/AlumniCircle/userDetail/userDetail?id=" + id,
|
|||
|
});
|
|||
|
},
|
|||
|
// 去聊天
|
|||
|
GoChat(id) {
|
|||
|
this.showInfo = false;
|
|||
|
uni.navigateTo({
|
|||
|
url:
|
|||
|
"../../message/dialogBox/dialogBox?id=" + id + "&chatType=0&type=0",
|
|||
|
});
|
|||
|
},
|
|||
|
toSearch() {
|
|||
|
if (this.vuex_user.isAttestationXY || true) {
|
|||
|
this.$u.route({
|
|||
|
url: "/pages/main/search/search",
|
|||
|
params: {
|
|||
|
type: 'index',
|
|||
|
},
|
|||
|
});
|
|||
|
} else {
|
|||
|
this.$refs.uToast.show({
|
|||
|
title: "认证后才能进行搜索哦",
|
|||
|
url: "/pages/my/ShoolList/ShoolList",
|
|||
|
});
|
|||
|
}
|
|||
|
},
|
|||
|
test() {
|
|||
|
map.getRegion({
|
|||
|
success: (res) => {
|
|||
|
this.Showip = {
|
|||
|
northeast: res.northeast,
|
|||
|
southwest: res.southwest,
|
|||
|
};
|
|||
|
this.getList();
|
|||
|
},
|
|||
|
});
|
|||
|
},
|
|||
|
getEpsByScale(zoomLevel) {
|
|||
|
// 缩放等级范围限制在 1 到 18
|
|||
|
if (zoomLevel < 1) zoomLevel = 1;
|
|||
|
if (zoomLevel > 18) zoomLevel = 18;
|
|||
|
|
|||
|
// 基础距离(单位:米),通常在缩放等级 0 时显示整个地球(约 40075 公里)
|
|||
|
const earthCircumference = 40075000; // 地球赤道周长(米)
|
|||
|
|
|||
|
// 根据缩放等级计算当前等级下的单个瓦片的地面宽度
|
|||
|
const tileSize = 256; // 地图瓦片大小(通常是 256x256 像素)
|
|||
|
const scale = 2 ** zoomLevel; // 缩放等级对应的缩放比例
|
|||
|
|
|||
|
// 计算每个像素代表的地面距离(米/像素)
|
|||
|
const metersPerPixel = earthCircumference / (tileSize * scale);
|
|||
|
|
|||
|
// 返回一个以 50 像素为参考的距离(可以根据需求调整这个值)
|
|||
|
const pixelRange = 50; // 聚合点之间的最小像素距离
|
|||
|
return metersPerPixel * pixelRange;
|
|||
|
},
|
|||
|
//根据东北 西南经纬度 以及后台返回标记点 格式化成marker点
|
|||
|
async getFortMatMarkerList(northeast, southwest, scale, backendMarkerList) {
|
|||
|
// 当前显示的坐标范围
|
|||
|
|
|||
|
// console.log(southwest.longitude, northeast.longitude, northeast.latitude, southwest.latitude)
|
|||
|
//屏幕中显示的经度的长度和纬度的长度
|
|||
|
let mapWidth = southwest.longitude - northeast.longitude;
|
|||
|
let mapHeight = northeast.latitude - southwest.latitude;
|
|||
|
//将屏幕中地图分割的横向 格子数和 纵向格子数
|
|||
|
// let widthSize = scale;
|
|||
|
// let heightSize = widthSize + parseInt(scale / 2);
|
|||
|
let widthSize = 30 - parseInt(scale / 2);
|
|||
|
let heightSize = 30 - parseInt(scale / 2);
|
|||
|
|
|||
|
//计算每个格子的经纬度的长度
|
|||
|
let unitWidth = mapWidth / widthSize;
|
|||
|
let unitHeight = mapHeight / heightSize;
|
|||
|
// console.log(southwest.longitude - northeast.longitude, unitHeight)
|
|||
|
let pointData = {};
|
|||
|
backendMarkerList.forEach((latLng) => {
|
|||
|
//如果点在显示范围内
|
|||
|
if (
|
|||
|
latLng.latitude < northeast.latitude &&
|
|||
|
latLng.latitude > southwest.latitude &&
|
|||
|
latLng.longitude < northeast.longitude &&
|
|||
|
latLng.longitude > southwest.longitude
|
|||
|
) {
|
|||
|
let relativeX = latLng.longitude - northeast.longitude;
|
|||
|
let relativeY = latLng.latitude - southwest.latitude;
|
|||
|
//计算出这个点,处于哪个格子
|
|||
|
let x = parseInt(Math.floor(relativeX / unitWidth));
|
|||
|
let y = parseInt(Math.floor(relativeY / unitHeight));
|
|||
|
if (x < 0 || y < 0) {
|
|||
|
console.log("点位不在格子内", "失败");
|
|||
|
}
|
|||
|
//生成单元格点位数据
|
|||
|
let pointKey = x + "," + y;
|
|||
|
if (pointData[pointKey] == undefined) {
|
|||
|
pointData[pointKey] = [];
|
|||
|
}
|
|||
|
if (pointData[pointKey].length <= 99) {
|
|||
|
pointData[pointKey].push(latLng);
|
|||
|
} else {
|
|||
|
return
|
|||
|
}
|
|||
|
} else {
|
|||
|
console.log("点位不在显示范围内", "失败");
|
|||
|
}
|
|||
|
});
|
|||
|
console.info("🚀 ~ file:index method:getFortMatMarkerList line:547 -----", pointData)
|
|||
|
|
|||
|
|
|||
|
// ===== 新增:使用 DBSCAN 进行全局点聚合 start =====
|
|||
|
const eps = this.getEpsByScale(scale); // 根据缩放级别调整 eps 值
|
|||
|
pointData = this.dbscan(pointData, parseInt(eps)); // 聚合参数可调整
|
|||
|
|
|||
|
|
|||
|
this.pointDataList = pointData;
|
|||
|
|
|||
|
|
|||
|
let resultMapArray = [];
|
|||
|
for (let y = 0; y < heightSize; y++) {
|
|||
|
for (let x = 0; x < widthSize; x++) {
|
|||
|
let pointKey = x + "," + y;
|
|||
|
if (
|
|||
|
pointData[pointKey] != undefined &&
|
|||
|
(pointData[pointKey][0].userId ||
|
|||
|
pointData[pointKey][0].userId === 0)
|
|||
|
) {
|
|||
|
let markerItem = {};
|
|||
|
|
|||
|
//聚合点与单点共存 , 长度等于一 不聚合点
|
|||
|
if (pointData[pointKey].length == 1) {
|
|||
|
var HeadImg = "/static/common/img/adminHeaderImg.png?number=1";
|
|||
|
if (pointData[pointKey][0].userId == 0) {
|
|||
|
if (this.vuex_user.sex == "男") {
|
|||
|
HeadImg = "/static/common/img/male.png?number=1";
|
|||
|
}
|
|||
|
if (this.vuex_user.sex == "女") {
|
|||
|
HeadImg = "/static/common/img/female.png?number=1";
|
|||
|
}
|
|||
|
}
|
|||
|
// 请求在线图片文件 判断是否存在
|
|||
|
// const imgExist = await this.doesImageExist(this.$u.http.config.imgUrl +
|
|||
|
// pointData[pointKey][0].imageUrl +
|
|||
|
// "?number=1");
|
|||
|
//
|
|||
|
// if (imgExist) {
|
|||
|
// HeadImg =
|
|||
|
// this.$u.http.config.imgUrl +
|
|||
|
// pointData[pointKey][0].imageUrl +
|
|||
|
// "?number=1";
|
|||
|
// }
|
|||
|
// 线上会出现奇怪的问题,所以先注释掉
|
|||
|
|
|||
|
if (pointData[pointKey][0].imageUrl != "") {
|
|||
|
HeadImg =
|
|||
|
this.$u.http.config.imgUrl +
|
|||
|
pointData[pointKey][0].imageUrl +
|
|||
|
"?number=1";
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
markerItem = {
|
|||
|
id: parseInt(
|
|||
|
(pointData[pointKey][0].latitude -
|
|||
|
-pointData[pointKey][0].longitude) *
|
|||
|
100000
|
|||
|
),
|
|||
|
latitude: pointData[pointKey][0].latitude,
|
|||
|
longitude: pointData[pointKey][0].longitude,
|
|||
|
iconPath: HeadImg,
|
|||
|
width: 50,
|
|||
|
height: 50,
|
|||
|
callout: {
|
|||
|
content: scale == 18 ? pointData[pointKey][0].aiLabel : '',
|
|||
|
display: 'ALWAYS',
|
|||
|
borderRadius: 100,
|
|||
|
bgColor: '#3CB5FB',
|
|||
|
color: '#fff',
|
|||
|
borderColor: '#3CB5FB',
|
|||
|
},
|
|||
|
label: {
|
|||
|
...this.callout,
|
|||
|
width: 50,
|
|||
|
height: 50,
|
|||
|
content: !this.careerList[pointData[pointKey][0].workFieldName] ? `` : `<img src="${this.careerList[pointData[pointKey][0].workFieldName].minIcon}">`
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
|
|||
|
if (pointData[pointKey][0].userId === 0) {
|
|||
|
markerItem.id = '0'
|
|||
|
if (!pointData[pointKey][0].aiLabel) {
|
|||
|
markerItem.title = '我的位置'
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
// let iconPath = pointData[pointKey][0].ScanAndCharge == 1 ? '/img/scanMarkerIcon.png' : '/img/markerIcon.png';
|
|||
|
//长度大于一聚合点
|
|||
|
} else if (pointData[pointKey].length > 1) {
|
|||
|
// console.log('聚合点', pointData[pointKey]);
|
|||
|
// id会被转为number类型,字符类型不能触发点击时间,
|
|||
|
// 最后面加1是为了在结尾为0的情况时,0在转被为number不被裁剪掉,
|
|||
|
// 对比判断id时去掉加上的 1
|
|||
|
if (pointData[pointKey].length > 99) {
|
|||
|
pointData[pointKey] = pointData[pointKey].slice(0, 99)
|
|||
|
}
|
|||
|
|
|||
|
markerItem = {
|
|||
|
id: pointKey.split(",")[0] + "." + pointKey.split(",")[1] + "1",
|
|||
|
latitude: pointData[pointKey][0].latitude,
|
|||
|
longitude: pointData[pointKey][0].longitude,
|
|||
|
iconPath: await this.generateGroupAvatar(pointData[pointKey]), // 使用生成的群组头像
|
|||
|
width: 50,
|
|||
|
height: 50,
|
|||
|
list: pointData[pointKey],
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
resultMapArray.push(markerItem);
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return resultMapArray;
|
|||
|
}
|
|||
|
,
|
|||
|
async doesImageExist(url) {
|
|||
|
try {
|
|||
|
const response = await fetch(url, {method: 'HEAD'});
|
|||
|
return response.ok;
|
|||
|
} catch (error) {
|
|||
|
console.error('图片不存在或发生错误:', error);
|
|||
|
return false;
|
|||
|
}
|
|||
|
},
|
|||
|
generateGroupAvatar(users) {
|
|||
|
|
|||
|
return new Promise((resolve, reject) => {
|
|||
|
try {
|
|||
|
const canvasSize = 100;
|
|||
|
const maxAvatars = 9;
|
|||
|
const filteredUsers = users.slice(0, maxAvatars);
|
|||
|
const total = filteredUsers.length;
|
|||
|
|
|||
|
if (total === 0) {
|
|||
|
resolve('/static/common/img/map_icon.png');
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// 缓存 key
|
|||
|
const userIds = filteredUsers.map(u => u.userId).join(',');
|
|||
|
const cacheKey = `groupAvatar_${userIds}`;
|
|||
|
if (this[cacheKey]) {
|
|||
|
resolve(this[cacheKey]);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// 创建 Canvas
|
|||
|
const canvas = document.createElement('canvas');
|
|||
|
canvas.width = canvasSize;
|
|||
|
canvas.height = canvasSize;
|
|||
|
const ctx = canvas.getContext('2d');
|
|||
|
ctx.fillStyle = '#ffffff';
|
|||
|
ctx.fillRect(0, 0, canvasSize, canvasSize);
|
|||
|
|
|||
|
let drawnCount = 0;
|
|||
|
const padding = 2;
|
|||
|
const verticalPadding = 3;
|
|||
|
|
|||
|
// 布局配置
|
|||
|
const layoutConfig = {
|
|||
|
1: {rows: 1, cols: 1, centeredRows: [0]},
|
|||
|
2: {rows: 1, cols: 2}, // 横向排列,但垂直居中
|
|||
|
3: {rows: 2, cols: [1, 2], centeredRows: [0]}, // 上1 下2,上居中
|
|||
|
4: {rows: 2, cols: 2}, // 2x2 四宫格
|
|||
|
5: {rows: 2, cols: [2, 3], centeredRows: [0]}, // 上2 下3,上居中
|
|||
|
6: {rows: 2, cols: 3}, // 上3 下3
|
|||
|
7: {rows: 3, cols: [1, 3, 3], centeredRows: [0]}, // 上1 中3 下3
|
|||
|
8: {rows: 3, cols: [2, 3, 3], centeredRows: [0]}, // 上2 中3 下3
|
|||
|
9: {rows: 3, cols: 3},
|
|||
|
};
|
|||
|
|
|||
|
const config = layoutConfig[total] || layoutConfig[9];
|
|||
|
const {rows, cols} = config;
|
|||
|
const colCounts = Array.isArray(cols) ? cols : Array(rows).fill(cols);
|
|||
|
const centeredRows = config.centeredRows || [];
|
|||
|
|
|||
|
// 计算最大列数,用于计算每个头像宽度
|
|||
|
const maxCols = Math.max(...colCounts);
|
|||
|
const avatarWidth = (canvasSize - padding * (maxCols - 1)) / maxCols;
|
|||
|
const rowHeight = (canvasSize - verticalPadding * (rows - 1)) / rows;
|
|||
|
|
|||
|
|
|||
|
// 绘制每个头像
|
|||
|
filteredUsers.forEach((user, index) => {
|
|||
|
const image = new Image();
|
|||
|
image.crossOrigin = 'anonymous';
|
|||
|
image.src = this.$u.http.config.imgUrl + user.imageUrl;
|
|||
|
|
|||
|
|
|||
|
// 确定当前头像属于哪一行
|
|||
|
let currentRow = 0;
|
|||
|
let count = 0;
|
|||
|
for (let i = 0; i < rows; i++) {
|
|||
|
const colCount = colCounts[i];
|
|||
|
if (index < count + colCount) {
|
|||
|
currentRow = i;
|
|||
|
break;
|
|||
|
}
|
|||
|
count += colCount;
|
|||
|
}
|
|||
|
|
|||
|
const colIndex = index - count;
|
|||
|
const currentColCount = colCounts[currentRow];
|
|||
|
|
|||
|
// 如果该行需要居中,则计算起始X坐标
|
|||
|
const startX = centeredRows.includes(currentRow)
|
|||
|
? (canvasSize - currentColCount * (avatarWidth + padding) + padding) / 2
|
|||
|
: 0;
|
|||
|
|
|||
|
const x = startX + colIndex * (avatarWidth + padding);
|
|||
|
const y = currentRow * (rowHeight + verticalPadding);
|
|||
|
|
|||
|
// 固定绘制为正方形,避免拉伸
|
|||
|
const drawSize = Math.min(avatarWidth, rowHeight);
|
|||
|
|
|||
|
// 居中绘制
|
|||
|
const offsetY = (rowHeight - drawSize) / 2;
|
|||
|
|
|||
|
image.onload = () => {
|
|||
|
ctx.save();
|
|||
|
ctx.beginPath();
|
|||
|
ctx.rect(x, y + offsetY, drawSize, drawSize);
|
|||
|
ctx.clip();
|
|||
|
|
|||
|
// 从图片中心裁剪出一个正方形
|
|||
|
const imgRatio = image.width / image.height;
|
|||
|
const cropSize = imgRatio > 1 ? image.height : image.width;
|
|||
|
const offsetX = (image.width - cropSize) / 2;
|
|||
|
const offsetYImg = (image.height - cropSize) / 2;
|
|||
|
|
|||
|
ctx.drawImage(
|
|||
|
image,
|
|||
|
offsetX, offsetYImg, cropSize, cropSize, // 源图裁剪区域
|
|||
|
x, y + offsetY, drawSize, drawSize // 目标绘制区域
|
|||
|
);
|
|||
|
|
|||
|
ctx.restore();
|
|||
|
|
|||
|
drawnCount++;
|
|||
|
if (drawnCount === filteredUsers.length) {
|
|||
|
// 如果超过最大数量,覆盖整个画布并居中显示 "+n"
|
|||
|
if (users.length > maxAvatars) {
|
|||
|
ctx.fillStyle = 'rgba(0, 0, 0, 0.2)';
|
|||
|
ctx.beginPath();
|
|||
|
ctx.rect(0, 0, canvasSize, canvasSize);
|
|||
|
ctx.fill();
|
|||
|
|
|||
|
ctx.fillStyle = '#ffffff';
|
|||
|
ctx.font = 'bold 40px Arial';
|
|||
|
ctx.textAlign = 'center';
|
|||
|
ctx.textBaseline = 'middle';
|
|||
|
ctx.fillText(`+${users.length}`, canvasSize / 2, canvasSize / 2);
|
|||
|
}
|
|||
|
|
|||
|
// 缓存结果并返回
|
|||
|
const result = canvas.toDataURL('image/png');
|
|||
|
this[cacheKey] = result;
|
|||
|
resolve(result); // 异步完成
|
|||
|
}
|
|||
|
};
|
|||
|
|
|||
|
image.onerror = (err) => {
|
|||
|
image.src = "";
|
|||
|
image.onload = () => {
|
|||
|
ctx.save();
|
|||
|
ctx.beginPath();
|
|||
|
ctx.rect(x, y + offsetY, drawSize, drawSize);
|
|||
|
ctx.clip();
|
|||
|
|
|||
|
// 从图片中心裁剪出一个正方形
|
|||
|
const imgRatio = image.width / image.height;
|
|||
|
const cropSize = imgRatio > 1 ? image.height : image.width;
|
|||
|
const offsetX = (image.width - cropSize) / 2;
|
|||
|
const offsetYImg = (image.height - cropSize) / 2;
|
|||
|
|
|||
|
ctx.drawImage(
|
|||
|
image,
|
|||
|
offsetX, offsetYImg, cropSize, cropSize, // 源图裁剪区域
|
|||
|
x, y + offsetY, drawSize, drawSize // 目标绘制区域
|
|||
|
);
|
|||
|
|
|||
|
ctx.restore();
|
|||
|
|
|||
|
drawnCount++;
|
|||
|
if (drawnCount === filteredUsers.length) {
|
|||
|
// 如果超过最大数量,覆盖整个画布并居中显示 "+n"
|
|||
|
if (users.length > maxAvatars) {
|
|||
|
ctx.fillStyle = 'rgba(0, 0, 0, 0.2)';
|
|||
|
ctx.beginPath();
|
|||
|
ctx.rect(0, 0, canvasSize, canvasSize);
|
|||
|
ctx.fill();
|
|||
|
|
|||
|
ctx.fillStyle = '#ffffff';
|
|||
|
ctx.font = 'bold 40px Arial';
|
|||
|
ctx.textAlign = 'center';
|
|||
|
ctx.textBaseline = 'middle';
|
|||
|
ctx.fillText(`+${users.length}`, canvasSize / 2, canvasSize / 2);
|
|||
|
}
|
|||
|
|
|||
|
// 缓存结果并返回
|
|||
|
const result = canvas.toDataURL('image/png');
|
|||
|
this[cacheKey] = result;
|
|||
|
resolve(result); // 异步完成
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
});
|
|||
|
} catch (error) {
|
|||
|
console.error('生成聚合头像时发生异常:', error);
|
|||
|
resolve('/static/common/img/map_icon.png');
|
|||
|
}
|
|||
|
});
|
|||
|
}
|
|||
|
,
|
|||
|
/**
|
|||
|
* DBSCAN 聚类算法(输出格式与 pointData 一致)
|
|||
|
* @param points 原始点数组 { latitude, longitude }
|
|||
|
* @param eps 聚类距离阈值(单位:米)
|
|||
|
* @returns {{}} 统一格式的 pointData 结构,key 格式为 "x,y"
|
|||
|
*/
|
|||
|
dbscan(points, eps = 50) {
|
|||
|
|
|||
|
function getDistanceInMeters(lat1, lon1, lat2, lon2) {
|
|||
|
const R = 6371; // 地球半径(公里)
|
|||
|
const dLat = (lat2 - lat1) * Math.PI / 180;
|
|||
|
const dLon = (lon2 - lon1) * Math.PI / 180;
|
|||
|
const a =
|
|||
|
Math.sin(dLat / 2) * Math.sin(dLat / 2) +
|
|||
|
Math.cos(lat1 * Math.PI / 180) * Math.cos(lat2 * Math.PI / 180) *
|
|||
|
Math.sin(dLon / 2) * Math.sin(dLon / 2);
|
|||
|
const c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a));
|
|||
|
const distanceInKm = R * c;
|
|||
|
const distanceInMeters = distanceInKm * 1000; // 将公里转换为米
|
|||
|
console.info("🚀 ~ file:index method:getDistanceInMeters line:886 -----", distanceInMeters)
|
|||
|
|
|||
|
return parseInt(distanceInMeters) > 100000 ? 100000 : parseInt(distanceInMeters);
|
|||
|
}
|
|||
|
|
|||
|
let deletedPoints = [];
|
|||
|
for (let key in points) {
|
|||
|
if (deletedPoints.includes(key)) {
|
|||
|
continue;
|
|||
|
}
|
|||
|
let X = key.split(',')[0]
|
|||
|
let Y = key.split(',')[1]
|
|||
|
for (let keyTwo in points) {
|
|||
|
let innerX = keyTwo.split(',')[0]
|
|||
|
let innerY = keyTwo.split(',')[1]
|
|||
|
if (key === keyTwo) {
|
|||
|
continue
|
|||
|
}
|
|||
|
if (Math.abs(X - innerX) <= 1 && Math.abs(Y - innerY) <= 1) {
|
|||
|
if (getDistanceInMeters(points[key][0].latitude, points[key][0].longitude, points[keyTwo][0].latitude, points[keyTwo][0].longitude) < eps) {
|
|||
|
points[key] = points[key].concat(points[keyTwo])
|
|||
|
deletedPoints.push(keyTwo)
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
for (let key in points) {
|
|||
|
if (deletedPoints.includes(key)) {
|
|||
|
delete points[key]
|
|||
|
}
|
|||
|
}
|
|||
|
return points;
|
|||
|
// ===== 构造统一格式 end =====
|
|||
|
}
|
|||
|
|
|||
|
,
|
|||
|
/**
|
|||
|
* 将聚类结果转为类似 pointData 的结构("x,y": [])
|
|||
|
* @param clusters 聚类结果二维数组
|
|||
|
* @param noise 噪声点
|
|||
|
* @returns [[]] 合并后的伪格子结构
|
|||
|
*/
|
|||
|
_generateFakeGrids(clusters, noise) {
|
|||
|
const result = [...clusters]; // 每个 cluster 单独作为一个格子
|
|||
|
|
|||
|
// 如果需要可选地将噪声点也作为一个格子加入
|
|||
|
if (noise.length > 0) {
|
|||
|
result.push(noise); // 可选
|
|||
|
}
|
|||
|
|
|||
|
return result;
|
|||
|
}
|
|||
|
,
|
|||
|
updated() {
|
|||
|
// console.log("渲染完成");
|
|||
|
}
|
|||
|
,
|
|||
|
// 当视野发生改变
|
|||
|
regionchange(e) {
|
|||
|
// console.log("当视野发生改变");
|
|||
|
// 视野移动结束再获取数据
|
|||
|
if (e.type == "begin") return;
|
|||
|
// 关闭显示的窗口
|
|||
|
this.showInfo = false
|
|||
|
|
|||
|
this.test()
|
|||
|
|
|||
|
// let loadDelay = 200;
|
|||
|
// if (this.load) {
|
|||
|
// // console.log("等待加载");
|
|||
|
// } else {
|
|||
|
// this.load = true;
|
|||
|
// setTimeout(() => {
|
|||
|
// // this.getList();
|
|||
|
// this.test();
|
|||
|
// }, loadDelay);
|
|||
|
// }
|
|||
|
}
|
|||
|
,
|
|||
|
isWechat() {
|
|||
|
var ua = window.navigator.userAgent.toLowerCase();
|
|||
|
if (ua.match(/micromessenger/i) == "micromessenger") {
|
|||
|
// console.log(‘是微信客户端’)
|
|||
|
return true;
|
|||
|
} else {
|
|||
|
// console.log(‘不是微信客户端’)
|
|||
|
return false;
|
|||
|
}
|
|||
|
}
|
|||
|
,
|
|||
|
//在需要定位页面调用
|
|||
|
getlocation: function (callback) {
|
|||
|
if (!this.isWechat()) {
|
|||
|
//console.log('不是微信客户端')
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
,
|
|||
|
//打开位置
|
|||
|
openlocation: function (data) {
|
|||
|
if (!this.isWechat()) {
|
|||
|
//console.log('不是微信客户端')
|
|||
|
return;
|
|||
|
}
|
|||
|
jweixin.ready(function () {
|
|||
|
jweixin.openLocation({
|
|||
|
latitude: data.latitude,
|
|||
|
longitude: data.longitude,
|
|||
|
name: data.name,
|
|||
|
address: data.address,
|
|||
|
scale: 14,
|
|||
|
});
|
|||
|
});
|
|||
|
}
|
|||
|
,
|
|||
|
//定位当前
|
|||
|
location(state) {
|
|||
|
var that = this;
|
|||
|
uni.showLoading({
|
|||
|
title: '正在加载...',
|
|||
|
mask: true
|
|||
|
})
|
|||
|
if (state && this.isWechat()) {
|
|||
|
const isiOS = !!navigator.userAgent.match(
|
|||
|
/\(i[^;]+;( U;)? CPU.+Mac OS X/
|
|||
|
); //ios终端
|
|||
|
// 进行签名的时候 Android 不用使用之前的链接, ios 需要
|
|||
|
const signLink = isiOS
|
|||
|
? window.entryUrl
|
|||
|
: window.location.href.split("#")[0];
|
|||
|
//获取当前url然后传递给后台获取授权和签名信息,后台需要解码才能使用
|
|||
|
// const url =encodeURIComponent(signLink);
|
|||
|
const url = signLink;
|
|||
|
const data = {
|
|||
|
url: url,
|
|||
|
};
|
|||
|
|
|||
|
this.$u.api.GetInfoMation(data).then((res) => {
|
|||
|
jweixin.config({
|
|||
|
debug: false,
|
|||
|
appId: res.appId,
|
|||
|
timestamp: res.timestamp,
|
|||
|
nonceStr: res.noncestr,
|
|||
|
signature: res.signature,
|
|||
|
jsApiList: [
|
|||
|
//这里是需要用到的接口名称
|
|||
|
"getLocation", //获取位置
|
|||
|
"openLocation", //打开位置
|
|||
|
],
|
|||
|
});
|
|||
|
|
|||
|
// jweixin.config 执行失败时调用
|
|||
|
jweixin.error((err) => {
|
|||
|
uni.hideLoading();
|
|||
|
that.location(false);
|
|||
|
console.log("授权失败,您可能无法使用部分功能", err);
|
|||
|
});
|
|||
|
jweixin.ready(function () {
|
|||
|
jweixin.getLocation({
|
|||
|
type: "gcj02", // 默认为wgs84的gps坐标,如果要返 回直接给openLocation用的火星坐标,可传入'gcj02'
|
|||
|
success: (res) => {
|
|||
|
uni.hideLoading();
|
|||
|
var resdata = res;
|
|||
|
// console.log(res,'success');
|
|||
|
map.moveToLocation({
|
|||
|
latitude: resdata.latitude,
|
|||
|
longitude: resdata.longitude,
|
|||
|
});
|
|||
|
// alert(JSON.stringify(res))
|
|||
|
that.center.latitude = resdata.latitude; //纬度
|
|||
|
that.center.longitude = resdata.longitude; //经度
|
|||
|
|
|||
|
// 更新坐标
|
|||
|
var data = {
|
|||
|
userId: that.vuex_user.id,
|
|||
|
longitude: resdata.longitude + "",
|
|||
|
latitude: resdata.latitude + "",
|
|||
|
};
|
|||
|
that.$u.api.upPosition(data).then(() => {
|
|||
|
that.getList(resdata.latitude, resdata.longitude);
|
|||
|
that.test();
|
|||
|
});
|
|||
|
},
|
|||
|
fail: function (res) {
|
|||
|
uni.hideLoading();
|
|||
|
// console.log(res, "err");
|
|||
|
},
|
|||
|
complete: function (res) {
|
|||
|
uni.hideLoading();
|
|||
|
// console.log(res, "is");
|
|||
|
},
|
|||
|
});
|
|||
|
});
|
|||
|
});
|
|||
|
} else {
|
|||
|
uni.getLocation({
|
|||
|
// type: "wgs84 ",
|
|||
|
type: "gcj02 ",
|
|||
|
isHighAccuracy: true,
|
|||
|
highAccuracyExpireTime: 3000,
|
|||
|
success: (res) => {
|
|||
|
|
|||
|
uni.hideLoading();
|
|||
|
// console.info("🚀 ~ file:index method:success line:497 -----", res);
|
|||
|
|
|||
|
// console.log(res.latitude + "," + res.longitude, "gcj02");
|
|||
|
map.moveToLocation({
|
|||
|
latitude: res.latitude,
|
|||
|
longitude: res.longitude,
|
|||
|
});
|
|||
|
this.center.latitude = res.latitude; //纬度
|
|||
|
this.center.longitude = res.longitude; //经度
|
|||
|
|
|||
|
// 更新坐标
|
|||
|
var data = {
|
|||
|
userId: this.vuex_user.id,
|
|||
|
longitude: res.longitude + "",
|
|||
|
latitude: res.latitude + "",
|
|||
|
};
|
|||
|
this.$u.api.upPosition(data).then(() => {
|
|||
|
this.getList(res.latitude, res.longitude);
|
|||
|
this.test();
|
|||
|
});
|
|||
|
},
|
|||
|
fail: (err) => {
|
|||
|
uni.hideLoading();
|
|||
|
console.log("地理位置获取失败", err);
|
|||
|
},
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
,
|
|||
|
|
|||
|
testlist() {
|
|||
|
// this.getList();
|
|||
|
this.test();
|
|||
|
}
|
|||
|
,
|
|||
|
// 获取用户列表
|
|||
|
getList(la = 0, lo = 0) {
|
|||
|
var arr = [];
|
|||
|
map.getCenterLocation({
|
|||
|
success: (res) => {
|
|||
|
let latitude = res.latitude;
|
|||
|
let longitude = res.longitude;
|
|||
|
if (la !== 0) {
|
|||
|
latitude = la;
|
|||
|
longitude = lo;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
// console.info("🚀 ~ file:index method:success line:1038 -----", latitude,longitude)
|
|||
|
this.latitude = latitude;
|
|||
|
this.longitude = longitude;
|
|||
|
let data = {
|
|||
|
userId: this.vuex_user.id,
|
|||
|
lat: this.center.latitude,
|
|||
|
lon: this.center.longitude,
|
|||
|
nort_lat: this.Showip?.northeast?.latitude || 0,
|
|||
|
nort_lon: this.Showip?.northeast?.longitude || 0,
|
|||
|
sout_lat: this.Showip?.southwest?.latitude || 0,
|
|||
|
sout_lon: this.Showip?.southwest?.longitude || 0,
|
|||
|
};
|
|||
|
if (this.timeer) {
|
|||
|
clearTimeout(this.timeer)
|
|||
|
}
|
|||
|
this.timeer = setTimeout(() => {
|
|||
|
this.$u.api.HomeMap(data).then((res) => {
|
|||
|
if (res) {
|
|||
|
res.unshift(this.center);
|
|||
|
// res = res.reverse();
|
|||
|
this.list = res;
|
|||
|
// this.test();
|
|||
|
} else {
|
|||
|
/* uni.showToast({
|
|||
|
title: res.data.msg,
|
|||
|
icon: "none",
|
|||
|
}); */
|
|||
|
}
|
|||
|
map.getScale({
|
|||
|
success: async (ress) => {
|
|||
|
this.covers = await this.getFortMatMarkerList(
|
|||
|
this.Showip.northeast,
|
|||
|
this.Showip.southwest,
|
|||
|
ress.scale,
|
|||
|
this.list
|
|||
|
);
|
|||
|
document.querySelectorAll(
|
|||
|
"#map .csssprite:nth-child(1)"
|
|||
|
)[0].style.border = "none";
|
|||
|
},
|
|||
|
});
|
|||
|
}).catch(() => {
|
|||
|
this.list = [this.center];
|
|||
|
map.getScale({
|
|||
|
success: async (ress) => {
|
|||
|
this.covers = await this.getFortMatMarkerList(
|
|||
|
this.Showip.northeast,
|
|||
|
this.Showip.southwest,
|
|||
|
ress.scale,
|
|||
|
this.list
|
|||
|
);
|
|||
|
document.querySelectorAll(
|
|||
|
"#map .csssprite:nth-child(1)"
|
|||
|
)[0].style.border = "none";
|
|||
|
},
|
|||
|
});
|
|||
|
});
|
|||
|
}, 500)
|
|||
|
},
|
|||
|
});
|
|||
|
}
|
|||
|
,
|
|||
|
tap() {
|
|||
|
this.showInfo = false;
|
|||
|
}
|
|||
|
,
|
|||
|
// 点击获取当前用户信息 注意:<每个 marker 点都必须要有id 不然点击maxker的事件不会触发>
|
|||
|
getUserDetail(i) {
|
|||
|
// console.info(
|
|||
|
// "🚀 ~ file:index method:getUserDetail line:574 -----",
|
|||
|
// i.detail.markerId
|
|||
|
// );
|
|||
|
|
|||
|
//点击中心点时不显示
|
|||
|
if (i.detail.markerId == 0) return;
|
|||
|
// 判断是否是聚合点
|
|||
|
this.detailInfoList = [];
|
|||
|
if (String(i.detail.markerId).split(".").length > 1) {
|
|||
|
var pointKey = String(i.detail.markerId).replace(".", ",");
|
|||
|
pointKey = pointKey.slice(0, pointKey.length - 1);
|
|||
|
let arr = this.pointDataList[pointKey];
|
|||
|
let isprovince = "";
|
|||
|
let iscity = "";
|
|||
|
// 地址解析 免费 每日10000请求量
|
|||
|
uni.showLoading({
|
|||
|
title: "加载中",
|
|||
|
});
|
|||
|
this.$jsonp(
|
|||
|
"https://apis.map.qq.com/ws/geocoder/v1/?key=2OLBZ-OOSRQ-RYZ5A-GMCM2-DJ43O-3QFLS&location=" +
|
|||
|
arr[0].latitude +
|
|||
|
"," +
|
|||
|
arr[0].longitude +
|
|||
|
"&output=jsonp&get_poi=1"
|
|||
|
).then((res) => {
|
|||
|
if (res.status == 0) {
|
|||
|
isprovince = res.result.address_component.province;
|
|||
|
iscity = res.result.address_component.city;
|
|||
|
for (let i in arr) {
|
|||
|
arr[i].province = isprovince ? isprovince : "未";
|
|||
|
arr[i].city = iscity ? iscity : "知";
|
|||
|
}
|
|||
|
this.detailInfoList = arr;
|
|||
|
this.isExpanded = '0vh';
|
|||
|
this.showPopup = true
|
|||
|
setTimeout(() => {
|
|||
|
this.isExpanded = '90vh';
|
|||
|
}, 10);
|
|||
|
// this.showList();
|
|||
|
uni.hideLoading();
|
|||
|
}
|
|||
|
});
|
|||
|
} else {
|
|||
|
|
|||
|
// 根据id筛选获取用户信息
|
|||
|
let a = this.list.filter((item) => {
|
|||
|
|
|||
|
return (
|
|||
|
parseInt((item.latitude - -item.longitude) * 100000) ==
|
|||
|
i.detail.markerId
|
|||
|
);
|
|||
|
})[0];
|
|||
|
|
|||
|
if (a.userId == this.detailInfo.userId) {
|
|||
|
setTimeout(() => {
|
|||
|
this.showInfo = true;
|
|||
|
this.showPopup = false
|
|||
|
}, 100);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
// 地址解析
|
|||
|
this.$jsonp(
|
|||
|
"https://apis.map.qq.com/ws/geocoder/v1/?key=2OLBZ-OOSRQ-RYZ5A-GMCM2-DJ43O-3QFLS&location=" +
|
|||
|
a.latitude +
|
|||
|
"," +
|
|||
|
a.longitude +
|
|||
|
"&output=jsonp&get_poi=1"
|
|||
|
).then((res) => {
|
|||
|
// console.log("解析地址成功", res);
|
|||
|
// console.log("当前地址:", res.result.address);
|
|||
|
if (res.status == 0) {
|
|||
|
a.province = res.result.address_component.province;
|
|||
|
a.city = res.result.address_component.city;
|
|||
|
}
|
|||
|
this.detailInfo = a;
|
|||
|
setTimeout(() => {
|
|||
|
this.showInfo = true;
|
|||
|
this.showPopup = false
|
|||
|
}, 100);
|
|||
|
});
|
|||
|
}
|
|||
|
}
|
|||
|
,
|
|||
|
|
|||
|
togglePopup() {
|
|||
|
this.isExpanded = '90vh';
|
|||
|
}
|
|||
|
,
|
|||
|
touchStart(e) {
|
|||
|
// 记录起始触摸点
|
|||
|
this.touchStartY = e.touches[0].clientY;
|
|||
|
// 记录起始滚动位置
|
|||
|
this.startScrollTop = e.currentTarget.scrollTop;
|
|||
|
}
|
|||
|
,
|
|||
|
touchMove(e) {
|
|||
|
|
|||
|
const currentY = e.touches[0].clientY;
|
|||
|
const diff = this.touchStartY - currentY;
|
|||
|
const scrollTop = e.currentTarget.scrollTop;
|
|||
|
|
|||
|
// 判断是否在内容顶部或底部
|
|||
|
const isAtTop = scrollTop <= 0;
|
|||
|
const isAtBottom = scrollTop + e.currentTarget.clientHeight >= e.currentTarget.scrollHeight;
|
|||
|
|
|||
|
// 在顶部向下滑动或在底部向上滑动时,阻止默认行为
|
|||
|
if ((isAtTop && diff < 0) || (isAtBottom && diff > 0)) {
|
|||
|
e.preventDefault();
|
|||
|
}
|
|||
|
|
|||
|
// 向上滑动超过50px时展开
|
|||
|
if (diff > 50) {
|
|||
|
this.isExpanded = '90vh';
|
|||
|
}
|
|||
|
// 向下滑动超过50px时收起
|
|||
|
else if (diff < -50) {
|
|||
|
if (this.isExpanded == '40vh') {
|
|||
|
console.log("关闭");
|
|||
|
this.isExpanded = '0vh';
|
|||
|
setTimeout(() => {
|
|||
|
this.showPopup = false
|
|||
|
}, 100);
|
|||
|
} else {
|
|||
|
console.log("收起");
|
|||
|
this.isExpanded = '40vh';
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
,
|
|||
|
touchMoveFn(e) {
|
|||
|
if (this.timeer) {
|
|||
|
clearTimeout(this.timeer)
|
|||
|
}
|
|||
|
this.timeer = setTimeout(() => {
|
|||
|
this.touchMove(e)
|
|||
|
}, 100)
|
|||
|
}
|
|||
|
,
|
|||
|
touchEnd(e) {
|
|||
|
// 可以在这里添加滑动结束后的处理逻辑
|
|||
|
}
|
|||
|
,
|
|||
|
},
|
|||
|
};
|
|||
|
</script>
|
|||
|
|
|||
|
<style scoped lang="scss">
|
|||
|
.AIdescribe {
|
|||
|
display: block;
|
|||
|
font-size: 26rpx;
|
|||
|
color: rgba(0, 0, 0, 0.4);
|
|||
|
line-height: 30rpx;
|
|||
|
text-align: left;
|
|||
|
margin-top: 20rpx;
|
|||
|
max-width: 67vw;
|
|||
|
white-space: break-spaces;
|
|||
|
}
|
|||
|
|
|||
|
.tags {
|
|||
|
gap: 10rpx;
|
|||
|
margin-top: 20rpx;
|
|||
|
}
|
|||
|
|
|||
|
.h100 {
|
|||
|
height: 100%
|
|||
|
}
|
|||
|
|
|||
|
.search-box {
|
|||
|
width: 72rpx;
|
|||
|
height: 72rpx;
|
|||
|
border-radius: 50%;
|
|||
|
background: #fff;
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
justify-content: center;
|
|||
|
|
|||
|
position: absolute;
|
|||
|
top: 50rpx;
|
|||
|
left: 50rpx;
|
|||
|
z-index: 1;
|
|||
|
box-shadow: 0rpx 10rpx 10rpx -6rpx rgba(0, 0, 0, 0.1),
|
|||
|
0rpx 16rpx 20rpx 2rpx rgba(0, 0, 0, 0.06),
|
|||
|
0rpx 6rpx 28rpx 4rpx rgba(0, 0, 0, 0.05);
|
|||
|
}
|
|||
|
|
|||
|
::v-deep .uni-scroll-view-content {
|
|||
|
background: #f6f8f9 !important;
|
|||
|
}
|
|||
|
|
|||
|
::v-deep #map .csssprite {
|
|||
|
border-radius: 50%;
|
|||
|
border: 2px solid #358ddf !important;
|
|||
|
filter: drop-shadow(0px 0px 2px #358ddf);
|
|||
|
transform: scale(0.8);
|
|||
|
}
|
|||
|
|
|||
|
::v-deep #map > div > div > div > div > div:nth-last-child(4) > div > div {
|
|||
|
overflow: initial !important;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
// ::v-deep #map>div>div>div>div>div:nth-last-child(4)>div>div:nth-child(1)>.csssprite[src=''] {
|
|||
|
// border: none !important;
|
|||
|
// }
|
|||
|
.name-work {
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
padding-left: 20rpx;
|
|||
|
}
|
|||
|
|
|||
|
.info {
|
|||
|
position: absolute;
|
|||
|
right: 0;
|
|||
|
left: 0;
|
|||
|
transform: translateY(-125%);
|
|||
|
|
|||
|
.section_2 {
|
|||
|
padding: 0.11rem 0.18rem 0.15rem 0.21rem;
|
|||
|
background-color: rgb(255, 255, 255);
|
|||
|
box-shadow: 0px 0px 0.1rem rgba(0, 0, 0, 0.1);
|
|||
|
border-radius: 0.1rem;
|
|||
|
|
|||
|
.text_3 {
|
|||
|
color: #3CB5FB;
|
|||
|
font-size: 26rpx;
|
|||
|
white-space: nowrap;
|
|||
|
overflow: hidden;
|
|||
|
text-overflow: ellipsis;
|
|||
|
margin-left: 10rpx;
|
|||
|
vertical-align: middle;
|
|||
|
|
|||
|
img {
|
|||
|
vertical-align: middle;
|
|||
|
Width: 48rpx;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.group_6 {
|
|||
|
white-space: nowrap;
|
|||
|
border-top: 1px solid #F4F6F8;
|
|||
|
padding-top: 20rpx;
|
|||
|
margin-top: 20rpx;
|
|||
|
|
|||
|
.text_4 {
|
|||
|
margin-right: 0.1rem;
|
|||
|
color: rgb(63, 63, 63);
|
|||
|
font-size: 0.14rem;
|
|||
|
line-height: 0.16rem;
|
|||
|
white-space: nowrap;
|
|||
|
}
|
|||
|
|
|||
|
.text_5 {
|
|||
|
font-size: 28rpx;
|
|||
|
color: rgba(0, 0, 0, 0.6);
|
|||
|
line-height: 0.16rem;
|
|||
|
letter-spacing: 0.014rem;
|
|||
|
white-space: nowrap;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.group_5 {
|
|||
|
color: rgb(56, 58, 63);
|
|||
|
font-size: 0.18rem;
|
|||
|
line-height: 0.17rem;
|
|||
|
white-space: nowrap;
|
|||
|
|
|||
|
.image_5 {
|
|||
|
width: 0.44rem;
|
|||
|
height: 0.44rem;
|
|||
|
border-radius: 50%;
|
|||
|
}
|
|||
|
|
|||
|
.text_1 {
|
|||
|
font-size: 30rpx;
|
|||
|
// margin-left: 0.13rem;
|
|||
|
//margin-top: 22rpx;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.section_3 {
|
|||
|
padding: 20rpx 20rpx;
|
|||
|
color: rgb(255, 255, 255);
|
|||
|
font-size: 24rpx;
|
|||
|
white-space: nowrap;
|
|||
|
background-color: #3cb5fb;
|
|||
|
border-radius: 0.15rem;
|
|||
|
height: 56rpx;
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
justify-content: center;
|
|||
|
position: absolute;
|
|||
|
right: 0;
|
|||
|
top: 50%;
|
|||
|
transform: translateY(-50%);
|
|||
|
|
|||
|
.image_6 {
|
|||
|
width: 32rpx;
|
|||
|
height: 32rpx;
|
|||
|
}
|
|||
|
|
|||
|
.text_2 {
|
|||
|
margin-left: 0.055rem;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// .text {
|
|||
|
// text-transform: uppercase;
|
|||
|
// }
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.bottom {
|
|||
|
position: fixed;
|
|||
|
z-index: 10;
|
|||
|
bottom: 0;
|
|||
|
right: 2%;
|
|||
|
left: 2%;
|
|||
|
padding-bottom: 0.48rem;
|
|||
|
|
|||
|
.btn {
|
|||
|
display: flex;
|
|||
|
justify-content: center;
|
|||
|
margin-top: 0.22rem;
|
|||
|
}
|
|||
|
|
|||
|
.section_4 {
|
|||
|
margin-right: 0.025rem;
|
|||
|
margin-top: 0.17rem;
|
|||
|
padding: 0.2rem 0.2rem 0.3rem;
|
|||
|
color: rgb(193, 196, 204);
|
|||
|
font-size: 0.14rem;
|
|||
|
line-height: 0.14rem;
|
|||
|
white-space: nowrap;
|
|||
|
background-color: rgb(255, 255, 255);
|
|||
|
box-shadow: 0px 0px 0.03rem rgba(0, 0, 0, 0.1);
|
|||
|
border-radius: 0.1rem 0.1rem 0 0;
|
|||
|
|
|||
|
.search {
|
|||
|
margin-left: 0.14rem;
|
|||
|
margin-right: 0.1rem;
|
|||
|
padding: 0.14rem 0 0.13rem;
|
|||
|
background-color: rgb(246, 247, 250);
|
|||
|
border-radius: 0.05rem;
|
|||
|
|
|||
|
// .text {
|
|||
|
// text-transform: uppercase;
|
|||
|
// }
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
.popup-box {
|
|||
|
position: fixed;
|
|||
|
left: 0;
|
|||
|
right: 0;
|
|||
|
bottom: 0;
|
|||
|
min-height: 240rpx;
|
|||
|
box-shadow: 0rpx -14rpx 16rpx 0rpx rgba(0, 0, 0, 0.06);
|
|||
|
border-top-left-radius: 40rpx;
|
|||
|
border-top-right-radius: 40rpx;
|
|||
|
transition: all 0.3s ease;
|
|||
|
z-index: 100;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
.popup-top {
|
|||
|
// position: fixed;
|
|||
|
// top: 0;
|
|||
|
width: 100%;
|
|||
|
padding-top: 20rpx;
|
|||
|
background: #fff;
|
|||
|
border-top-left-radius: 40rpx;
|
|||
|
border-top-right-radius: 40rpx;
|
|||
|
cursor: pointer; // 添加手型光标
|
|||
|
}
|
|||
|
|
|||
|
.pop-top {
|
|||
|
width: 88rpx;
|
|||
|
height: 10rpx;
|
|||
|
background: #9ea1b9;
|
|||
|
border-radius: 200rpx;
|
|||
|
margin: 20rpx auto 0;
|
|||
|
}
|
|||
|
|
|||
|
.pop-title {
|
|||
|
padding: 20rpx 32rpx 40rpx;
|
|||
|
font-weight: 600;
|
|||
|
font-size: 32rpx;
|
|||
|
color: #191919;
|
|||
|
background: #fff;
|
|||
|
}
|
|||
|
|
|||
|
.content {
|
|||
|
height: calc(100% - 174rpx) !important;
|
|||
|
overflow-y: auto;
|
|||
|
background: #f6f8f9;
|
|||
|
padding: 30rpx 30rpx 50rpx;
|
|||
|
// -webkit-overflow-scrolling: touch;
|
|||
|
}
|
|||
|
|
|||
|
.back-to-location {
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
justify-content: center;
|
|||
|
position: absolute;
|
|||
|
bottom: 180rpx;
|
|||
|
right: 50rpx;
|
|||
|
z-index: 1;
|
|||
|
}
|
|||
|
|
|||
|
::v-deep #map > div:nth-child(1) > div > div:nth-child(1) > div > div:nth-child(2) {
|
|||
|
z-index: 999999 !important;
|
|||
|
}
|
|||
|
|
|||
|
::v-deep #map a[title="到腾讯地图查看此区域"] {
|
|||
|
display: none !important;
|
|||
|
}
|
|||
|
|
|||
|
.position-relative {
|
|||
|
position: relative;
|
|||
|
}
|
|||
|
</style>
|