在Android開發中地圖和定位是很多軟件不可或缺的內容,這些特色功能也給人們帶來了很多方便。
首先介紹一下地圖包中的主要類:
MapController : 主要控制地圖移動,伸縮,以某個GPS坐標為中心,控制MapView中的view組件,管理Overlay,提供View的基本功能。使用多種地圖模式(地圖模式(某些城市可實時對交通狀況進行更新),衛星模式,街景模式)來查看Google Map。常用方法:animateTo(GeoPoint point) setCenter(GeoPoint point) setZoom(int zoomLevel) 等。
Mapview : 是用來顯示地圖的view, 它派生自android.view.ViewGroup。當MapView獲得焦點,可以控制地圖的移動和縮放。地圖可以以不同的形式來顯示出來,如街景模式,衛星模式等,通過setSatellite(boolean) setTraffic(boolean), setStreetView(boolean) 方法。
Overlay : 是覆蓋到MapView的最上層,可以擴展其ondraw接口,自定義在MapView中顯示一些自己的東西。MapView通過MapView.getOverlays()對Overlay進行管理。
Projection :MapView中GPS坐標與設備坐標的轉換(GeoPoint和Point)。
定位系統包中的主要類:
LocationManager:本類提供訪問定位服務的功能,也提供獲取最佳定位提供者的功能。另外,臨近警報功能也可以借助該類來實現。
LocationProvider:該類是定位提供者的抽象類。定位提供者具備周期性報告設備地理位置的功能。
LocationListener:提供定位信息發生改變時的回調功能。必須事先在定位管理器中注冊監聽器對象。
Criteria:該類使得應用能夠通過在LocationProvider中設置的屬性來選擇合適的定位提供者。
Geocoder:用於處理地理編碼和反向地理編碼的類。地理編碼是指將地址或其他描述轉變為經度和緯度,反向地理編碼則是將經度和緯度轉變為地址或描述語言,其中包含了兩個構造函數,需要傳入經度和緯度的坐標。getFromLocation方法可以得到一組關於地址的數組。
下面開始地圖定位實例的開發,在開發地圖前需要 獲取Android 地圖 API 密鑰 網上有很多資料,這裡就不再復述。
首先要在manifest.xml中設置全相應的權限和maps庫:
[html]
- <application
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name" >
- <activity
- android:label="@string/app_name"
- android:name=".MyMapActivity" >
- <intent-filter >
- <action android:name="android.intent.action.MAIN" />
-
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- <span style="color:#FF6666;">
- <uses-library android:name="com.google.android.maps" /></span>
- </application>
-
- <span style="color:#FF6666;"> <uses-permission android:name="android.permission.INTERNET" />
- <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
- <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" /></span>
在上面我標紅的千萬不要忘記。
layout下的main.xml:
[html]
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- android:orientation="vertical" >
-
- <com.google.android.maps.MapView
- android:id="@+id/mapview"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- android:apiKey="008uu0x2a7GWlK2LzCW872afBAPLhJ-U2R26Wgw"
- />
-
- </LinearLayout>
下面是核心代碼,重要的地方我做了注釋:
[html]
- public class MyMapActivity extends MapActivity {
- /** Called when the activity is first created. */
- private MapController mapController;
- private MapView mapView;
- private MyOverLay myOverLay;
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
-
- LocationManager locationManager=(LocationManager) getSystemService(Context.LOCATION_SERVICE);
- mapView=(MapView) this.findViewById(R.id.mapview);
- //設置交通模式
- mapView.setTraffic(true);
- //設置衛星模式
- mapView.setSatellite(false);
- //設置街景模式
- mapView.setStreetView(false);
- //設置縮放控制
- mapView.setBuiltInZoomControls(true);
- mapView.setClickable(true);
- mapView.setEnabled(true);
- //得到MapController實例
- mapController=mapView.getController();
- mapController.setZoom(15);
-
- myOverLay=new MyOverLay();
- List<Overlay> overLays=mapView.getOverlays();
- overLays.add(myOverLay);
-
- Criteria criteria=new Criteria();
- criteria.setAccuracy(Criteria.ACCURACY_FINE);
- criteria.setAltitudeRequired(false);
- criteria.setBearingRequired(false);
- criteria.setCostAllowed(false);
- criteria.setPowerRequirement(Criteria.POWER_LOW);
- //取得效果最好的Criteria
- String provider=locationManager.getBestProvider(criteria, true);
- //得到Location
- Location location=locationManager.getLastKnownLocation(provider);
- updateWithLocation(location);
- //注冊一個周期性的更新,3秒一次
- locationManager.requestLocationUpdates(provider, 3000, 0, locationListener);
-
- }
- @Override
- public boolean onCreateOptionsMenu(Menu menu) {
- // TODO Auto-generated method stub
- menu.add(0, 1, 1, "交通模式");
- menu.add(0,2,2,"衛星模式");
- menu.add(0,3,3,"街景模式");
-
- return super.onCreateOptionsMenu(menu);
- }
- @Override
- public boolean onOptionsItemSelected(MenuItem item) {
- // TODO Auto-generated method stub
- super.onOptionsItemSelected(item);
- switch (item.getItemId()) {
- case 1://交通模式
- mapView.setTraffic(true);
- mapView.setSatellite(false);
- mapView.setStreetView(false);
- break;
- case 2://衛星模式
- mapView.setSatellite(true);
- mapView.setStreetView(false);
- mapView.setTraffic(false);
- break;
- case 3://街景模式
- mapView.setStreetView(true);
- mapView.setTraffic(false);
- mapView.setSatellite(false);
- break;
- default:
- mapView.setTraffic(true);
- mapView.setSatellite(false);
- mapView.setStreetView(false);
- break;
- }
- return true;
- }
- private void updateWithLocation(Location location){
- if(location!=null){
- //為繪制類設置坐標
- myOverLay.setLocation(location);
- GeoPoint geoPoint=new GeoPoint((int)(location.getLatitude()*1E6), (int)(location.getLongitude()*1E6));
- //定位到指定的坐標
- mapController.animateTo(geoPoint);
- mapController.setZoom(15);
- }
- }
- private final LocationListener locationListener=new LocationListener() {
-
- @Override
- public void onStatusChanged(String provider, int status, Bundle extras) {
- // TODO Auto-generated method stub
-
- }
-
- @Override
- public void onProviderEnabled(String provider) {
- // TODO Auto-generated method stub
-
- }
-
- @Override
- public void onProviderDisabled(String provider) {
- // TODO Auto-generated method stub
-
- }
- //當坐標改變時出發此函數
- @Override
- public void onLocationChanged(Location location) {
- // TODO Auto-generated method stub
- updateWithLocation(location);
- }
- };
- class MyOverLay extends Overlay{
-
- private Location location;
- public void setLocation(Location location){
- this.location=location;
- }
-
- @Override
- public boolean draw(Canvas canvas, MapView mapView, boolean shadow,
- long when) {
- // TODO Auto-generated method stub
- super.draw(canvas, mapView, shadow);
- Paint paint=new Paint();
- Point myScreen=new Point();
- //將經緯度換成實際屏幕的坐標。
- GeoPoint geoPoint=new GeoPoint((int)(location.getLatitude()*1E6), (int)(location.getLongitude()*1E6));
- mapView.getProjection().toPixels(geoPoint, myScreen);
- paint.setStrokeWidth(1);
- paint.setARGB(255, 255, 0, 0);
- paint.setStyle(Paint.Style.STROKE);
- Bitmap bmp=BitmapFactory.decodeResource(getResources(), R.drawable.mypicture);
- //把這張圖片畫到相應的位置。
- canvas.drawBitmap(bmp, myScreen.x, myScreen.y,paint);
- canvas.drawText("天堂沒有路", myScreen.x, myScreen.y, paint);
- return true;
-
- }
- }
- @Override
- protected boolean isRouteDisplayed() {
- // TODO Auto-generated method stub
- return false;
- }
- @Override
- public boolean onKeyDown(int keyCode, KeyEvent event) {
- // TODO Auto-generated method stub
-
- if (keyCode == KeyEvent.KEYCODE_BACK) {
- AlertDialog.Builder builder = new AlertDialog.Builder(this);
- builder.setMessage("你確定退出嗎?")
- .setCancelable(false)
- .setPositiveButton("確定",
- new DialogInterface.OnClickListener() {
- public void onClick(DialogInterface dialog,
- int id) {
- MyMapActivity.this.finish();
- android.os.Process
- .killProcess(android.os.Process
- .myPid());
- android.os.Process.killProcess(android.os.Process.myTid());
- android.os.Process.killProcess(android.os.Process.myUid());
- }
- })
- .setNegativeButton("返回",
- new DialogInterface.OnClickListener() {
- public void onClick(DialogInterface dialog,
- int id) {
- dialog.cancel();
- }
- });
- AlertDialog alert = builder.create();
- alert.show();
- return true;
- }
-
- return super.onKeyDown(keyCode, event);
- }
- }
接下來看一下運行後效果:
可以放大縮小:
可是使用menu鍵,切換不同的模式:
上面是切換到了衛星模式。由於地圖需要耗費大量的網絡資源,如果網絡比較慢的話會等待很長時間。