上次提到uniapp开发地图app时得一些问题,最后提到使用renderjs实现app中使用任何地图(下面将以腾讯地图为例,uniapp中写app时推荐使用得是高德地图,无法使用腾讯地图(renderjs方式除外))。
1、renderjs介绍,这里直接参照uniapp开发官网介绍即可:renderjs | uni-app官网
官网也有示例,示例是使用echart得,不过使用方法和地图相类似。
下面先看一下demo效果:
这是一个app中使用地图得makers,点击单个makers会显示对应得图片在上面,并且点击得maker会被放大,这个功能使用uniapp得map组件时是会有很大得问题得,再次就不再赘述。下面看看代码实现。
1、先来看看整个页面得组成(文章最后附上相关示例代码):
从上面可以看到script标签有两个,一个是普通得script标签,另一个是lang="renderjs"得标签。
2、注意事项:上面提到得renderjs中,module得名字需要和template上得标签名称一致,如图:
mapModule.update中update是该模块下面得一个方法。
3、接下来介绍一下
<view class="curr" id="container" :prop="mapModel" :change:prop="mapModule.update"></view>
改代码,mapModel为script普通标签里面得一个响应式变量
:change:prop="mapModule.update" 含义,当script里面得mapModel数据发生变化时,会通过该处触发renderjs里面得update方法。
4、renderjs中不能使用http请求,相关请求只能放在普通标签里面,renderjs仅根据数据变化做逻辑处理(可以理解为一个子组件)
5、最后附上相关代码(mapDemo.vue):
<template>
<view class="index-wrap">
<view class="curr-position" @click="mapModule.setPosition">
<image src="../../static/mapImg/5.png"></image>
</view>
<view class="curr" id="container" :prop="mapModel" :change:prop="mapModule.update"></view>
</view>
</template>
<script>
export default {
components: { IndexTabs, UpdateApp },
data() {
return {
showMenu: false,
id: 0, // 使用 marker点击事件 需要填写id
mapModel: {}
};
},
computed: {
getShow() {
if (this.showMenu) {
return true;
} else {
const timer = setTimeout(() => {
clearTimeout(timer);
return false;
}, 3000)
}
}
},
methods: {
init() {
this.mapModel = {
markerLatlng: [{ lat: 40.77460069719003, lng: 114.8868407358159 },
{ lat: 40.7746130606868, lng: 114.88729786752276 },
{ lat: 40.7746130606868, lng: 114.88739786752276 },
{ lat: 40.77465839357235, lng: 114.88736317180303 },]
}
}
}
};
</script>
<script module="mapModule" lang="renderjs">
import {
initScript,
getMap,
getLatLng,
getMultiMarker,
getGeometriesItem,
getFitBounds
} from '../../utils/tmap';
import {getPosition} from '../../utils/utils';
export default {
data() {
return {
ownerInstance: null,
map: null,
center: null,
marker: null,
infoWindow: null,
currPosition: {},
mapModel: {},
};
},
methods: {
async setPosition(){
this.currPosition = await getPosition()
this.center = getLatLng(this.currPosition.lat, this.currPosition.lng)
this.map.setCenter(this.center)
},
init(){
this.ownerInstance.ownerInstance('init')
},
async refresh() {
// 待处理
if(this.mapModel.markerLatlng.length){
const first = this.mapModel.markerLatlng[0]
this.center = getLatLng(first.lat, first.lng)
}
if(this.map){
getFitBounds(this.map, this.mapModel.markerLatlng)
}else{
this.map = getMap(this.center);
}
getFitBounds(this.map, this.mapModel.markerLatlng, 0.2)
this.marker = getMultiMarker(this.map, this.mapModel.markerLatlng.map((item, index )=> getGeometriesItem(item, 'marker'+index)))
this.marker.on('click', this.onMarker)
},
onMarker(e){
const markers = this.marker.geometries.map(item => item.id === e.geometry.id ? {...item, styleId: 'currMarker'} : {...item, styleId: 'marker'})
this.marker.updateGeometries(markers)
this.dealInfoWindow(e.geometry.position)
},
dealInfoWindow(center){
if(this.infoWindow){
this.infoWindow.close()
}
this.infoWindow = new TMap.InfoWindow({
map: this.map,
position: center,
offset: {x: 0, y: -80},
content: "<div><img src='https://mapapi.qq.com/web/lbs/javascriptGL/demo/img/em.jpg'></div>"
});
},
update(newValue, _, ownerInstance) {
this.mapModel = newValue
if (this.ownerInstance) {
this.refresh()
} else {
this.ownerInstance = ownerInstance;
initScript(this.init)
}
}
}
};
</script>
<style lang="scss" scoped>
.index-wrap {
height: 100vh;
width: 100%;
display: flex;
flex-direction: column;
justify-content: center;
.curr-position {
position: fixed;
bottom: 340rpx;
right: 32rpx;
z-index: 1001;
display: flex;
justify-content: center;
align-items: center;
background: #fff;
border-radius: 12rpx;
height: 80rpx;
width: 80rpx;
image {
display: inline-block;
width: 48rpx;
height: 48rpx;
}
}
.curr {
display: flex;
justify-content: center;
align-items: center;
flex: 1;
}
.map-container {
flex: 1;
width: 100%;
height: 100%;
z-index: 100;
display: block;
// transition: all 0 ease-in-out 0;
&.active {
display: none;
// transition: all 0 ease-in-out 0.3s;
}
// margin-top: 200rpx;
}
}
</style>
下面时renderjs中引入得utils.js和tmap.js:
utils.js
/**
* @description 通过uni.getLocation获取相关地理信息
* @returns
*/
export function getPosition() {
return new Promise((resolve) => {
uni.getLocation({
type: "gcj02",//"wgs84",
geocode: true,
timeout: 6,
fail: () => {
resolve({ ...getLocationLatLng(), errMsg: "getLocation:fail" });
},
success: (res) => {
const {province, city, district, street, streetNum, poiName} = res.address
const params = {
...res,
latitude: res.latitude,
longitude: res.longitude,
address_info: `${province}${city}${district}${street}${streetNum}`,
address: poiName,
};
uni.setStorageSync("LatLng", JSON.stringify(res));
resolve(params);
},
});
const timer = setTimeout(() => {
clearTimeout(timer);
resolve({ ...getLocationLatLng(), errMsg: "getLocation:fail" });
}, 1000);
});
}
tmap.js: (key值需要替换成你自己项目中地图的key)
import {icon1} from './mapIcon'
const key = '此处为你申请的腾讯地图得key'
export function getLocation() {
return new TMap.maps.Geolocation(key, "腾讯地图API示例")
}
export function initScript(callback) {
var script = document.createElement("script");
script.type = "text/javascript";
script.src = "https://map.qq.com/api/gljs?v=1.exp&key=" + key;
// script.src = "https://map.qq.com/api/js?v=2.exp&key=";
document.body.appendChild(script);
script.onload = () => {
callback()
}
}
export function getMap(center, id, zoom = 15) {
return new TMap.Map(document.getElementById(id || 'container'), {
center,//设置地图中心点坐标
zoom, //设置地图缩放级别
showControl: false,
})
}
/**
*
* @param {Number} lat 经度
* @param {Number} lng 维度
* @returns
*/
export function getLatLng(lat, lng) {
return new TMap.LatLng(lat, lng);
}
/**
* @description 获取路径数组
* @param {Array} pathList 经纬度数组
* @returns
*/
export function getPath(pathList) {
return pathList.map(item => getLatLng(item.lat, item.lng));
}
/**
* @description 地图设置会放路径
* @param {Object} map 地图对象
* @param {Array} pathList 路径数组
* @returns
*/
export function getPolylineLayer(map, pathList) {
return new TMap.MultiPolyline({
map, // 绘制到目标地图
// 折线样式定义
styles: {
style_blue: new TMap.PolylineStyle({
color: '#3777FF', // 线填充色
width: 4, // 折线宽度
borderWidth: 2, // 边线宽度
borderColor: '#FFF', // 边线颜色
lineCap: 'round', // 线端头方式
eraseColor: 'rgba(190,188,188,1)',
}),
},
geometries: [
{
id: 'erasePath',
styleId: 'style_blue',
paths: getPath(pathList),
},
],
})
}
/**
*
* @param {Object} LatLng 经纬度对象{lat,lng}
* @param {String} id 标注点Id
* @param {String} styleId 样式id
* @returns
*/
export function getGeometriesItem(LatLng, id, styleId = 'marker') {
return { //点标注数据数组
id,
styleId,
"position": getLatLng(LatLng.lat, LatLng.lng),
// "properties": {
// "title": "marker"
// }
}
}
export function getMarkerUrlByType(type) {
return 'static/mapImg/marker.png'
}
export function getMarkerStyle(key, src, width = 70, height = 70) {
return {
[key]: new TMap.MarkerStyle({
width,
height,
src,
})
}
}
export function getMultiMarker(map, geometries) {
return new TMap.MultiMarker({
id: "marker-layer", //图层id
map,
styles: { //点标注的相关样式
...getMarkerStyle('marker', icon1),
...getMarkerStyle('currMarker', icon1, 90, 90),
'car': new TMap.MarkerStyle({
width: 40,
height: 40,
anchor: {
x: 20,
y: 20,
},
faceTo: 'map',
rotate: 180,
src: icon1
// src: 'https://mapapi.qq.com/web/lbs/javascriptGL/demo/img/car.png',
})
},
geometries
});
}
function getfitBoundItemNum(base, add){
return base + add
}
/**
*
* @param {Object} map 地图对象
* @param {Array} latlngList 坐标点集合,示例:[{lat: 0, lng: 0}]
*/
export function getFitBounds(map, latlngList, scale = 1) {
let latlist = []
let lnglist = []
latlngList.forEach(item => {
latlist.push(item.lat)
lnglist.push(item.lng)
})
const latMax = Math.max(...latlist)
const latMin = Math.min(...latlist)
const lngMax = Math.max(...lnglist)
const lngMin = Math.min(...lnglist)
const latDiff = (latMax - latMin) * scale
const lngDiff = (lngMax - lngMin) * scale
const ne = getLatLng(getfitBoundItemNum(latMax, latDiff), getfitBoundItemNum(lngMax, lngDiff))
const sw = getLatLng(getfitBoundItemNum(latMin, -latDiff*2.5), getfitBoundItemNum(lngMin, -lngDiff))
const latLngBounds = new TMap.LatLngBounds(sw, ne)
map.fitBounds(latLngBounds); //根据指定的范围调整地图视野
}
mapIcon.js(图片base64字符串),注意:腾讯地图中图片地址必须是绝对地址,如果你们可以使用相对地址,欢迎留言讨论:
export const icon1 = ``