baidu地图API叠加自定义图层(一)

百度地图API提供了叠加自定义图层的方法,地址如下:
官网例子:清华校园微观图地图
http://developer.baidu.com/map/jsdemo.htm#g0_2
API说明:
http://developer.baidu.com/map/reference/index.php?title=Class:%E5%9C%B0%E5%9B%BE%E5%9B%BE%E5%B1%82%E7%B1%BB/TileLayer

我在这里实现的是在baidu地图上叠加geoserver的WMS图层,代码如下:

<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
    <script type="text/javascript" src="http://api.map.baidu.com/api?v=2.0&ak=1nCQCnDr3Nt3GKDVeBmKGe2Y"></script>
    <script src="../JS/coordTransform.js"></script>
    <style type="text/css">
        body, html, #allmap {
            width: 100%;
            height: 100%;
            overflow: hidden;
            margin: 0;
        }
    </style>
    <title></title>
</head>
<body>
    <form id="form1" runat="server">
        <div id="divContainer" style="overflow: scroll">
            <div style="width: 100%; height: 600px; border: solid 1px red;" id="allmap"></div>
        </div>
    </form>
    <script type="text/javascript">
        var map;
        var icon = new BMap.Icon('img/center.gif', new BMap.Size(24, 24));
        var resolutions = [262144, 131072, 65536, 32768, 16384, 8192, 4096, 2048, 1024, 512, 256, 128, 64, 32, 16, 8, 4, 2, 1];

        function init() {
            map = new BMap.Map("allmap");
            map.centerAndZoom("佛山", 10);  // 初始化地图,设置中心点坐标和地图级别。
            map.enableScrollWheelZoom();

            map.addControl(new BMap.PanoramaControl());
            map.addControl(new BMap.NavigationControl());
            map.addControl(new BMap.ScaleControl());
            map.addControl(new BMap.ScaleControl({ anchor: BMAP_ANCHOR_BOTTOM_LEFT }));
            map.addControl(new BMap.MapTypeControl({ anchor: BMAP_ANCHOR_TOP_RIGHT }));

            /*
            *加载自定义图层
            *原理:
            *1)getTilesUrl返回一个瓦片的链接,因此用geoserver的WMS服务提供一个链接出来。
            *2)getTilesUrl传递了2个参数,分别是tileCoord:百度地图瓦片的xy,zoom:百度地图当前级别;
            *根据tileCoord、zoom以及百度地图的resolutions,计算出此百度地图瓦片的bbox,然后利用此bbox请求geoserver wms服务,实现图层叠加。
            3)因百度地图采用的是火星坐标 (GCJ-02)二次加密为百度经纬度坐标系(BD-09),其瓦片是百度经纬度坐标系(BD-09)基础上进行百度墨卡托投影后的平面坐标,因此geoserver中的图层数据需要预先进行坐标转换,
            推荐公瑾的工具进行坐标转换:https://github.com/FreeGIS/postgis_LayerTransform;
            我这里geoserver发布的图层是经过转换后的BD-09坐标,因此需要将百度平面坐标转为经纬度坐标;
            4)
            百度地图的resolutions是已知的,计算方法如下:
                for (var i = 0; i < 19; i++) {
                    resolutions[i] = Math.pow(2, 18 - i);
                }
                百度地图瓦片及比率尺的原理参考:http://www.cnblogs.com/cglNet/p/3443637.html
            */
            var tileLayer = new BMap.TileLayer({ isTransparentPng: true });
            tileLayer.getTilesUrl = function (tileCoord, zoom) {
                if (zoom >= 10) {
                    var x = tileCoord.x;
                    var y = tileCoord.y;
                    //console.log("X: " + x + " Y : " + y + " Z: " + zoom);

                    //方法一,通过baiduMap API提供的方法将平面坐标转成经纬度坐标
                    var PointConvert = new BaiduPointConvert(map);
                    var lonlat_0 = PointConvert.tileToLngLat(tileCoord);//瓦片左下角坐标;
                    var tileCoord2 = new Object();
                    tileCoord2.x = x + 1;
                    tileCoord2.y = y + 1;
                    var lonlat2_0 = PointConvert.tileToLngLat(tileCoord2);//瓦片右上角坐标;
                    var bbox = [lonlat_0.lng, lonlat_0.lat, lonlat2_0.lng, lonlat2_0.lat, ];//左下角与右上角坐标构成一个bbox;
                    console.log(bbox);

                    //下面这一段代码和上面的代码效果一模一样!!!利用的公瑾提供的一个js实现坐标转换,地址为:https://github.com/FreeGIS/coordtransform
                    var res = resolutions[zoom];
                    var tileWidth = 256;
                    var minx = x * tileWidth * res;
                    var miny = y * tileWidth * res;
                    var maxx = (x + 1) * tileWidth * res;
                    var maxy = (y + 1) * tileWidth * res;

                    var bottomLeft = coordtransform.bd_mkt2bd_wgs(minx, miny); //百度墨卡托坐标-》百度经纬度坐标
                    var topRight = coordtransform.bd_mkt2bd_wgs(maxx, maxy);
                    var bbox2 = [bottomLeft[0], bottomLeft[1], topRight[0], topRight[1]]; //计算出bbox
                    console.log(bbox2);
                    //根据geoserver WMS服务的规则设置URL
                    var url = 'http://localhost:8080/geoserver/fsum/wms?SERVICE=WMS&VERSION=1.1.1&REQUEST=GetMap&FORMAT=image/png&LAYERS=fsum:T_UM_MAP_DISTRICT_GEOMETRY&TRANSPARENT=true&tiled=true&SRS=EPSG:4326&WIDTH=256&HEIGHT=256&BBOX='
                    + bbox2.join(',');

                    //console.log(bbox.join(','));
                    return url;
                }
            }
            map.addTileLayer(tileLayer);
        }

        window.onload = function () {
            init();
        }

        //百度地图坐标转换
        var BaiduPointConvert = function (map) {
            this.map = map;
            //瓦片xy计算出经纬度坐标
            //step1: this.tileToPixel(pixel);百度地图瓦片大小为 256*256,根据 瓦片xy * 256计算出瓦片的像素坐标;
            //step2 : this.pixelToWorld(this.tileToPixel(pixel)) ; 将像素坐标转为平面坐标。
            //step3: this.worldToLngLat(this.pixelToWorld(this.tileToPixel(pixel))); 调用API提供的方法将平面坐标转为经纬度坐标;
            //API说明请参考:http://developer.baidu.com/map/reference/index.php?title=Class:%E5%9C%B0%E5%9B%BE%E7%B1%BB%E5%9E%8B%E7%B1%BB/Projection
            this.tileToLngLat = function (pixel) {
                return this.worldToLngLat(this.pixelToWorld(this.tileToPixel(pixel)));
            }
            this.lngLatToTile = function (lngLat) {
                return this.pixelToTile(this.worldToPixel(this.lngLatToWorld(lngLat)));
            }
            this.pixelToLngLat = function (pixel) {
                return this.worldToLngLat(this.pixelToWorld(pixel));
            }
            this.lngLatToPixel = function (lngLat) {
                return this.worldToPixel(this.lngLatToWorld(lngLat));
            }
            this.tileToPixel = function (pixel) {
                return new BMap.Pixel(pixel.x * 256,
                    pixel.y * 256);
            }
            this.pixelToWorld = function (pixel) {
                var zoom = this.map.getZoom();
                return new BMap.Pixel(pixel.x / Math.pow(2, zoom - 18),
                    pixel.y / Math.pow(2, zoom - 18));
            }
            this.worldToLngLat = function (pixel) {
                var projection = this.map.getMapType().getProjection();
                return projection.pointToLngLat(pixel)
            }
            this.pixelToTile = function (pixel) {
                return new BMap.Pixel(pixel.x / 256,
                    pixel.y / 256);
            }
            this.worldToPixel = function (pixel) {
                var zoom = this.map.getZoom();
                return new BMap.Pixel(pixel.x * Math.pow(2, zoom - 18),
                    pixel.y * Math.pow(2, zoom - 18));
            }
            this.lngLatToWorld = function (lngLat) {
                var projection = this.map.getMapType().getProjection();
                return projection.lngLatToPoint(lngLat)
            }
        }
</script>
</body>
</html>

以上代码实现了叠加geoserver的WMS图层,但是WMS是实时渲染的,对于数据经常变化的适用于WMS,对于数据不常变化的使用预处理好的切片更能提高效率,如何使用切片呢?且听下回分解。

推荐

F3Earth是一群志同道合的伙伴发起的国产Web 3D Engine项目,github地址:https://github.com/f3earth/f3earth , 目前正在开发中,DEMO已初具功能,期待更多朋友的参与。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容