(function (global, factory) { typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() : typeof define === 'function' && define.amd ? define(factory) : (global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.mapboxgl1.TrackLayer = factory()); }(window, (function () { const icon = '' const arrow = '' function base64ToImg(base64) { return new Promise((resolve, reject) => { const img = new Image() img.src = base64 img.onload = function () { resolve(img) } img.onerror = function (e) { reject(e) } }) } return class { constructor(map, json, fields = { lng: 'l', lat: 'a' },callback) { this._data = json || []; this._map = map; this._callback = callback; this._features = this._data.map(i => turf.point([i.l,i.a].map(Number),i)); this._json = turf.lineString(json.map(i => [i[fields.lng], i[fields.lat]].map(Number))); this._play = true; this.init(); } init() { if (this._data.length <= 1) return; this.destory(); const that = this that._index = 0 that._count = 5500 that._step = turf.length(that._json) / that._count that._flag = 0 that._playId = 'play-' + Date.now() // 添加路径图层 that._map.addSource(that._playId, { type: 'geojson', data: that._json }) that._map.addLayer({ id: that._playId, type: 'line', source: that._playId, 'layout': { 'line-cap': 'round', 'line-join': 'round' }, 'paint': { 'line-color': '#aaaaaa', 'line-width': 5 } }) // 添加已播放路径 that._map.addSource(that._playId + '-played', { type: 'geojson', data: that._json }) that._map.addLayer({ id: that._playId + '-played', type: 'line', source: that._playId + '-played', 'layout': { 'line-cap': 'round', 'line-join': 'round' }, 'paint': { 'line-color': '#09801a', 'line-width': 5 } }) // 添加路径上的箭头 base64ToImg(arrow).then(img => { that._map.addImage(that._playId + '-arrow', img) that._map.addLayer({ 'id': that._playId + '-arrow', 'source': that._playId, 'type': 'symbol', 'layout': { 'symbol-placement': 'line', 'symbol-spacing': 50, 'icon-image': that._playId + '-arrow', 'icon-size': 0.6, 'icon-allow-overlap': true } }) }) // 添加动态图标 base64ToImg(icon).then(img => { that._map.addImage(that._playId + '-icon', img) that._map.addSource(that._playId + '-point', { 'type': 'geojson', 'data': that._getDataByCoords() }) that._map.addLayer({ 'id': that._playId + '-point', 'source': that._playId + '-point', 'type': 'symbol', 'layout': { // 'icon-image': 'pz-point', 'icon-image': that._playId + '-icon', 'icon-size': 0.4, 'icon-allow-overlap': true, 'icon-rotation-alignment': 'map', 'icon-pitch-alignment': 'map', 'icon-rotate': 50 } }) that._animatePath() }) } _animatePath() { if (this._index > this._count) { window.cancelAnimationFrame(this._flag) } else { const coords = turf.along(this._json, this._step * this._index).geometry.coordinates // 已播放的线 const start = turf.along(this._json, 0).geometry.coordinates this._map.getSource(this._playId + '-played').setData(turf.lineSlice(start, coords, this._json)) // 车的图标位置 this._map.getSource(this._playId + '-point').setData(this._getDataByCoords(coords)) // 计算旋转角度 const nextIndex = this._index === this._count ? this._count - 1 : this._index + 1 const coordsNext = turf.along(this._json, this._step * nextIndex).geometry.coordinates let angle = turf.bearing(turf.point(coords), turf.point(coordsNext)) - 90 if (this._index === this._count) angle += 180 this._map.setLayoutProperty(this._playId + '-point', 'icon-rotate', angle) this._index++ const feature = turf.nearestPoint(this._getDataByCoords(coords),turf.featureCollection(this._features)); this._callback && this._callback(feature.properties,this._index); if (this._play) this._flag = requestAnimationFrame(() => { this._animatePath() }) } } _getDataByCoords(coords) { if (!coords || coords.length !== 2) return null return turf.point(coords, { 'label': this._formatDistance(this._step * this._index) }) } _formatDistance(dis) { if (dis < 1) { dis = dis * 1000 return dis.toFixed(0) + '米' } else { return dis.toFixed(2) + '千米' } } status(flag = true) { this._play = flag; this._animatePath() } restart() { this.init(); } show() { this.setLayerParams(this._playId + '-point', { visibility: 'visible' }); this.setLayerParams(this._playId + '-played', { visibility: 'visible' }); this.setLayerParams(this._playId, { visibility: 'visible' }); this.status(); } hide() { this.setLayerParams(this._playId + '-point', { visibility: 'none' }); this.setLayerParams(this._playId + '-played', { visibility: 'none' }); this.setLayerParams(this._playId, { visibility: 'none' }); this.status(false); } setOpacity(opacity) { this.setLayerParams(this._playId + '-point', {}, { 'raster-opacity': opacity }); this.setLayerParams(this._playId + '-played', {}, { 'raster-opacity': opacity }); this.setLayerParams(this._playId, {}, { 'raster-opacity': opacity }); } destory() { window.cancelAnimationFrame(this._flag); if (!(this._playId)) return; if (this._map.getSource(this._playId + '-point')) { this._map.removeLayer(this._playId + '-point') // this._map.removeLayer(this._playId + '-label') this._map.removeSource(this._playId + '-point') } if (this._map.getSource(this._playId)) { this._map.removeLayer(this._playId); this._map.removeLayer(this._playId + '-arrow'); this._map.removeSource(this._playId) } if (this._map.getSource(this._playId + '-played')) { this._map.removeLayer(this._playId + '-played') this._map.removeSource(this._playId + '-played') } this._popup && this._popup.remove(); } setLayerParams(id, layout = {}, paint = {}) { let layoutKeys = Object.keys(layout); let paintKeys = Object.keys(paint); layoutKeys.forEach(key => this._map.setLayoutProperty(id, key, layout[key])); paintKeys.forEach(key => this._map.setPaintProperty(id, key, paint[key])); } } })));