Android 百度地圖 SDK v3.0.0 (四) 引入離線地圖功能

yangxi_001發表於2014-09-23

轉載請標明出處:http://blog.csdn.net/lmj623565791/article/details/37758097

一直覺得地圖應用支援離線地圖很重要啊,我等移動2G屌絲,流量不易,且用且珍惜。

對於官方開發指南對於離線地圖的教程,提供了兩種方案:

第一,手動匯入,先將從官網下載的離線包解壓,把vmp資料夾拷入SD卡根目錄下的BaiduMapSDK資料夾內。好吧,我表示不能接受,無視了。

第二,介面下載方法如下:mOffline.start(cityid);還比較靠譜,就是沒詳細介紹。

今天,我們主要對第二種方式進行詳細介紹,然後整合到我們的已經整合了定位方向感測器的地圖中,如果你還不瞭解:Android百度地圖 SDK v3.0.0 (三) 新增覆蓋物Marker與InfoWindow的使用

效果圖:
為了方便,我又新增了個選單按鈕~可以看到能夠對下載位置的儲存,支援多個等待下載,已經取消下載等。最主要當然是,下載過後,只需要定位的流量(甚至不用)就能很好的使用我們們的地圖拉~
順便提一下:本來想搞個執行緒池,支援多個同時下載,這塊可能很多不注意會有一些問題,但是百度地圖公開出來的start(cityCode)不管我怎麼嘗試(嘗試了多個離線地圖例項都不行),每次同時都只能下載一個。
1、百度地圖離線相關API介紹
a 、類 MKOfflineMap 提供地圖的下載,離線地圖列表的獲取,已下載地圖的查詢等
java.util.ArrayList<MKOLUpdateElement> getAllUpdateInfo() 返回各城市離線地圖更新資訊
java.util.ArrayList<MKOLSearchRecord> getHotCityList() 返回熱門城市列表
java.util.ArrayList<MKOLSearchRecord> getOfflineCityList() 返回支援離線地圖城市列表
MKOLUpdateElement getUpdateInfo(int cityID) 返回指定城市ID離線地圖更新資訊
java.util.ArrayList<MKOLSearchRecord> searchCity(java.lang.String cityName) 根據城市名搜尋該城市離線地圖記錄
boolean init(MKOfflineMapListener listener) 初使化
boolean pause(int cityID) 暫停下載指定城市ID的離線地圖
boolean remove(int cityID) 刪除指定城市ID的離線地圖
boolean start(int cityID) 啟動下載指定城市ID的離線地圖
void destroy()  銷燬離線地圖管理模組,不用時呼叫 

介面 MKOfflineMapListener 離線地圖事件通知介面。
void onGetOfflineMapState(int type, int state)

類 MKOLUpdateElement 和 類 MKOLSearchRecord
基本就是包含一些cityName , cityId, size 等等

API在百度的幫助文件中也很詳細,下面會在程式碼中使用這些API。

2、離線地圖城市資訊的實體Bean
[java] view plaincopy
  1. package com.zhy.zhy_baidu_ditu_demo00;  
  2.   
  3. public class OfflineMapCityBean  
  4. {  
  5.     private String cityName;  
  6.     private int cityCode;  
  7.     /** 
  8.      * 下載的進度 
  9.      */  
  10.     private int progress;  
  11.       
  12.     private Flag flag = Flag.NO_STATUS;  
  13.     /** 
  14.      * 下載的狀態:無狀態,暫停,正在下載 
  15.      * @author zhy 
  16.      * 
  17.      */  
  18.     public enum Flag  
  19.     {  
  20.         NO_STATUS,PAUSE,DOWNLOADING  
  21.     }  
  22.   
  23.     public Flag getFlag()  
  24.     {  
  25.         return flag;  
  26.     }  
  27.   
  28.     public void setFlag(Flag flag)  
  29.     {  
  30.         this.flag = flag;  
  31.     }  
  32.   
  33.     public OfflineMapCityBean()  
  34.     {  
  35.     }  
  36.   
  37.     public OfflineMapCityBean(String cityName, int cityCode, int progress)  
  38.     {  
  39.         this.cityName = cityName;  
  40.         this.cityCode = cityCode;  
  41.         this.progress = progress;  
  42.     }  
  43.   
  44.     public String getCityName()  
  45.     {  
  46.         return cityName;  
  47.     }  
  48.   
  49.     public void setCityName(String cityName)  
  50.     {  
  51.         this.cityName = cityName;  
  52.     }  
  53.   
  54.     public int getCityCode()  
  55.     {  
  56.         return cityCode;  
  57.     }  
  58.   
  59.     public void setCityCode(int cityCode)  
  60.     {  
  61.         this.cityCode = cityCode;  
  62.     }  
  63.   
  64.     public int getProgress()  
  65.     {  
  66.         return progress;  
  67.     }  
  68.   
  69.     public void setProgress(int progress)  
  70.     {  
  71.         this.progress = progress;  
  72.     }  
  73.   
  74. }  

包含了,城市名稱:用於listview上的顯示,城市id:用於查詢下載情況,進度:更新listview下載時的顯示,標誌:使用者開啟或者取消下載時的標誌。

3、離線地圖的使用
在Actvity啟動時,首先初始化離線地圖
[java] view plaincopy
  1. /** 
  2.      * 初始化離線地圖 
  3.      */  
  4.     private void initOfflineMap()  
  5.     {  
  6.         mOfflineMap = new MKOfflineMap();  
  7.         // 設定監聽  
  8.         mOfflineMap.init(new MKOfflineMapListener()  
  9.         {  
  10.             @Override  
  11.             public void onGetOfflineMapState(int type, int state)  
  12.             {  
  13.                 switch (type)  
  14.                 {  
  15.                 case MKOfflineMap.TYPE_DOWNLOAD_UPDATE:  
  16.                     // 離線地圖下載更新事件型別  
  17.                     MKOLUpdateElement update = mOfflineMap.getUpdateInfo(state);  
  18.                     Log.e(TAG, update.cityName + " ," + update.ratio);  
  19.                     for (OfflineMapCityBean bean : mDatas)  
  20.                     {  
  21.                         if (bean.getCityCode() == state)  
  22.                         {  
  23.                             bean.setProgress(update.ratio);  
  24.                             bean.setFlag(Flag.DOWNLOADING);  
  25.                             break;  
  26.                         }  
  27.                     }  
  28.                     mAdapter.notifyDataSetChanged();  
  29.                     Log.e(TAG, "TYPE_DOWNLOAD_UPDATE");  
  30.                     break;  
  31.                 case MKOfflineMap.TYPE_NEW_OFFLINE:  
  32.                     // 有新離線地圖安裝  
  33.                     Log.e(TAG, "TYPE_NEW_OFFLINE");  
  34.                     break;  
  35.                 case MKOfflineMap.TYPE_VER_UPDATE:  
  36.                     // 版本更新提示  
  37.                     break;  
  38.                 }  
  39.   
  40.             }  
  41.         });  
  42.     }  
設定離線地圖的下載監聽介面,目前我們只關注type為MKOfflineMap.TYPE_DOWNLOAD_UPDATE , 此時傳入的state為cityId, 然後我們通過mOfflineMap.getUpdateInfo(state);可以獲得該城市的下載資料,接下來更新我們listview的資料集,最後重新整理介面。

初始化資料:
[java] view plaincopy
  1. private void initData()  
  2. {  
  3.   
  4.     // 獲得所有熱門城市  
  5.     ArrayList<MKOLSearchRecord> offlineCityList = mOfflineMap  
  6.             .getHotCityList();  
  7.     // 手動新增了西安  
  8.     MKOLSearchRecord xian = new MKOLSearchRecord();  
  9.     xian.cityID = 233;  
  10.     xian.cityName = "西安市";  
  11.     offlineCityList.add(xian);  
  12.     // 獲得所有已經下載的城市列表  
  13.     ArrayList<MKOLUpdateElement> allUpdateInfo = mOfflineMap  
  14.             .getAllUpdateInfo();  
  15.     // 設定所有資料的狀態  
  16.     for (MKOLSearchRecord record : offlineCityList)  
  17.     {  
  18.         OfflineMapCityBean cityBean = new OfflineMapCityBean();  
  19.         cityBean.setCityName(record.cityName);  
  20.         cityBean.setCityCode(record.cityID);  
  21.   
  22.         if (allUpdateInfo != null)//沒有任何下載記錄,返回null,為啥不返回空列表~~  
  23.         {  
  24.             for (MKOLUpdateElement ele : allUpdateInfo)  
  25.             {  
  26.                 if (ele.cityID == record.cityID)  
  27.                 {  
  28.                     cityBean.setProgress(ele.ratio);  
  29.                 }  
  30.             }  
  31.   
  32.         }  
  33.         mDatas.add(cityBean);  
  34.     }  
  35.   
  36. }  

進入Activity先通過mOfflineMap.getHotCityList();獲得熱門城市列表,不過熱門裡面竟然沒有西安,我手動增加了一個西安。(這裡我是為了方便,有興趣的可以列出全國支援的城市);接下來mOfflineMap.getAllUpdateInfo();獲得已經下載城市的資料資訊;然後就行交叉對比,設定資料來源資料。

初始化listview
[java] view plaincopy
  1. private void initListView()  
  2.     {  
  3.         mListView = (ListView) findViewById(R.id.id_offline_map_lv);  
  4.         mAdapter = new MyOfflineCityBeanAdapter();  
  5.         mListView.setAdapter(mAdapter);  
  6.   
  7.         mListView.setOnItemClickListener(new OnItemClickListener()  
  8.         {  
  9.             @Override  
  10.             public void onItemClick(AdapterView<?> parent, View view,  
  11.                     int position, long id)  
  12.             {  
  13.                 int cityId = mDatas.get(position).getCityCode();  
  14.                 if (mCityCodes.contains(cityId))  
  15.                 {  
  16.                     removeTaskFromQueue(position, cityId);  
  17.                 } else  
  18.                 {  
  19.                     addToDownloadQueue(position, cityId);  
  20.                 }  
  21.   
  22.             }  
  23.         });  
  24.     }  

為listview的item設定點選事件,第一次點選時加入下載佇列,第二次點選時取消下載。
listview的介面卡
[java] view plaincopy
  1. /** 
  2.      * 熱門城市地圖列表的Adapter 
  3.      *  
  4.      * @author zhy 
  5.      *  
  6.      */  
  7.     class MyOfflineCityBeanAdapter extends BaseAdapter  
  8.     {  
  9.   
  10.         @Override  
  11.         public boolean isEnabled(int position)  
  12.         {  
  13.             if (mDatas.get(position).getProgress() == 100)  
  14.             {  
  15.                 return false;  
  16.             }  
  17.             return super.isEnabled(position);  
  18.         }  
  19.   
  20.         @Override  
  21.         public int getCount()  
  22.         {  
  23.             return mDatas.size();  
  24.         }  
  25.   
  26.         @Override  
  27.         public Object getItem(int position)  
  28.         {  
  29.             return mDatas.get(position);  
  30.         }  
  31.   
  32.         @Override  
  33.         public long getItemId(int position)  
  34.         {  
  35.             return position;  
  36.         }  
  37.   
  38.         @Override  
  39.         public View getView(int position, View convertView, ViewGroup parent)  
  40.         {  
  41.             OfflineMapCityBean bean = mDatas.get(position);  
  42.             ViewHolder holder = null;  
  43.             if (convertView == null)  
  44.             {  
  45.                 holder = new ViewHolder();  
  46.                 convertView = mInflater.inflate(R.layout.offlinemap_item,  
  47.                         parent, false);  
  48.                 holder.cityName = (TextView) convertView  
  49.                         .findViewById(R.id.id_cityname);  
  50.                 holder.progress = (TextView) convertView  
  51.                         .findViewById(R.id.id_progress);  
  52.                 convertView.setTag(holder);  
  53.             } else  
  54.             {  
  55.                 holder = (ViewHolder) convertView.getTag();  
  56.             }  
  57.   
  58.             holder.cityName.setText(bean.getCityName());  
  59.             int progress = bean.getProgress();  
  60.             String progressMsg = "";  
  61.             // 根據進度情況,設定顯示  
  62.             if (progress == 0)  
  63.             {  
  64.                 progressMsg = "未下載";  
  65.             } else if (progress == 100)  
  66.             {  
  67.                 bean.setFlag(Flag.NO_STATUS);  
  68.                 progressMsg = "已下載";  
  69.             } else  
  70.             {  
  71.                 progressMsg = progress + "%";  
  72.             }  
  73.             // 根據當前狀態,設定顯示  
  74.             switch (bean.getFlag())  
  75.             {  
  76.             case PAUSE:  
  77.                 progressMsg += "【等待下載】";  
  78.                 break;  
  79.             case DOWNLOADING:  
  80.                 progressMsg += "【正在下載】";  
  81.                 break;  
  82.             default:  
  83.                 break;  
  84.             }  
  85.             holder.progress.setText(progressMsg);  
  86.             return convertView;  
  87.         }  
  88.   
  89.         private class ViewHolder  
  90.         {  
  91.             TextView cityName;  
  92.             TextView progress;  
  93.   
  94.         }  
  95.     }  

介面卡的程式碼比較簡單,主要就是getView中根本bean的資料設定顯示。
最後在我們主Activity增加一個選單項,開啟此Activity就完工了:
[java] view plaincopy
  1. @Override  
  2. public boolean onOptionsItemSelected(MenuItem item)  
  3. {  
  4.   
  5.     switch (item.getItemId())  
  6.     {  
  7.     case R.id.id_menu_map_offline:  
  8.         Intent intent = new Intent(MainActivity.this,  
  9.                     OfflineMapActivity.class);  
  10.         startActivity(intent);  
  11.         break;  
  12.     ...  
  13.     }     
  14. }     

好了,關於百度地圖,目前就用到這麼多~~如果後期有別的需求,還會繼續更新~


注:開發者key需要換成自己申請的,不清楚申請的請看第一篇部落格的。

相關文章