1. 程式人生 > >android 百度地圖系列之新增覆蓋物和覆蓋物的點選事件

android 百度地圖系列之新增覆蓋物和覆蓋物的點選事件

之前講了百度地圖定位和地圖基本操作,這篇部落格講一下,怎麼去給地圖新增覆蓋物,並當點選覆蓋物的時候顯示詳細資訊。
要給地圖新增覆蓋物,首先需要覆蓋物的經緯度,如果還要實現點選事件,顯示詳細的資訊,還需要覆蓋物的描述資訊(如圖片,位置名稱等),所以先新建一個實體類,來存放這些資訊。
實體類必須實現序列化介面

package com.zwinsoft.mybaidumap.entity;

import java.io.Serializable;

/**
 * 地圖標註資訊實體類
 * @author jing__jie
 *
 */
public class MarkerInfoUtil implements
Serializable{
private static final long serialVersionUID = 8633299996744734593L; private double latitude;//緯度 private double longitude;//經度 private String name;//名字 private int imgId;//圖片 private String description;//描述 //構造方法 public MarkerInfoUtil() {} public MarkerInfoUtil
(double latitude, double longitude, String name, int imgId, String description) { super(); this.latitude = latitude; this.longitude = longitude; this.name = name; this.imgId = imgId; this.description = description; } //toString方法 @Override public
String toString() { return "MarkerInfoUtil [latitude=" + latitude + ", longitude=" + longitude + ", name=" + name + ", imgId=" + imgId + ", description=" + description + "]"; } //getter setter public double getLatitude() { return latitude; } public void setLatitude(double latitude) { this.latitude = latitude; } public double getLongitude() { return longitude; } public void setLongitude(double longitude) { this.longitude = longitude; } public String getName() { return name; } public void setName(String name) { this.name = name; } public int getImgId() { return imgId; } public void setImgId(int imgId) { this.imgId = imgId; } public String getDescription() { return description; } public void setDescription(String description) { this.description = description; } }

在activity中新增幾條markerinfo資料,用來顯示marker

private void setMarkerInfo() {
    infos = new ArrayList<MarkerInfoUtil>();
    infos.add(new MarkerInfoUtil(117.216624,39.142693,"天津站",R.drawable.tianjinzhan,"天津站,俗稱天津東站,隸屬北京鐵路局管轄"));
    infos.add(new MarkerInfoUtil(117.176955,39.111345,"南開大學",R.drawable.nankai,"正式成立於1919年,是由嚴修、張伯苓秉承教育救國理念創辦的綜合性大學。"));
    infos.add(new MarkerInfoUtil(117.174081,39.094994,"天津水上公園",R.drawable.shuishang,"天津水上公園原稱青龍潭,1951年7月1日正式對遊客開放,有北方的小西子之稱。"));
    }

在activity_main.xml中,新增一個按鈕,當點選的時候顯示marker,再次點選的時候marker消失。

case R.id.btn_marker:
    if(!showMarker){
        //顯示marker
        addOverlay(infos);
        showMarker = true;
    }else{
        //關閉顯示marker
        mBaiduMap.clear();
        showMarker = false;
    }
    break;

//顯示marker
private void addOverlay(List<MarkerInfoUtil> infos2) {
    //清空地圖
    mBaiduMap.clear();
    //建立marker的顯示圖示
    BitmapDescriptor bitmap = BitmapDescriptorFactory.fromResource(R.drawable.ditu1);
    LatLng latLng = null;
    Marker marker;
    OverlayOptions options;
    for(MarkerInfoUtil info:infos){
        //獲取經緯度
        latLng = new LatLng(info.getLatitude(),info.getLongitude());
        //設定marker
        options = new MarkerOptions()
                .position(latLng)//設定位置
                .icon(bitmap)//設定圖示樣式
                .zIndex(9) // 設定marker所在層級
                .draggable(true); // 設定手勢拖拽;
        //新增marker
        marker = (Marker) mBaiduMap.addOverlay(options);
        //使用marker攜帶info資訊,當點選事件的時候可以通過marker獲得info資訊
        Bundle bundle = new Bundle();
        //info必須實現序列化介面
        bundle.putSerializable("info", info);
        marker.setExtraInfo(bundle);
    }
    //將地圖顯示在最後一個marker的位置
    MapStatusUpdate msu = MapStatusUpdateFactory.newLatLng(latLng);
    mBaiduMap.setMapStatus(msu);
}

這樣marker就顯示出來了。效果圖:
這裡寫圖片描述
下面需要新增marker的點選事件,使marker在點選的時候,顯示info中的資訊。首先需要一個佈局用來展示圖片,名稱等資訊。

<RelativeLayout
    android:id="@+id/rl_marker"
    android:layout_alignParentTop="true"
    android:layout_width="match_parent"
    android:layout_height="220dp" 
    android:background="#ffffff"
    android:visibility="gone"
    android:clickable="true"><!-- 如果不新增這個屬性,當點選佈局時,會和地圖點選事件干擾 -->
    <ImageView 
        android:id="@+id/iv_img"
        android:layout_width="match_parent"
        android:layout_height="150dp"
        android:layout_margin="10dp"/>
    <TextView 
        android:id="@+id/tv_name"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/iv_img"
        android:textSize="18sp"
        android:textColor="#000000"
        android:layout_marginBottom="5dp"/>
    <TextView 
        android:id="@+id/tv_description"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/tv_name"
        android:textSize="14sp"
        android:textColor="#000000"/>
</RelativeLayout>

新增marker的點選事件

//新增marker點選事件的監聽
mBaiduMap.setOnMarkerClickListener(new OnMarkerClickListener() {
    @Override
    public boolean onMarkerClick(Marker marker) {
        //從marker中獲取info資訊
        Bundle bundle = marker.getExtraInfo();
        MarkerInfoUtil infoUtil = (MarkerInfoUtil) bundle.getSerializable("info");
        //將資訊顯示在介面上
        ImageView iv_img = (ImageView)rl_marker.findViewById(R.id.iv_img);
        iv_img.setBackgroundResource(infoUtil.getImgId());
        TextView tv_name = (TextView)rl_marker.findViewById(R.id.tv_name);
        tv_name.setText(infoUtil.getName());
        TextView tv_description = (TextView)rl_marker.findViewById(R.id.tv_description);
        tv_description.setText(infoUtil.getDescription());
        //將佈局顯示出來
        rl_marker.setVisibility(View.VISIBLE);
        return true;
    }
});

效果圖:
這裡寫圖片描述
再新增一個地圖點選事件,當點選地圖的時候,將詳細資訊佈局影藏

//地圖點選事件
mBaiduMap.setOnMapClickListener(new OnMapClickListener() {
@Override
public boolean onMapPoiClick(MapPoi arg0) {
    return false;
}
@Override
public void onMapClick(LatLng arg0) {
    rl_marker.setVisibility(View.GONE);
}
});

其中點選顯示info資訊的佈局是固定的,有時候我們需要顯示一個infowindow,跟隨在marker上,這就需要百度地圖提供的InfoWindow來實現。
在marker的點選監聽中新增

//infowindow中的佈局
TextView tv = new TextView(MainActivity.this);
tv.setBackgroundResource(R.drawable.infowindow);
tv.setPadding(20, 10, 20, 20);
tv.setGravity(Gravity.CENTER);
tv.setTextColor(android.graphics.Color.WHITE);
tv.setText(infoUtil.getName());
bitmapDescriptor = BitmapDescriptorFactory.fromView(tv);
//infowindow位置
LatLng latLng = new LatLng(infoUtil.getLatitude(), infoUtil.getLongitude());
//infowindow點選事件
OnInfoWindowClickListener listener = new OnInfoWindowClickListener() {
    @Override
    public void onInfoWindowClick() {
        //隱藏infowindow
        mBaiduMap.hideInfoWindow();
    }
};
//顯示infowindow,-47是偏移量,使infowindow向上偏移,不會擋住marker
InfoWindow infoWindow = new InfoWindow(bitmapDescriptor, latLng, -47, listener);
mBaiduMap.showInfoWindow(infoWindow);

效果圖:
這裡寫圖片描述
下面總結一下MainActivity程式碼,包含地圖控制(地圖模式控制,地圖縮放控制),地圖定位,帶方向的定位,顯示覆蓋物,顯示覆蓋物點選事件等,希望大家多多指正。

package com.zwinsoft.mybaidumap;

import java.util.ArrayList;
import java.util.List;

import com.baidu.location.BDLocation;
import com.baidu.location.BDLocationListener;
import com.baidu.location.LocationClient;
import com.baidu.location.LocationClientOption;
import com.baidu.mapapi.SDKInitializer;
import com.baidu.mapapi.map.BaiduMap;
import com.baidu.mapapi.map.BaiduMap.OnMapClickListener;
import com.baidu.mapapi.map.BaiduMap.OnMapStatusChangeListener;
import com.baidu.mapapi.map.BaiduMap.OnMarkerClickListener;
import com.baidu.mapapi.map.BitmapDescriptor;
import com.baidu.mapapi.map.BitmapDescriptorFactory;
import com.baidu.mapapi.map.InfoWindow;
import com.baidu.mapapi.map.InfoWindow.OnInfoWindowClickListener;
import com.baidu.mapapi.map.MapPoi;
import com.baidu.mapapi.map.MapStatus;
import com.baidu.mapapi.map.MapStatusUpdate;
import com.baidu.mapapi.map.MapStatusUpdateFactory;
import com.baidu.mapapi.map.MapView;
import com.baidu.mapapi.map.Marker;
import com.baidu.mapapi.map.MarkerOptions;
import com.baidu.mapapi.map.MyLocationConfiguration;
import com.baidu.mapapi.map.MyLocationConfiguration.LocationMode;
import com.baidu.mapapi.map.MyLocationData;
import com.baidu.mapapi.map.OverlayOptions;
import com.baidu.mapapi.model.LatLng;
import com.zwinsoft.mybaidumap.entity.MarkerInfoUtil;
import com.zwinsoft.mybaidumap.tools.MyOrientationListener;
import com.zwinsoft.mybaidumap.tools.MyOrientationListener.OnOrientationListener;

import android.app.Activity;
import android.os.Bundle;
import android.view.Gravity;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.ImageButton;
import android.widget.ImageView;
import android.widget.RelativeLayout;
import android.widget.TextView;
import android.widget.Toast;

/**
 * 我的百度地圖首頁
 * @author jing__jie
 *
 */
public class MainActivity extends Activity implements OnClickListener {
    private MapView mMapView = null;
    private BaiduMap mBaiduMap;
    private ImageButton ib_large,ib_small,ib_mode,ib_loc,ib_traffic,ib_marker;
    //模式切換,正常模式
    private boolean modeFlag = true;
    //當前地圖縮放級別
    private float zoomLevel; 
    //定位相關
    private LocationClient mLocationClient;
    private MyLocationListener mLocationListener;
    //是否第一次定位,如果是第一次定位的話要將自己的位置顯示在地圖 中間
    private boolean isFirstLocation = true;
    //建立自己的箭頭定位
    private BitmapDescriptor bitmapDescriptor;
    //經緯度
    double mLatitude;
    double mLongitude;
    //方向感測器監聽
    private MyOrientationListener myOrientationListener;
    private float mLastX;
    private List<MarkerInfoUtil> infos;
    //顯示marker
    private boolean showMarker = false;
    private RelativeLayout rl_marker;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        //在使用SDK各元件之前初始化context資訊,傳入ApplicationContext  
        //注意該方法要再setContentView方法之前實現  
        SDKInitializer.initialize(getApplicationContext());  
        setContentView(R.layout.activity_main);  
        //初始化控制元件
        initView();
        //初始化地圖
        initMap();
        //定位
        initLocation();
        //建立自己的定點陣圖標,結合方向感測器,定位的時候顯示自己的方向
        initMyLoc();
        //建立marker資訊
        setMarkerInfo();
    }  
    private void setMarkerInfo() {
        infos = new ArrayList<MarkerInfoUtil>();
        infos.add(new MarkerInfoUtil(117.216624,39.142693,"天津站",R.drawable.tianjinzhan,"天津站,俗稱天津東站,隸屬北京鐵路局管轄"));
        infos.add(new MarkerInfoUtil(117.176955,39.111345,"南開大學",R.drawable.nankai,"正式成立於1919年,是由嚴修、張伯苓秉承教育救國理念創辦的綜合性大學。"));
        infos.add(new MarkerInfoUtil(117.174081,39.094994,"天津水上公園",R.drawable.shuishang,"天津水上公園原稱青龍潭,1951年7月1日正式對遊客開放,有北方的小西子之稱。"));
    }
    private void initMyLoc() {
        //初始化圖示
        bitmapDescriptor = BitmapDescriptorFactory.fromResource(R.drawable.arrow);
        //方向感測器監聽
        myOrientationListener = new MyOrientationListener(this);
        myOrientationListener.setOnOrientationListener(new OnOrientationListener() {

            @Override
            public void onOrientationChanged(float x) {
                mLastX = x;
            }
        });
    }
    private void initMap() {
        //獲取地圖控制元件引用  
        mMapView = (MapView) findViewById(R.id.bmapView);
        // 不顯示縮放比例尺
        mMapView.showZoomControls(false);
        // 不顯示百度地圖Logo
        mMapView.removeViewAt(1);
        //百度地圖
        mBaiduMap = mMapView.getMap();
        // 改變地圖狀態
        MapStatus mMapStatus = new MapStatus.Builder().zoom(15).build();
        MapStatusUpdate mMapStatusUpdate = MapStatusUpdateFactory.newMapStatus(mMapStatus);
        mBaiduMap.setMapStatus(mMapStatusUpdate);
        //設定地圖狀態改變監聽器
        mBaiduMap.setOnMapStatusChangeListener(new OnMapStatusChangeListener() {
            @Override
            public void onMapStatusChangeStart(MapStatus arg0) {
            }
            @Override
            public void onMapStatusChangeFinish(MapStatus arg0) {
            }
            @Override
            public void onMapStatusChange(MapStatus arg0) {
                //當地圖狀態改變的時候,獲取放大級別
                zoomLevel = arg0.zoom;
            }
        });
        //地圖點選事件
        mBaiduMap.setOnMapClickListener(new OnMapClickListener() {
            @Override
            public boolean onMapPoiClick(MapPoi arg0) {
                return false;
            }
            @Override
            public void onMapClick(LatLng arg0) {
                rl_marker.setVisibility(View.GONE);
            }
        });
    }
    private void initLocation() {
        //定位客戶端的設定
        mLocationClient = new LocationClient(this);
        mLocationListener = new MyLocationListener();
        //註冊監聽
        mLocationClient.registerLocationListener(mLocationListener);
        //配置定位
        LocationClientOption option = new LocationClientOption();
        option.setCoorType("bd09ll");//座標型別
        option.setIsNeedAddress(true);//可選,設定是否需要地址資訊,預設不需要
        option.setOpenGps(true);//開啟Gps
        option.setScanSpan(1000);//1000毫秒定位一次
        option.setIsNeedLocationPoiList(true);//可選,預設false,設定是否需要POI結果,可以在BDLocation.getPoiList裡得到
        mLocationClient.setLocOption(option);

    }
    private void initView() {
        //地圖控制按鈕
        ib_large = (ImageButton)findViewById(R.id.ib_large);
        ib_large.setOnClickListener(this);
        ib_small = (ImageButton)findViewById(R.id.ib_small);
        ib_small.setOnClickListener(this);
        ib_mode = (ImageButton)findViewById(R.id.ib_mode);
        ib_mode.setOnClickListener(this);
        ib_loc = (ImageButton)findViewById(R.id.ib_loc);
        ib_loc.setOnClickListener(this);
        ib_traffic = (ImageButton)findViewById(R.id.ib_traffic);
        ib_traffic.setOnClickListener(this);
        ib_marker = (ImageButton)findViewById(R.id.ib_marker);
        ib_marker.setOnClickListener(this);
        rl_marker = (RelativeLayout)findViewById(R.id.rl_marker);
    }
    @Override
    public void onClick(View v) {
        switch (v.getId()) {
        case R.id.ib_large:
            if (zoomLevel < 18) {
                mBaiduMap.setMapStatus(MapStatusUpdateFactory.zoomIn());
                ib_small.setEnabled(true);
            } else {
                showInfo("已經放至最大,可繼續滑動操作");
                ib_large.setEnabled(false);
            }
            break;
        case R.id.ib_small:
            if (zoomLevel > 6) {
                mBaiduMap.setMapStatus(MapStatusUpdateFactory.zoomOut());
                ib_large.setEnabled(true);
            } else {
                ib_small.setEnabled(false);
                showInfo("已經縮至最小,可繼續滑動操作");
            }
            break;
        case R.id.ib_mode://衛星模式和普通模式
            if(modeFlag){
                modeFlag = false;
                mBaiduMap.setMapType(BaiduMap.MAP_TYPE_SATELLITE);
                showInfo("開啟衛星模式");
            }else{
                modeFlag = true;
                mBaiduMap.setMapType(BaiduMap.MAP_TYPE_NORMAL);
                showInfo("開啟普通模式");
            }
            break;
        case R.id.ib_loc:
            isFirstLocation = true;
            showInfo("返回自己位置");
            break;
        case R.id.ib_traffic://是否開啟交通圖
            if(mBaiduMap.isTrafficEnabled()){
                mBaiduMap.setTrafficEnabled(false);
                ib_traffic.setBackgroundResource(R.drawable.offtraffic);
                showInfo("關閉實時交通圖");
            }else{
                mBaiduMap.setTrafficEnabled(true);
                ib_traffic.setBackgroundResource(R.drawable.ontraffic);
                showInfo("開啟實時交通圖");
            }
            break;
        case R.id.ib_marker:
            if(!showMarker){
                //顯示marker
                showInfo("顯示覆蓋物");
                addOverlay(infos);
                showMarker = true;
                ib_marker.setBackgroundResource(R.drawable.ditu4);
            }else{
                //關閉顯示marker
                showInfo("關閉覆蓋物");
                mBaiduMap.clear();
                showMarker = false;
                ib_marker.setBackgroundResource(R.drawable.ditu3);
            }
            break;
        default:
            break;
        }
    }  
    //顯示marker
    private void addOverlay(List<MarkerInfoUtil> infos) {
        //清空地圖
        mBaiduMap.clear();
        //建立marker的顯示圖示
        BitmapDescriptor bitmap = BitmapDescriptorFactory.fromResource(R.drawable.ditu1);
        LatLng latLng = null;
        Marker marker;
        OverlayOptions options;
        for(MarkerInfoUtil info:infos){
            //獲取經緯度
            latLng = new LatLng(info.getLatitude(),info.getLongitude());
            //設定marker
            options = new MarkerOptions()
                    .position(latLng)//設定位置
                    .icon(bitmap)//設定圖示樣式
                    .zIndex(9) // 設定marker所在層級
                    .draggable(true); // 設定手勢拖拽;
            //新增marker
            marker = (Marker) mBaiduMap.addOverlay(options);
            //使用marker攜帶info資訊,當點選事件的時候可以通過marker獲得info資訊
            Bundle bundle = new Bundle();
            //info必須實現序列化介面
            bundle.putSerializable("info", info);
            marker.setExtraInfo(bundle);
        }
        //將地圖顯示在最後一個marker的位置
        MapStatusUpdate msu = MapStatusUpdateFactory.newLatLng(latLng);
        mBaiduMap.setMapStatus(msu);
        //新增marker點選事件的監聽
        mBaiduMap.setOnMarkerClickListener(new OnMarkerClickListener() {
            @Override
            public boolean onMarkerClick(Marker marker) {
                //從marker中獲取info資訊
                Bundle bundle = marker.getExtraInfo();
                MarkerInfoUtil infoUtil = (MarkerInfoUtil) bundle.getSerializable("info");
                //將資訊顯示在介面上
                ImageView iv_img = (ImageView)rl_marker.findViewById(R.id.iv_img);
                iv_img.setBackgroundResource(infoUtil.getImgId());
                TextView tv_name = (TextView)rl_marker.findViewById(R.id.tv_name);
                tv_name.setText(infoUtil.getName());
                TextView tv_description = (TextView)rl_marker.findViewById(R.id.tv_description);
                tv_description.setText(infoUtil.getDescription());
                //將佈局顯示出來
                rl_marker.setVisibility(View.VISIBLE);

                //infowindow中的佈局
                TextView tv = new TextView(MainActivity.this);
                tv.setBackgroundResource(R.drawable.infowindow);
                tv.setPadding(20, 10, 20, 20);
                tv.setTextColor(android.graphics.Color.WHITE);
                tv.setText(infoUtil.getName());
                tv.setGravity(Gravity.CENTER);
                bitmapDescriptor = BitmapDescriptorFactory.fromView(tv);
                //infowindow位置
                LatLng latLng = new LatLng(infoUtil.getLatitude(), infoUtil.getLongitude());
                //infowindow點選事件
                OnInfoWindowClickListener listener = new OnInfoWindowClickListener() {
                    @Override
                    public void onInfoWindowClick() {
                        //隱藏infowindow
                        mBaiduMap.hideInfoWindow();
                    }
                };
                //顯示infowindow
                InfoWindow infoWindow = new InfoWindow(bitmapDescriptor, latLng, -47, listener);
                mBaiduMap.showInfoWindow(infoWindow);
                return true;
            }
        });
    }
    //自定義的定位監聽
    private class MyLocationListener implements BDLocationListener{
        @Override
        public void onReceiveLocation(BDLocation location) {
            //將獲取的location資訊給百度map
            MyLocationData data = new MyLocationData.Builder()  
                    .accuracy(location.getRadius())  
                    // 此處設定開發者獲取到的方向資訊,順時針0-360  
                    .direction(mLastX)
                    .latitude(location.getLatitude())  
                    .longitude(location.getLongitude())
                    .build();
            mBaiduMap.setMyLocationData(data);
            //更新經緯度
            mLatitude = location.getLatitude();
            mLongitude = location.getLongitude();
            //配置定點陣圖層顯示方式,使用自己的定點陣圖標
            MyLocationConfiguration configuration = new MyLocationConfiguration(LocationMode.NORMAL, true, bitmapDescriptor);
            mBaiduMap.setMyLocationConfigeration(configuration);
            if(isFirstLocation){
                //獲取經緯度
                LatLng ll = new LatLng(location.getLatitude(),location.getLongitude());
                MapStatusUpdate status = MapStatusUpdateFactory.newLatLng(ll);
                //mBaiduMap.setMapStatus(status);//直接到中間
                mBaiduMap.animateMapStatus(status);//動畫的方式到中間
                isFirstLocation = false;
                showInfo("位置:" + location.getAddrStr());
            }
        }

    }
    @Override
    protected void onStart() {
        super.onStart();
        //開啟定位
        mBaiduMap.setMyLocationEnabled(true);
        if(!mLocationClient.isStarted()){
            mLocationClient.start();
        }
        //開啟方向感測器
        myOrientationListener.start();
    }
    @Override
    protected void onStop() {
        super.onStop();
        //關閉定位
        mBaiduMap.setMyLocationEnabled(false);
        if(mLocationClient.isStarted()){
            mLocationClient.stop();
        }
        //關閉方向感測器
        myOrientationListener.stop();
    }
    @Override  
    protected void onDestroy() {  
        super.onDestroy();  
        //在activity執行onDestroy時執行mMapView.onDestroy(),實現地圖生命週期管理  
        mMapView.onDestroy();  
    }  
    @Override  
    protected void onResume() {  
        super.onResume();  
        //在activity執行onResume時執行mMapView. onResume (),實現地圖生命週期管理  
        mMapView.onResume();  
    }  
    @Override  
    protected void onPause() {  
        super.onPause();  
        //在activity執行onPause時執行mMapView. onPause (),實現地圖生命週期管理  
        mMapView.onPause();  
    }
    //顯示訊息
    private void showInfo(String str){
        Toast.makeText(MainActivity.this, str, Toast.LENGTH_SHORT).show();
    }
}