YingXingAI/pages/main/index/index.vue

1634 lines
57 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<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" 缩放max18 min5-->
<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>