Newer
Older
DH_Apicture / public / static / libs / mapbox / extend / TrackLayer.js
@zhangqy zhangqy on 29 Nov 12 KB first commit
(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 = 1500
            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]));
        }
    }
})));