搜索
 找回密码
 立即注册

简单一步 , 微信登陆

百度地图开发(二)之添加覆盖物 + 地理编码和反地理编码

作者:liuwei | 时间:2016-9-28 12:24:07 | 阅读:3567| 只看该作者

    之前写过一篇关于百度地图开发的blog,主要介绍了百度地图的基本地图的显示。
    详见:Android百度地图开发(一)之初体验
    下面来看一下地图上覆盖物的添加,以及地理编码和反地理编码。
添加覆盖物
    在地图上添加覆盖物,一般需要以下几个步骤:
    1.定义坐标点,有可能是一个,有可能是多个(比如:多边形覆盖物)。
    2.构造OverlayOpti**(地图覆盖物选型基类)。
    3.在地图上添加覆盖物。
    4.添加相应的**事件。
    在API中可以看到,BaiDuMap类中有一个方法:                              
这个方法就是用来在地图上添加覆盖物的。此方法需要一个OverlayOpti**参数。
    进而查看这个类。从API中可以看到这是个抽象类,实例化它就必须用它的子类。api中已经把它的所有子类列出来了,分别是:ArcOpti**(弧线形覆盖物选型类)、CircleOpti**(圆形覆盖物选型类)、DotOpti**(圆点覆盖物选型类)、GroundOverlayOpti**(地形图图层选型类)、MarkerOpti**(标注覆盖物选型类)、PolygonOpti**(多边形选型类)、PolylineOpti**(折线选型类)、TextOpti**(文本选型类)。
MarkerOpti**--标注覆盖物。
    查看一下API,只需关注返回值是MarkerOpti**的方法即可。

    此时很简单就可以构造出一个MarkerOpti**对象了。此时需要注意的是:必须添加图标,不然会报错!

    添加图标的话,就可以使用icon(BitmapDescriptor icon)这个方法:

   BitmapDescriptor是一个bitmap描述信息类。但是在api中没有看到任何关于构造这个类的方法,只有一个回收对象的方法recycle()。其实在android中,提到bitmap,一般都会跟一个“xxxFactory” 工厂类联系到一起。所以才想百度api中也会有一个关于构造BitmapDescriptor的工厂类。果不其然,看到了一个BitmapDescriptorFactory这个类。     
    这个类中包括了很多构造bitmap描述信息类的静态方法:

    OK,有了这一步,就可以构造出一个BitmapDescriptor对象了,进而对marker添加一个图片。
[java] view plaincopy
1.   // 定义marker坐标点  
2.           LatLng point = new LatLng(latitude, longitude);  
3.     
4.           // 构建markerOption,用于在地图上添加marker  
5.           OverlayOpti** opti** = new MarkerOpti**()//  
6.                   .position(point)// 设置marker的位置  
7.                   .icon(bitmap)// 设置marker的图标  
8.                   .zIndex(9)// 設置marker的所在層級  
9.                   .draggable(true);// 设置手势拖拽  
10.          // 在地图上添加marker,并显示  
11.          marker1 = (Marker) bdMap.addOverlay(opti**);  
   接下来对marker添加**事件。一开始我猜想,应该是MarkerOpti**这个类中有一个 “setOnxxx” 这种形式的方法来绑定事件,但是没有找到。最后还是在BaiDuMap这个类下面找到了两个静态的接口。


    先看一下点击事件。onMarkerClickListener接口中只有一个函数:

    此时就需要你添加overlay的返回值(marker)了。实际上BaiDuMap的添加覆盖物的方法addOverlay(OverlayOpti** opti**)的返回值是Overlay。而Overlay恰好是一个好抽象类。

    它的子类分别是Arc、Circle、Dot、GroundOverlay、Marker、Polygon、Polyline、Text。
正好与verlayOption的子类一一对应。
    下面是点击事件的代码:
[java] view plaincopy
1.   bdMap.setOnMapClickListener(new OnMapClickListener() {  
2.     
3.               @Override  
4.               public boolean onMapPoiClick(MapPoi arg0) {  
5.                   return false;  
6.               }  
7.     
8.               @Override  
9.               public void onMapClick(LatLng latLng) {  
10.                  displayInfoWindow(latLng);  
11.              }  
12.          });  
   此时有一个问题,我得到了LatLng地理坐标这个对象了,怎么得到正常的地理信息呢(xx省xx市)?不要着急,慢慢往后看,反地理编码会解决这个问题的!
    下面看一下,拖拽事件(onMarkerDragListener):

    这三个方法有点类似于onTouch方法(按下,拖动,抬起)。
[java] view plaincopy
1.   bdMap.setOnMarkerDragListener(new OnMarkerDragListener() {  
2.               @Override  
3.               public void onMarkerDragStart(Marker arg0) {  
4.     
5.               }  
6.     
7.               @Override  
8.               public void onMarkerDragEnd(Marker arg0) {  
9.                   Toast.makeText(  
10.                          MainActivity.this,  
11.                          "拖拽结束,新位置:" + arg0.getPosition().latitude + ", "  
12.                                  + arg0.getPosition().longitude,  
13.                          Toast.LENGTH_LONG).show();  
14.                  reverseGeoCode(arg0.getPosition());  
15.              }  
16.   
17.              @Override  
18.              public void onMarkerDrag(Marker arg0) {  
19.   
20.              }  
21.          });  
   此时需要注意:拖拽事件,需要长按才能响应。
PolygonOpti**(多边形覆盖)
    继续查看API,查看这个类中的返回值是PolygonOpti**的方法:

    设置多边形覆盖物,当然需要多个地图上的点了。正好有一个points(List<LatLng> points)的方法用于设置多边形坐标点列表。OK,此时构造出一个PolygonOpti**就不难了。
[java] view plaincopy
1.   LatLng pt1 = new LatLng(latitude + 0.02, longitude);  
2.           LatLng pt2 = new LatLng(latitude, longitude - 0.03);  
3.           LatLng pt3 = new LatLng(latitude - 0.02, longitude - 0.01);  
4.           LatLng pt4 = new LatLng(latitude - 0.02, longitude + 0.01);  
5.           LatLng pt5 = new LatLng(latitude, longitude + 0.03);  
6.           List<LatLng> points = new ArrayList<LatLng>();  
7.           points.add(pt1);  
8.           points.add(pt2);  
9.           points.add(pt3);  
10.          points.add(pt4);  
11.          points.add(pt5);  
12.          //  
13.          PolygonOpti** polygonOpti** = new PolygonOpti**();  
14.          polygonOpti**.points(points);  
15.          polygonOpti**.fillColor(0xAAFFFF00);  
16.          polygonOpti**.stroke(new Stroke(2, 0xAA00FF00));  
17.          Overlay polygon = bdMap.addOverlay(polygonOpti**);  
TextOpti**(文字覆盖物)
    设置文字覆盖物的时候,需要注意文字的颜色,字体大小,位置等属性:


[java] view plaincopy
1.   LatLng latLng = new LatLng(latitude, longitude);  
2.           TextOpti** textOpti** = new TextOpti**();  
3.           textOpti**.bgColor(0xAAFFFF00)  //設置文字覆蓋物背景顏色  
4.               .fontSize(28)  //设置字体大小  
5.               .fontColor(0xFFFF00FF)// 设置字体颜色  
6.               .text("我在这里啊!!!!")  //文字内容  
7.               .rotate(-30)  //设置文字的旋转角度  
8.               .position(latLng);// 设置位置  
9.           bdMap.addOverlay(textOpti**);  
GroundOverlay(地形图图层覆盖物)
    地形图图层可以跟随地图进行平移,伸缩等变换,位于底图和标注图层之家,不会遮挡地图标注的信息。定义这个覆盖物的时候,需要指定宽高。这里百度API提供了两种方法:
1. 指定一个地理坐标(LatLng),在用dimensi**方法来指定宽度和高度。
2. 使用positionFromBounds(LagLngBounds bounds)方法。LatLngBounds方法表示一个地理范围,包括一个东北角坐标和一个西南角坐标,这样也能确定一个矩形。
    在LatLng中有一个静态内部类Builder--地理范围构造器。

[java] view plaincopy
1.   LatLng southwest = new LatLng(latitude - 0.01, longitude - 0.012);//西南  
2.           LatLng northeast = new LatLng(latitude + 0.01, longitude + 0.012);//东北  
3.           LatLngBounds bounds = new LatLngBounds.Builder().include(southwest)  
4.                   .include(northeast).build();//得到一个地理范围对象  
5.           BitmapDescriptor bitmap2 = BitmapDescriptorFactory  
6.               .fromResource(R.drawable.csdn_blog);  
7.           GroundOverlayOpti** groundOverlayOpti** = new GroundOverlayOpti**();  
8.           groundOverlayOpti**.image(bitmap2);//显示的图片  
9.           groundOverlayOpti**.positionFromBounds(bounds);//显示的位置  
10.          groundOverlayOpti**.transparency(0.7f);//显示的透明度  
11.          bdMap.addOverlay(groundOverlayOpti**);  
PolylineOpti**(折线覆盖物)
    添加折线与添加多边形的方法大致相同。这里就不多赘述了。

[java] view plaincopy
1.   CircleOpti** circleOpti** = new CircleOpti**();  
2.           circleOpti**.center(new LatLng(latitude, longitude));//设置圆心坐标  
3.           circleOpti**.fillColor(0XFFfaa755);//圆的填充颜色  
4.           circleOpti**.radius(150);//设置半径  
5.           circleOpti**.stroke(new Stroke(5, 0xAA00FF00));//设置边框  
6.           bdMap.addOverlay(circleOpti**);  
DotOpti**(圆点覆盖物)
[java] view plaincopy
1.   DotOpti** dotOpti** = new DotOpti**();  
2.           dotOpti**.center(new LatLng(latitude, longitude));//设置圆心坐标  
3.           dotOpti**.color(0XFFfaa755);//颜色  
4.           dotOpti**.radius(25);//设置半径  
5.           bdMap.addOverlay(dotOpti**);  
CircleOpti**(圆形(空心)覆盖物)
[java] view plaincopy
1.   CircleOpti** circleOpti** = new CircleOpti**();  
2.           circleOpti**.center(new LatLng(latitude, longitude));//设置圆心坐标  
3.           circleOpti**.fillColor(0XFFfaa755);//圆的填充颜色  
4.           circleOpti**.radius(150);//设置半径  
5.           circleOpti**.stroke(new Stroke(5, 0xAA00FF00));//设置边框  
6.           bdMap.addOverlay(circleOpti**);  
ArcOpti**(弧线覆盖物)
[java] view plaincopy
1.   LatLng pt1 = new LatLng(latitude, longitude - 0.01);  
2.           LatLng pt2 = new LatLng(latitude - 0.01, longitude - 0.01);  
3.           LatLng pt3 = new LatLng(latitude, longitude + 0.01);  
4.           ArcOpti** arcOpti** = new ArcOpti**();  
5.           arcOpti**.points(pt1, pt2, pt3);//设置弧线的起点、中点、终点坐标  
6.           arcOpti**.width(5);//线宽  
7.           arcOpti**.color(0xFF000000);  
8.           bdMap.addOverlay(arcOpti**);  
弹出窗覆盖物
    在百度地图上可以添加一种可以弹出的覆盖物,弹出的窗口布局可以自定义。


   API中说的很清楚,第二个构造函数,只能做显示用,没有响应事件。通过第一个构造方法,可以添加点击事件。
[java] view plaincopy
1.   /**
2.        * 显示弹出窗口覆盖物
3.        */  
4.       private void displayInfoWindow(final LatLng latLng) {  
5.           // 创建infowindow展示的view  
6.           Button btn = new Button(getApplicationContext());  
7.           btn.setBackgroundResource(R.drawable.popup);  
8.           btn.setText("点我点我~");  
9.           BitmapDescriptor bitmapDescriptor = BitmapDescriptorFactory  
10.                  .fromView(btn);  
11.          // infowindow点击事件  
12.          OnInfoWindowClickListener infoWindowClickListener = new OnInfoWindowClickListener() {  
13.              @Override  
14.              public void onInfoWindowClick() {  
15.                  reverseGeoCode(latLng);  
16.                  //隐藏InfoWindow  
17.                  bdMap.hideInfoWindow();  
18.              }  
19.          };  
20.          // 创建infowindow  
21.          InfoWindow infoWindow = new InfoWindow(bitmapDescriptor, latLng, -47,  
22.                  infoWindowClickListener);  
23.   
24.          // 显示InfoWindow  
25.          bdMap.showInfoWindow(infoWindow);  
26.      }  

地理编码与反地理编码
    地理编码指的是将地理信息转化成坐标关系的过程。分为正向的和反向的编码。
正向的就是指将地址信息转成坐标点的过程。比如:北京市**--> (123.23111, 123.23231)(我瞎写的)。反地理编码就是将地理坐标转换成具体的地址信息,通过百度的坐标定位引擎,插叙出坐标对应的物体所在的行政区划、街道等信息。
    所以我们,在地图上点击的时候,或者拖动marker的时候得到的LatLng对象,就可以通过反地理编码得到具体的地址了。
    实现起来也比较简单:
[java] view plaincopy
1.   // 创建地理编码检索实例  
2.           GeoCoder geoCoder = GeoCoder.newInstance();  
3.           //  
4.           OnGetGeoCoderResultListener listener = new OnGetGeoCoderResultListener() {  
5.               // 反地理编码查询结果回调函数  
6.               @Override  
7.               public void onGetReverseGeoCodeResult(ReverseGeoCodeResult result) {  
8.                   if (result == null  
9.                           || result.error != SearchResult.ERRORNO.NO_ERROR) {  
10.                      // 没有检测到结果  
11.                      Toast.makeText(MainActivity.this, "抱歉,未能找到结果",  
12.                              Toast.LENGTH_LONG).show();  
13.                  }  
14.                  Toast.makeText(MainActivity.this,  
15.                          "位置:" + result.getAddress(), Toast.LENGTH_LONG)  
16.                          .show();  
17.              }  
18.   
19.              // 地理编码查询结果回调函数  
20.              @Override  
21.              public void onGetGeoCodeResult(GeoCodeResult result) {  
22.                  if (result == null  
23.                          || result.error != SearchResult.ERRORNO.NO_ERROR) {  
24.                      // 没有检测到结果  
25.                  }  
26.              }  
27.          };  
28.          // 设置地理编码检索**者  
29.          geoCoder.setOnGetGeoCodeResultListener(listener);  
30.          //  
31.          geoCoder.reverseGeoCode(new ReverseGeoCodeOption().location(latLng));  
32.          // 释放地理编码检索实例  
33.          // geoCoder.destroy();  

转载请注明出处:http://blog.csdn.net/crazy1235/article/details/43377545

百度地图开发(二)之添加覆盖物 地理编码和反地理编码.rar

188.8 KB, 下载次数: 0

收藏
收藏0
分享
分享
点赞
点赞0
反对
反对0
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册
手机版