Leaflet 调用百度瓦片地图服务

在使用 leaflet 调用第三方瓦片地图服务的项目,主要谷歌地图、高德地图、百度地图和 OSM 地图,与其他三种地图对比,百度地图的瓦片组织方式是不同的。百度从中心点经纬度(0,0)度开始计算瓦片,而谷歌地图是从左上角经纬度(-180,90)度开始计算瓦片;如果直接使用百度瓦片地图服务会请求不到瓦片,因此需要转换一下。借助 leaflet-tileLayer-baidu 这个插件:

//需要引入 proj4.js 和 proj4leaflet.js 插件,使用script标签引入的方式
L.CRS.Baidu = new L.Proj.CRS(‘EPSG:900913‘, ‘+proj=merc +a=6378206 +b=6356584.314245179 +lat_ts=0.0 +lon_0=0.0 +x_0=0 +y_0=0 +k=1.0 +units=m [email protected] +wktext  +no_defs‘, {
    resolutions: function () {
        level = 19
        var res = [];
        res[0] = Math.pow(2, 18);
        for (var i = 1; i < level; i++) {
            res[i] = Math.pow(2, (18 - i))
        }
        return res;
    }(),
    origin: [0, 0],
    bounds: L.bounds([20037508.342789244, 0], [0, 20037508.342789244])
});

L.tileLayer.baidu = function (option) {
    option = option || {};

    var layer;
    var subdomains = ‘0123456789‘;
    switch (option.layer) {
        //单图层
        case "vec":
        default:
            //‘http://online{s}.map.bdimg.com/tile/?qt=tile&x={x}&y={y}&z={z}&styles=pl&b=0&limit=60&scaler=1&udt=20170525‘
            layer = L.tileLayer(‘http://online{s}.map.bdimg.com/onlinelabel/?qt=tile&x={x}&y={y}&z={z}&styles=‘ + (option.bigfont ? ‘ph‘ : ‘pl‘) + ‘&scaler=1&p=1‘, {
                name:option.name,subdomains: subdomains, tms: true
            });
            break;
        case "img_d":
            layer = L.tileLayer(‘http://shangetu{s}.map.bdimg.com/it/u=x={x};y={y};z={z};v=009;type=sate&fm=46‘, {
                name: option.name, subdomains: subdomains, tms: true
            });
            break;
        case "img_z":
            layer = L.tileLayer(‘http://online{s}.map.bdimg.com/tile/?qt=tile&x={x}&y={y}&z={z}&styles=‘ + (option.bigfont ? ‘sh‘ : ‘sl‘) + ‘&v=020‘, {
                name: option.name, subdomains: subdomains, tms: true
            });
            break;

        case "custom"://Custom 各种自定义样式
            //可选值:dark,midnight,grayscale,hardedge,light,redalert,googlelite,grassgreen,pink,darkgreen,bluish
            option.customid = option.customid || ‘midnight‘;
            layer = L.tileLayer(‘http://api{s}.map.bdimg.com/customimage/tile?&x={x}&y={y}&z={z}&scale=1&customid=‘ + option.customid, {
                name: option.name, subdomains: "012", tms: true
            });
            break;

        case "time"://实时路况
            var time = new Date().getTime();
            layer = L.tileLayer(‘http://its.map.baidu.com:8002/traffic/TrafficTileService?x={x}&y={y}&level={z}&time=‘ + time + ‘&label=web2D&v=017‘, {
                name: option.name, subdomains: subdomains, tms: true
            });
            break;

            //合并
        case "img":
            layer = L.layerGroup([
                L.tileLayer.baidu({ name: "底图", layer: ‘img_d‘, bigfont: option.bigfont }),
                L.tileLayer.baidu({ name: "注记", layer: ‘img_z‘, bigfont: option.bigfont })
            ]);
            break;
    }
    return layer;
};

初始化之后,就可以直接实例化生成调用百度瓦片地图服务的实例;

this.map = L.map(‘_map‘,{crs: L.CRS.Baidu, fullscreenControl: true}).setView(this.originPoint, this.originZoom);
            L.tileLayer.baidu({layer: ‘vec‘}, {maxZoom: this.maxZoom, minZoom: this.minZoom}).addTo(this.map);
/* 这里是VUE项目,使用了 fullscreenControl 全屏插件 */

需要注意:

1、地图切换:

如果只是实现高德、谷歌地图、天地图的Leaflet 切换,可以使用 leaflet插件 Leaflet.ChineseTmsProviders

2、坐标系系统:

国内使用的地图坐标都是经过设备返回的坐标(WGS84)加密的,一般是GCJ02(高德地图、谷歌地图中国区服务都是用这一标准);百度地图则在GCJ02的基础上又进行了二次加密,使用的BD09坐标;因此直接在地图显示原始点的位置信息会有几十米到几百米的误差。地图官方都提供了在线的坐标转换接口以及web服务api接口:以百度地图为例:

//百度地图坐标转换api
var points = [
    new BMap.Point(116.3786889372559,39.90762965106183),
    new BMap.Point(116.38632786853032,39.90795884517671),
    new BMap.Point(116.39534009082035,39.907432133833574),
    new BMap.Point(116.40624058825688,39.90789300648029),
    new BMap.Point(116.41413701159672,39.90795884517671)
]

var convertor = new BMap.Convertor();
convertor.translate(points, 1, 5, translateCallback)  //translateCallback 回调
// 百度地图在线转换接口地址
http://api.map.baidu.com/geoconv/v1/?coords=114.21892734521,29.575429778924&from=1&to=5&ak=你的密钥 //GET请求

百度地图支持单个点转换和批量点转换,同时支持 WGS84转BD09,和GCJ02转BD09坐标;

但是存在以下问题:

①、百度地图在线api 最多一次支持10个点,web服务api单次请求可批量解析100个坐标,对于海量点(几千到几万)来说,这样的处理速度显然是不够的;高德地图同样满足不了业务需求;

②、不支持逆向解析,例如BD09 没有对应的接口转GCJ02和WGS84;因此在地图切换的时候,使用百度地图拾取的点就无法在其他地图上准确显示(高德地图提供BD09到GCJ02的坐标转换,但业务使用的只是高德的瓦片服务);

可以使用 gcoord 插件解决上述问题,支持上述三种坐标系互转,且是同步的接口,海量数据无性能瓶颈。(因为逆向解析没有公开的算法,原始坐标只是无限接近,基本满足地图使用要求。)

3、海量点性能问题

使用百度地图点的数量在2000左右时候,无论是打点、还是缩放、平移都会影响用户体验了,海量点接口虽可以支持万级别的数据,但是不能自定义图标、不能显示label;这是要是DOM操作严重影响了性能,可以使用Leaflet.Canvas-Markers解决海量点性能问题。

4、国外坐标识别

百度在线的api转换接口 ,会智能判断点的坐标是否在国内还是在国外,如果判断在国外就直接返回数据;但如果使用 gcoord 就无法智能识别;需要手动写一个坐标判断识别库,并且需要国界线经纬度数据。由于精确显示国界线的经纬度数据十分庞大,因此随着点精度的提高,需要牺牲浏览器性能,(在线转会有点的限制,估计也是性能考究)。

判断一个点是否在国内,可以参考 判断一个点是否在多边形内部 - 射线法思路,结合国界线的经纬度数据,就可以判断点的坐标是否在国内。射线法判断算法:

function pointInChina(position = [31.172800343248,121.406021546488]) {
    if(position.length !== 2) {
        console.error(‘The argument is an array of longitude and latitude, with longitude first and latitude last‘)
        return;
    };
    if((position[0] < 3.85 || position[0] > 53.55) || (position[1] < 73.55) || position[1] > 135.08) {
        console.log(‘Must be abroad‘)
        return false;
    }function isInPolygon(checkPoint, polygonPoints) {
        var counter = 0;
        var i;
        var xinters;
        var p1, p2;
        var pointCount = polygonPoints.length;
        p1 = polygonPoints[0];
        for (i = 1; i <= pointCount; i++) {
            p2 = polygonPoints[i % pointCount];
            if (checkPoint[0] > Math.min(p1[0], p2[0]) && checkPoint[0] <= Math.max(p1[0], p2[0])) {
                if (checkPoint[1] <= Math.max(p1[1], p2[1])) {
                    if (p1[0] != p2[0]) {
                        xinters = (checkPoint[0] - p1[0]) * (p2[1] - p1[1]) / (p2[0] - p1[0]) + p1[1];
                        if (p1[1] == p2[1] || checkPoint[1] <= xinters) {
                            counter++;
                        }
                    }
                }
            }
            p1 = p2;
        }
        if (counter % 2 == 0) {
            return false;
        } else {
            return true;
        }
    }

    try {
        let a = isInPolygon(position, mainLand_array);  //大陆
        let b = isInPolygon(position, taiWan_array);  // 台北
        if(a || b ) {
            return true;
        }else {
            return false;
        }
    }catch(err){
        console.log(err);
    }
}

参考链接:

1、百度与谷歌地图瓦片组织方式对比

2、leaflet-tileLayer-baidu  作者还提供了一个 在线预览的地址

3、点在多边形内算法,JS判断一个点是否在一个复杂多边形的内部

原文地址:https://www.cnblogs.com/engeng/p/11880994.html

时间: 2024-10-02 18:54:38

Leaflet 调用百度瓦片地图服务的相关文章

Google Map API V3调用arcgis发布的瓦片地图服务

由于最近项目需要用到CAD制作的地图,但之前一直使用的是用谷歌离线瓦片地图的方式,怎么样把CAD图像地图一样有缩放,移动的功能放到网页显示成了难题, 原先的谷歌地图的代码难道就不能用了?重新写一套代码得多久呢,得花多少时间精力呢? 于是开始尝试了N种方法,最后找的可能算是最好的方法可以沿用现有的代码,方法是:首先要用ARCGIS软件,把CAD图转化为mxd文件(当然其中各种困难就不提了,而且转换后的效果不一定能和CAD完全一样), 最后请教的是学地理信息系统专业的朋友才搞定的.其次把mxd发布成

Leaflet调用腾讯地图

leaflet是不能直接调用腾讯地图的,原因是因为腾讯地图的瓦片图规则和谷歌之类的不一样,具体的差异可参见这篇文章:腾讯与百度地图瓦片规则分析.如果要在leaflet中使用腾讯地图或者任何规则不统一的地图做底图,必须专门对L.TileLayer写一个扩展才能实现,重写getTileUrl函数. 默认的getTileUrl方法是这样的,比较简单: getTileUrl: function (tilePoint) { return L.Util.template(this._url, L.exten

矢量瓦片地图服务发布后坐标系发生变化

发布矢量瓦片地图有两种方式: 通过ArcGIS Pro发布矢量瓦片(工具栏:Share->Publish Web Layer): 先制作.vtpk瓦片离线地图包,再通过ArcGIS Enterprise上传发布服务. 通过第二种方式发布服务后坐标系直接变成了投影坐标系.其中Tile不能为中文名,虽然很多时候服务发布是正常的,但偶尔会遇到服务发布后坐标系更改的情况,建议遇到命名的地方统一都为英文命名. 原文地址:https://www.cnblogs.com/yj2010/p/11533522.h

调用百度API地图

<link rel="stylesheet" href="http://api.map.baidu.com/library/DrawingManager/1.4/src/DrawingManager_min.css" /><script type="text/javascript">    function loadJScript() {        var script = document.createElement

openlayers3应用(二):加载百度离线瓦片地图

上一篇文章介绍了使用openlayers3加载百度在线地图,对某些项目或应用场景比如不允许上外网的单位,某些项目只针对一定区域地图加载应用,比如一个县的地图,可以采用下载百度瓦片地图,在服务器或者本机单独部署的方式进行. 本篇主要讲述如何使用openlayers3调用下载的百度离线瓦片地图.瓦片地图下载器,网上有很多,在此不做详细描述. Openlayers3加载离线百度瓦片地图,效果以及代码如下: 代码如下: <!DOCTYPE html> <html xmlns="http

快速搭建简单的LBS程序——地图服务

很多时候,我们的程序需要提供需要搭建基于位置的服务(LBS),本文这里简单的介绍一下其涉及的一些基本知识. 墨卡托投影 地图本身是一个三维图像,但在电脑上展示时,往往需要将其转换为二维的平面图形,需要通过投影的方式将三维空间中的点映射到二维空间中.地图投影需要建立地球表面点与投影平面点的一一对应关系. 我们经常使用的一种投影算法是墨卡托投影,大概做法就是先拿一个圆柱体使它的轴与地球自转轴重合,先把球面上的点投影到圆柱的侧面上,再把圆柱展开就得到长方形的地图了. 关于墨卡托投影可以更多信息可以参看

arcgis api for js入门开发系列二不同地图服务展示(含源代码)

上一篇介绍了arcgis api离线部署,这篇开始正式介绍arcgis api for js开发:想要学习webgis开发,首先得熟悉了解前端技术,比如界面布局设计的html+css,核心的是javascript(js),arcgis api就是js写的,就是说想要开发gis功能前提下,你得熟悉了解js,不然你连源代码都看不懂.在这里,推荐esri官网的arcgis api for js:https://developers.arcgis.com/javascript/3/jsapi/:里面详细

Node.js调用百度地图Web服务API的Geocoding接口进行点位反地理信息编码

(从我的新浪博客上搬来的,做了一些修改.) 最近迷上了node.js以及JavaScript.现在接到一个活,要解析一个出租车点位数据的地理信息.于是就想到使用Node.js调用百度地图API进行解析. 使用的库主要就是有fs.request. // 请求包 var fs = require('fs');var request = require('request'); // 设置百度API的参数var baiduApiKey = "cQV9U4QhamoOjg6rjdOTAQSiUMxxxxx

delphi 调用百度地图WEBSERVICE转换GPS坐标

百度地图的API说明 使用方法 第一步,申请密钥(ak),作为访问服务的依据: 第二步,按照请求参数说明拼写发送http请求的url,注意需使用第一步申请的ak: 第三步,接收返回的数据(json或者xml格式). 注:本接口支持回调. 服务地址 http://api.map.baidu.com/geoconv/v1/? 组成说明: 域名:http://api.map.baidu.com 服务名:geoconv 服务版本号:v1 服务参数说明 参数 含义 取值范围 是否必须 默认取值 coord