提交 6ec25b33 编写于 作者: X xzs02

fix: 解决map组件hidden属性无法控制隐藏状态的问题

上级 ad04afa5
<template> <template>
<uni-map <uni-map :id="id">
:id="id"
class="uni-map">
<div <div
ref="map" ref="map"
style="width: 100%; height: 100%; position: relative; overflow: hidden;"/> style="width: 100%; height: 100%; position: relative; overflow: hidden;"/>
...@@ -17,46 +15,8 @@ ...@@ -17,46 +15,8 @@
import { import {
subscriber subscriber
} from 'uni-mixins' } from 'uni-mixins'
import {
supportsPassive
} from 'uni-shared'
var maps var maps
/**
* 自定义气泡类
*/
var Callout = function (option) {
this.option = option = option || {}
var map = option.map
this.position = option.position
this.index = 1
this.visible = this.alwaysVisible = option.display === 'ALWAYS'
this.init()
if (map) {
this.setMap(map)
}
}
/**
* 自定义tap事件监听
*/
function onTap (dom, callback) {
var startTime = 0
var delayTime = 200
var x
var y
dom.addEventListener('touchstart', function ($event) {
startTime = Date.now()
var touch = $event.targetTouches[0]
x = touch.screenX
y = touch.screenY
}, supportsPassive ? { passive: true } : false)
dom.addEventListener('touchend', function ($event) {
var touch = $event.changedTouches[0]
if (Math.abs(touch.screenX - x) > 20 || Math.abs(touch.screenY - y) > 20 || Date.now() - startTime > delayTime) {
return
}
callback($event)
}, supportsPassive ? { passive: true } : false)
}
export default { export default {
name: 'Map', name: 'Map',
mixins: [subscriber], mixins: [subscriber],
...@@ -65,13 +25,13 @@ export default { ...@@ -65,13 +25,13 @@ export default {
type: String, type: String,
default: '' default: ''
}, },
longitude: { latitude: {
type: [String, Number], type: [String, Number],
default: 116.397720 default: 39.92
}, },
latitude: { longitude: {
type: [String, Number], type: [String, Number],
default: 39.903230 default: 116.46
}, },
scale: { scale: {
type: [String, Number], type: [String, Number],
...@@ -89,25 +49,25 @@ export default { ...@@ -89,25 +49,25 @@ export default {
return [] return []
} }
}, },
polyline: { includePoints: {
type: Array, type: Array,
default () { default () {
return [] return []
} }
}, },
circles: { polyline: {
type: Array, type: Array,
default () { default () {
return [] return []
} }
}, },
controls: { circles: {
type: Array, type: Array,
default () { default () {
return [] return []
} }
}, },
includePoints: { controls: {
type: Array, type: Array,
default () { default () {
return [] return []
...@@ -120,57 +80,28 @@ export default { ...@@ -120,57 +80,28 @@ export default {
}, },
data () { data () {
return { return {
/**
* 地图中心点
*/
center: { center: {
latitude: 39.903230, latitude: 116.46,
longitude: 116.397720 longitude: 116.46
}, },
/**
* 地图初始化完毕
*/
isMapReady: false, isMapReady: false,
/**
* 视野初始化完毕
*/
isBoundsReady: false, isBoundsReady: false,
markersSync: [], markersSync: [],
polylineSync: [], polylineSync: [],
circlesSync: [], circlesSync: [],
controlsSync: [], controlsSync: []
/**
* 地图原始对象
*/
map: null,
mapLocation: null,
mapLocationPosition: null
} }
}, },
watch: { watch: {
latitude (val) { latitude () {
if (val && this.longitude) { this.centerChange()
this.center.latitude = Number(val)
}
},
longitude (val) {
if (val && this.latitude) {
this.center.longitude = Number(val)
}
}, },
center: { longitude () {
handler (val) { this.centerChange()
if (this.map) {
this.mapReady(() => {
this.map.setCenter(new maps.LatLng(val.latitude, val.longitude))
})
}
},
deep: true
}, },
scale (val) { scale (val) {
this.mapReady(() => { this.mapReady(() => {
this.map.setZoom(Number(val)) this._map.setZoom(Number(val) || 16)
}) })
}, },
markers (val, old) { markers (val, old) {
...@@ -259,7 +190,6 @@ export default { ...@@ -259,7 +190,6 @@ export default {
}, },
mounted () { mounted () {
this.loadMap(() => { this.loadMap(() => {
this.createClass()
this.init() this.init()
}) })
}, },
...@@ -269,7 +199,6 @@ export default { ...@@ -269,7 +199,6 @@ export default {
this.removeCircles() this.removeCircles()
this.removeControls() this.removeControls()
this.removeLocation() this.removeLocation()
this.map = null
}, },
methods: { methods: {
_handleSubscribe ({ _handleSubscribe ({
...@@ -293,7 +222,7 @@ export default { ...@@ -293,7 +222,7 @@ export default {
this.mapReady(() => { this.mapReady(() => {
var latitude var latitude
var longitude var longitude
var center = this.map.getCenter() var center = this._map.getCenter()
latitude = center.getLat() latitude = center.getLat()
longitude = center.getLng() longitude = center.getLng()
...@@ -304,13 +233,12 @@ export default { ...@@ -304,13 +233,12 @@ export default {
}) })
break break
case 'moveToLocation': case 'moveToLocation':
var locationPosition = this.mapLocationPosition var locationPosition = this._locationPosition
if (locationPosition) { if (locationPosition) {
this.map.setCenter(locationPosition) this._map.setCenter(locationPosition)
} }
break break
case 'translateMarker': case 'translateMarker':
// 此处和小程序一致,不会同步数据到当前子组件
this.mapReady(() => { this.mapReady(() => {
try { try {
var marker = this.getMarker(data.markerId) var marker = this.getMarker(data.markerId)
...@@ -321,11 +249,8 @@ export default { ...@@ -321,11 +249,8 @@ export default {
var rotation = marker.getRotation() var rotation = marker.getRotation()
var a = marker.getPosition() var a = marker.getPosition()
var b = new maps.LatLng(destination.latitude, destination.longitude) var b = new maps.LatLng(destination.latitude, destination.longitude)
// 距离-千米
var distance = maps.geometry.spherical.computeDistanceBetween(a, b) / 1000 var distance = maps.geometry.spherical.computeDistanceBetween(a, b) / 1000
// 时间-小时
var time = ((typeof duration === 'number') ? duration : 1000) / (1000 * 60 * 60) var time = ((typeof duration === 'number') ? duration : 1000) / (1000 * 60 * 60)
// 速度-千米/小时
var speed = distance / time var speed = distance / time
var movingEvent = maps.event.addListener(marker, 'moving', e => { var movingEvent = maps.event.addListener(marker, 'moving', e => {
var latLng = e.latLng var latLng = e.latLng
...@@ -371,12 +296,11 @@ export default { ...@@ -371,12 +296,11 @@ export default {
}) })
break break
case 'includePoints': case 'includePoints':
// 此处和小程序一致,不会同步数据到当前子组件
this.fitBounds(data.points) this.fitBounds(data.points)
break break
case 'getRegion': case 'getRegion':
this.boundsReady(() => { this.boundsReady(() => {
var latLngBounds = this.map.getBounds() var latLngBounds = this._map.getBounds()
var southwest = latLngBounds.getSouthWest() var southwest = latLngBounds.getSouthWest()
var northeast = latLngBounds.getNorthEast() var northeast = latLngBounds.getNorthEast()
callback({ callback({
...@@ -402,7 +326,7 @@ export default { ...@@ -402,7 +326,7 @@ export default {
}, },
init () { init () {
var center = new maps.LatLng(this.center.latitude, this.center.longitude) var center = new maps.LatLng(this.center.latitude, this.center.longitude)
var map = this.map = new maps.Map(this.$refs.map, { var map = this._map = new maps.Map(this.$refs.map, {
center, center,
zoom: Number(this.scale), zoom: Number(this.scale),
scrollwheel: false, scrollwheel: false,
...@@ -461,19 +385,28 @@ export default { ...@@ -461,19 +385,28 @@ export default {
} }
if (this.includePoints && Array.isArray(this.includePoints) && this.includePoints.length) { if (this.includePoints && Array.isArray(this.includePoints) && this.includePoints.length) {
this.fitBounds(this.includePoints, () => { this.fitBounds(this.includePoints, () => {
// 为和小程序一致,重置中心点
map.setCenter(center) map.setCenter(center)
}) })
} }
// 创建mapContext对象
// 更改mapReady状态
this.isMapReady = true this.isMapReady = true
this.$emit('mapready') this.$emit('mapready')
}, },
centerChange () {
var latitude = Number(this.latitude)
var longitude = Number(this.longitude)
if (latitude !== this.center.latitude || longitude !== this.center.longitude) {
this.center.latitude = latitude
this.center.longitude = longitude
if (this._map) {
this.mapReady(() => {
this._map.setCenter(new maps.LatLng(latitude, longitude))
})
}
}
},
createMarkers (markerOptions) { createMarkers (markerOptions) {
var map = this.map var map = this._map
var markers = this.markersSync var markers = this.markersSync
// 创建新标注
markerOptions.forEach(option => { markerOptions.forEach(option => {
var marker = new maps.Marker({ var marker = new maps.Marker({
map, map,
...@@ -482,13 +415,11 @@ export default { ...@@ -482,13 +415,11 @@ export default {
}) })
marker.id = option.id marker.id = option.id
this.changeMarker(marker, option) this.changeMarker(marker, option)
// 监听标注事件
var mouseup var mouseup
maps.event.addListener(marker, 'mouseup', e => { maps.event.addListener(marker, 'mouseup', e => {
mouseup = Date.now() mouseup = Date.now()
}) })
maps.event.addListener(marker, 'click', e => { maps.event.addListener(marker, 'click', e => {
// 通过过滤暂时解决click事件重复触发的bug
if (Date.now() - mouseup < 20) { if (Date.now() - mouseup < 20) {
return return
} }
...@@ -512,7 +443,7 @@ export default { ...@@ -512,7 +443,7 @@ export default {
}) })
}, },
changeMarker (marker, option) { changeMarker (marker, option) {
var map = this.map var map = this._map
var title = option.title || option.name var title = option.title || option.name
var position = new maps.LatLng(option.latitude, option.longitude) var position = new maps.LatLng(option.latitude, option.longitude)
var img = new Image() var img = new Image()
...@@ -535,11 +466,9 @@ export default { ...@@ -535,11 +466,9 @@ export default {
y = (typeof y === 'number' ? y : 1) * h y = (typeof y === 'number' ? y : 1) * h
top = h - (h - y) top = h - (h - y)
icon = new maps.MarkerImage(img.src, null, null, new maps.Point(x, y), new maps.Size(w, h)) icon = new maps.MarkerImage(img.src, null, null, new maps.Point(x, y), new maps.Size(w, h))
// 修改标注-暂不支持透明度
marker.setPosition(position) marker.setPosition(position)
marker.setIcon(icon) marker.setIcon(icon)
marker.setRotation(option.rotate || 0) marker.setRotation(option.rotate || 0)
// 创建标注文本
var labelOpt = option.label || {} var labelOpt = option.label || {}
if (marker.label) { if (marker.label) {
marker.label.setMap(null) marker.label.setMap(null)
...@@ -565,7 +494,6 @@ export default { ...@@ -565,7 +494,6 @@ export default {
}) })
marker.label = label marker.label = label
} }
// 创建标注气泡
var calloutOpt = option.callout || {} var calloutOpt = option.callout || {}
var callout = marker.callout var callout = marker.callout
var calloutStyle var calloutStyle
...@@ -598,7 +526,14 @@ export default { ...@@ -598,7 +526,14 @@ export default {
if (callout) { if (callout) {
callout.setOption(calloutStyle) callout.setOption(calloutStyle)
} else { } else {
callout = marker.callout = new Callout(calloutStyle) callout = marker.callout = new maps.Callout(calloutStyle)
callout.onclick = function ($event) {
self.$trigger('callouttap', $event, {
markerId: option.id || 1
})
$event.stopPropagation()
$event.preventDefault()
}
} }
} else { } else {
if (callout) { if (callout) {
...@@ -607,13 +542,9 @@ export default { ...@@ -607,13 +542,9 @@ export default {
} }
} }
} }
console.log(option.iconPath) img.src = option.iconPath ? this.$getRealPath(option.iconPath) : ''
console.log(this.$getRealPath(option.iconPath))
img.src = this.$getRealPath(option.iconPath) ||
''
}, },
removeMarkers (markers) { removeMarkers (markers) {
// 销毁全部标注
for (var index = 0; index < markers.length; index++) { for (var index = 0; index < markers.length; index++) {
var marker = markers[index] var marker = markers[index]
if (marker.label) { if (marker.label) {
...@@ -627,10 +558,9 @@ export default { ...@@ -627,10 +558,9 @@ export default {
} }
}, },
createPolyline () { createPolyline () {
var map = this.map var map = this._map
var polyline = this.polylineSync var polyline = this.polylineSync
this.removePolyline() this.removePolyline()
// 创建新路线
this.polyline.forEach(option => { this.polyline.forEach(option => {
var path = [] var path = []
option.points.forEach(point => { option.points.forEach(point => {
...@@ -661,17 +591,15 @@ export default { ...@@ -661,17 +591,15 @@ export default {
}, },
removePolyline () { removePolyline () {
var polyline = this.polylineSync var polyline = this.polylineSync
// 销毁全部路线
polyline.forEach(line => { polyline.forEach(line => {
line.setMap(null) line.setMap(null)
}) })
polyline.splice(0, polyline.length) polyline.splice(0, polyline.length)
}, },
createCircles () { createCircles () {
var map = this.map var map = this._map
var circles = this.circlesSync var circles = this.circlesSync
this.removeCircles() this.removeCircles()
// 创建新圆形
this.circles.forEach(option => { this.circles.forEach(option => {
var center = new maps.LatLng(option.latitude, option.longitude) var center = new maps.LatLng(option.latitude, option.longitude)
...@@ -698,17 +626,15 @@ export default { ...@@ -698,17 +626,15 @@ export default {
}, },
removeCircles () { removeCircles () {
var circles = this.circlesSync var circles = this.circlesSync
// 销毁全部圆形
circles.forEach(circle => { circles.forEach(circle => {
circle.setMap(null) circle.setMap(null)
}) })
circles.splice(0, circles.length) circles.splice(0, circles.length)
}, },
createControls () { createControls () {
var map = this.map var map = this._map
var controls = this.controlsSync var controls = this.controlsSync
this.removeControls() this.removeControls()
// 创建新控件
this.controls.forEach(option => { this.controls.forEach(option => {
var position = option.position || {} var position = option.position || {}
var control = document.createElement('div') var control = document.createElement('div')
...@@ -732,37 +658,34 @@ export default { ...@@ -732,37 +658,34 @@ export default {
style.maxWidth = 'initial' style.maxWidth = 'initial'
} }
img.src = this.$getRealPath(option.iconPath) img.src = this.$getRealPath(option.iconPath)
onTap(img, $event => { img.onclick = function ($event) {
if (option.clickable) { if (option.clickable) {
this.$trigger('controltap', $event, { this.$trigger('controltap', $event, {
controlId: option.id controlId: option.id
}) })
} }
}) }
map.controls[maps.ControlPosition.TOP_LEFT].push(control) map.controls[maps.ControlPosition.TOP_LEFT].push(control)
controls.push(control) controls.push(control)
}) })
}, },
removeControls () { removeControls () {
var controls = this.controlsSync var controls = this.controlsSync
// 销毁全部控件
controls.forEach(control => { controls.forEach(control => {
control.remove() control.remove()
}) })
controls.splice(0, controls.length) controls.splice(0, controls.length)
}, },
createLocation () { createLocation () {
var map = this.map var map = this._map
var location = this.mapLocation var location = this._location
// 移除地点标注
if (location) { if (location) {
this.removeLocation() this.removeLocation()
} }
// 创建新地点标注
uni.getLocation({ uni.getLocation({
type: 'gcj02', type: 'gcj02',
success: (res) => { success: (res) => {
if (location !== this.mapLocation) { if (location !== this._location) {
return return
} }
var position = new maps.LatLng(res.latitude, res.longitude) var position = new maps.LatLng(res.latitude, res.longitude)
...@@ -775,7 +698,7 @@ export default { ...@@ -775,7 +698,7 @@ export default {
flat: true, flat: true,
rotation: 0 rotation: 0
}) })
this.mapLocation = location this._location = location
refreshLocation() refreshLocation()
uni.onCompassChange(function (res) { uni.onCompassChange(function (res) {
location.setRotation(res.direction) location.setRotation(res.direction)
...@@ -788,14 +711,14 @@ export default { ...@@ -788,14 +711,14 @@ export default {
var self = this var self = this
function refreshLocation () { function refreshLocation () {
if (location !== self.mapLocation) { if (location !== self._location) {
return return
} }
setTimeout(() => { setTimeout(() => {
uni.getLocation({ uni.getLocation({
type: 'gcj02', type: 'gcj02',
success: (res) => { success: (res) => {
var locationPosition = self.mapLocationPosition = new maps.LatLng(res.latitude, res.longitude) var locationPosition = self._locationPosition = new maps.LatLng(res.latitude, res.longitude)
location.setPosition(locationPosition) location.setPosition(locationPosition)
}, },
fail: e => { fail: e => {
...@@ -809,18 +732,17 @@ export default { ...@@ -809,18 +732,17 @@ export default {
} }
}, },
removeLocation () { removeLocation () {
var location = this.mapLocation var location = this._location
if (location) { if (location) {
location.setMap(null) location.setMap(null)
this.mapLocation = null this._location = null
this.mapLocationPosition = null this._locationPosition = null
uni.stopCompass() uni.stopCompass()
} }
}, },
fitBounds (points, cb) { fitBounds (points, cb) {
// 设置地图范围
this.boundsReady(() => { this.boundsReady(() => {
var map = this.map var map = this._map
var bounds = new maps.LatLngBounds() var bounds = new maps.LatLngBounds()
points.forEach(point => { points.forEach(point => {
...@@ -862,9 +784,37 @@ export default { ...@@ -862,9 +784,37 @@ export default {
} }
} }
}, },
createClass () { loadMap (callback) {
var self = this if (maps) {
callback()
} else if (window.qq && window.qq.maps) {
maps = window.qq.maps
callback()
} else {
let key = 'JMRBZ-R4HCD-X674O-PXLN4-B7CLH-42BSB'
let callbackName = '_callback' + Date.now()
window[callbackName] = function () {
delete window[callbackName]
maps = window.qq.maps
var Callout = maps.Callout = function (option = {}) {
this.option = option
var map = option.map
this.position = option.position
this.index = 1
this.visible = this.alwaysVisible = option.display === 'ALWAYS'
this.init()
Object.defineProperty(this, 'onclick', {
setter (callback) {
this.div.onclick = callback
},
getter () {
return this.div.onclick
}
})
if (map) {
this.setMap(map)
}
}
Callout.prototype = new maps.Overlay() Callout.prototype = new maps.Overlay()
Callout.prototype.init = function () { Callout.prototype.init = function () {
var option = this.option var option = this.option
...@@ -885,35 +835,22 @@ export default { ...@@ -885,35 +835,22 @@ export default {
divStyle.display = this.visible ? 'block' : 'none' divStyle.display = this.visible ? 'block' : 'none'
} }
div.appendChild(triangle) div.appendChild(triangle)
onTap(div, $event => {
self.$trigger('callouttap', $event, {
markerId: option.id || 1
})
$event.stopPropagation()
$event.preventDefault()
})
} }
// 定义construct,实现这个接口来初始化自定义的Dom元素
Callout.prototype.construct = function () { Callout.prototype.construct = function () {
var div = this.div var div = this.div
// 将dom添加到覆盖物层
var panes = this.getPanes() var panes = this.getPanes()
// 设置panes的层级,overlayMouseTarget可接收点击事件
panes.floatPane.appendChild(div) panes.floatPane.appendChild(div)
} }
// 实现draw接口来绘制和更新自定义的dom元素
Callout.prototype.draw = function () { Callout.prototype.draw = function () {
var overlayProjection = this.getProjection() var overlayProjection = this.getProjection()
if (!this.position || !this.div || !overlayProjection) { if (!this.position || !this.div || !overlayProjection) {
return return
} }
// 返回覆盖物容器的相对像素坐标
var pixel = overlayProjection.fromLatLngToDivPixel(this.position) var pixel = overlayProjection.fromLatLngToDivPixel(this.position)
var divStyle = this.div.style var divStyle = this.div.style
divStyle.left = pixel.x + 'px' divStyle.left = pixel.x + 'px'
divStyle.top = pixel.y + 'px' divStyle.top = pixel.y + 'px'
} }
// 实现destroy接口来删除自定义的Dom元素,此方法会在setMap(null)后被调用
Callout.prototype.destroy = function () { Callout.prototype.destroy = function () {
this.div.parentNode.removeChild(this.div) this.div.parentNode.removeChild(this.div)
this.div = null this.div = null
...@@ -932,7 +869,7 @@ export default { ...@@ -932,7 +869,7 @@ export default {
Callout.prototype.setStyle = function (option) { Callout.prototype.setStyle = function (option) {
var div = this.div var div = this.div
var divStyle = div.style var divStyle = div.style
div.innerHTML = option.content div.innerText = option.content
divStyle.lineHeight = (option.fontSize || 14) + 'px' divStyle.lineHeight = (option.fontSize || 14) + 'px'
divStyle.fontSize = (option.fontSize || 14) + 'px' divStyle.fontSize = (option.fontSize || 14) + 'px'
divStyle.padding = (option.padding || 8) + 'px' divStyle.padding = (option.padding || 8) + 'px'
...@@ -946,22 +883,9 @@ export default { ...@@ -946,22 +883,9 @@ export default {
this.position = position this.position = position
this.draw() this.draw()
} }
},
loadMap (callback) {
// 加载腾讯地图js
if ('qq' in window && typeof window.qq === 'object' && 'maps' in window.qq) {
maps = window.qq.maps
callback()
} else {
let key = 'JMRBZ-R4HCD-X674O-PXLN4-B7CLH-42BSB'
let callbackName = '_callback' + Date.now()
window[callbackName] = function () {
delete window[callbackName]
maps = window.qq.maps
callback() callback()
} }
let script = document.createElement('script') let script = document.createElement('script')
script.type = 'text/javascript'
script.src = `https://map.qq.com/api/js?v=2.exp&key=${key}&callback=${callbackName}&libraries=geometry` script.src = `https://map.qq.com/api/js?v=2.exp&key=${key}&callback=${callbackName}&libraries=geometry`
document.body.appendChild(script) document.body.appendChild(script)
} }
...@@ -977,4 +901,7 @@ uni-map { ...@@ -977,4 +901,7 @@ uni-map {
height: 150px; height: 150px;
display: block; display: block;
} }
uni-map[hidden] {
display: none;
}
</style> </style>
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册