uniapp使用地图开发app, renderjs使用方法及注意事项

上次提到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、接下来介绍一下

html 复制代码
<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):

javascript 复制代码
<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

javascript 复制代码
/**
 * @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)

javascript 复制代码
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字符串),注意:腾讯地图中图片地址必须是绝对地址,如果你们可以使用相对地址,欢迎留言讨论:

javascript 复制代码
export const icon1 = ``
相关推荐
cy玩具15 分钟前
点击评论详情,跳到评论页面,携带对象参数写法:
前端
清灵xmf43 分钟前
TypeScript 类型进阶指南
javascript·typescript·泛型·t·infer
小白学大数据1 小时前
JavaScript重定向对网络爬虫的影响及处理
开发语言·javascript·数据库·爬虫
qq_390161771 小时前
防抖函数--应用场景及示例
前端·javascript
334554321 小时前
element动态表头合并表格
开发语言·javascript·ecmascript
John.liu_Test1 小时前
js下载excel示例demo
前端·javascript·excel
Yaml42 小时前
智能化健身房管理:Spring Boot与Vue的创新解决方案
前端·spring boot·后端·mysql·vue·健身房管理
PleaSure乐事2 小时前
【React.js】AntDesignPro左侧菜单栏栏目名称不显示的解决方案
前端·javascript·react.js·前端框架·webstorm·antdesignpro
哟哟耶耶2 小时前
js-将JavaScript对象或值转换为JSON字符串 JSON.stringify(this.SelectDataListCourse)
前端·javascript·json
getaxiosluo2 小时前
react jsx基本语法,脚手架,父子传参,refs等详解
前端·vue.js·react.js·前端框架·hook·jsx