Android定位&地圖&導航——自定義公交路線代碼

来源:http://www.cnblogs.com/jerehedu/archive/2016/02/14/5188746.html
-Advertisement-
Play Games

一、問題描述 基於百度地圖實現檢索指定城市指定公交的交通路線圖,效果如圖所示 二、通用組件Application類,主要創建並初始化BMapManager public class App extends Application { static App mDemoApp; //百度MapAPI的管


一、問題描述

基於百度地圖實現檢索指定城市指定公交的交通路線圖,效果如圖所示

二、通用組件Application類,主要創建並初始化BMapManager
    public class App extends Application {
    static App mDemoApp;
    //百度MapAPI的管理類
    public BMapManager mBMapMan = null;
    // 授權Key
    // 申請地址:http://dev.baidu.com/wiki/static/imap/key/
    public String mStrKey = "Your APPKey";
    boolean m_bKeyRight = true;    // 授權Key正確,驗證通過
    // 常用事件監聽,用來處理通常的網路錯誤,授權驗證錯誤等
    public static class MyGeneralListener implements MKGeneralListener {
        @Override
        public void onGetNetworkState(int iError) {
            Log.d("MyGeneralListener", "onGetNetworkState error is "+ iError);
            Toast.makeText(App.mDemoApp.getApplicationContext(), "您的網路出錯啦!",
                    Toast.LENGTH_LONG).show();
        }
        @Override
        public void onGetPermissionState(int iError) {
            Log.d("MyGeneralListener", "onGetPermissionState error is "+ iError);
            if (iError ==  MKEvent.ERROR_PERMISSION_DENIED) {
                // 授權Key錯誤:
                Toast.makeText(App.mDemoApp.getApplicationContext(), 
                        "文件輸入正確的授權Key!",
                        Toast.LENGTH_LONG).show();
                App.mDemoApp.m_bKeyRight = false;
            }
        }
    }
    @Override
    public void onCreate() {
        Log.v("BMapApiDemoApp", "onCreate");
        mDemoApp = this;
        mBMapMan = new BMapManager(this);
        mBMapMan.init(this.mStrKey, new MyGeneralListener());
        mBMapMan.getLocationManager().setNotifyInternal(10, 5);
        
        super.onCreate();
    }

    @Override
    //app的退出之前調用mapadpi的destroy()函數,避免重覆初始化帶來的時間消耗
    public void onTerminate() {
        if (mBMapMan != null) {
            mBMapMan.destroy();
            mBMapMan = null;
        }
        super.onTerminate();
    }

}
三、編寫公交的路線圖層(CustomRouteOverLay)和圖標標識(CustomOverlayItem)

  CustomRouteOverLay組件擴展RouteOverlay:

主要公交、步行和駕車線路圖層,將公交、步行和駕車出行方案的路線及關鍵點顯示在地圖上,根據車輛路線的起點和終點進行駕車路線的檢索;

  CustomOverlayItem擴展ItemizedOverlay<OverlayItem>:

覆蓋物的集合類,使用這個類可以將地圖上具有相同屬性或者特性的坐標使用圖標標識出來,OverLayItem 這個類對象則是ItemizedOverLay中一個一個的Item對象 也就是每個坐標對應的覆蓋物

  CustomRouteOverLay類代碼:

public class CustomRouteOverLay extends RouteOverlay {
    public Activity ac;
    private MapView mapView;
    static ArrayList<View> overlayviews = new ArrayList<View>();
    public CustomRouteOverLay(Activity arg0, MapView arg1) {
        super(arg0, arg1);
        ac = arg0;
        mapView = arg1;
        // TODO Auto-generated constructor stub
    }

    @Override
    protected boolean onTap(int arg0) {
        // TODO Auto-generated method stub
        // return super.onTap(arg0);
        return true;
    }

    @Override
    public void setData(MKRoute arg0) {
        // TODO Auto-generated method stub
        super.setData(arg0);
        addHint(arg0);
    }
    
    
    public void addHints(MKRoute routes) {
        for (int i = 0; i < routes.getNumSteps(); i++) {
            Drawable marker = ac.getResources().getDrawable(R.drawable.pop); // 得到需要標在地圖上的資源
            marker.setBounds(0, 0, marker.getIntrinsicWidth(),
                    marker.getIntrinsicHeight()); // 為maker定義位置和邊界
            OverItemT overitem = new OverItemT(marker,ac, routes.getStep(i).getContent(),routes.getStep(i).getPoint());
//            OverlayItem over=new OverlayItem(routes.GET, null, null);
            mapView.getOverlays().add(overitem); // 添加ItemizedOverlay實例到mMapView
        }
        mapView.invalidate();
    }
    /**
     * 增加 指示路線
     * @param routes
     */
    public void addHint(MKRoute routes) {
        mapView.getOverlays().clear();// 先清空
//        mapView.removeAllViewsInLayout();
        View mPopView = ac.getLayoutInflater().inflate(R.layout.popview,
                null);
        for(int i=0;i<    overlayviews.size();i++){
            System.out.println("remove &"+i);
            mapView.removeViewInLayout(overlayviews.get(i));
            overlayviews.remove(i);
        }
    
        mapView.invalidate();
        // 添加ItemizedOverlay
        for (int i = 0; i < routes.getNumSteps(); i++) {

            Drawable marker = ac.getResources().getDrawable(R.drawable.pop); // 得到需要標在地圖上的資源
            marker.setBounds(0, 0, marker.getIntrinsicWidth(),
                    marker.getIntrinsicHeight()); // 為maker定義位置和邊界
            GeoPoint pt = routes.getStep(i).getPoint();// =
                                                        // routes.get(i).getPoint();
            if (i != 0 && i != routes.getNumSteps() - 1) {
                mPopView = ac.getLayoutInflater().inflate(R.layout.popview,
                        null);
                mapView.addView(mPopView, new MapView.LayoutParams(
                        LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, null,
                        MapView.LayoutParams.TOP_LEFT));
                mPopView.setVisibility(View.GONE);
                mapView.updateViewLayout(mPopView, new MapView.LayoutParams(
                        LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, pt,
                        MapView.LayoutParams.BOTTOM_CENTER));
                mPopView.setVisibility(View.VISIBLE);
                Button button = (Button) mPopView.findViewById(R.id.overlay_pop);
                button.setText(routes.getStep(i).getContent());
                overlayviews.add(mPopView);
                overlayviews.add(button);
            } else {
                //修改起始點和終點樣式-自定義
                mPopView = ac.getLayoutInflater().inflate(R.layout.popview,
                        null);
                mapView.addView(mPopView, new MapView.LayoutParams(
                        LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, null,
                        MapView.LayoutParams.TOP_LEFT));
                mPopView.setVisibility(View.GONE);
                mapView.updateViewLayout(mPopView, new MapView.LayoutParams(
                        LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT, pt,
                        MapView.LayoutParams.BOTTOM_CENTER));
                mPopView.setVisibility(View.VISIBLE);
                Button button = (Button) mPopView.findViewById(R.id.overlay_pop);
                button.offsetTopAndBottom(100);
                button.setTextColor(Color.BLUE);
                button.setBackgroundColor(Color.TRANSPARENT);
                button.setText(routes.getStep(i).getContent());
                overlayviews.add(mPopView);
                overlayviews.add(button);
            }
        }
    }

    class OverItemT extends ItemizedOverlay<OverlayItem> {

        private Drawable marker;
        private Context mContext;
        private GeoPoint p;
        private OverlayItem o;
        public OverItemT(Drawable marker, Context context, String title,GeoPoint p) {
            super(boundCenterBottom(marker));
            this.marker = marker;
            this.mContext = context;
            this.p = p;
            // 構造OverlayItem的三個參數依次為:item的位置,標題文本,文字片段
            o = new OverlayItem(p, title, title);
            populate(); // createItem(int)方法構造item。一旦有了數據,在調用其它方法前,首先調用這個方法
        }
        public void updateOverlay() {
            populate();
        }

        @Override
        public void draw(Canvas canvas, MapView mapView, boolean shadow) {
            // Projection介面用於屏幕像素坐標和經緯度坐標之間的變換
            Projection projection = mapView.getProjection();
            for (int index = size() - 1; index >= 0; index--) { // 遍歷mGeoList
                OverlayItem overLayItem = getItem(index); // 得到給定索引的item
                String title = overLayItem.getTitle();
                // 把經緯度變換到相對於MapView左上角的屏幕像素坐標
                Point point = projection.toPixels(overLayItem.getPoint(), null);
                // 可在此處添加您的繪製代碼
                Paint paintText = new Paint();
                paintText.setColor(Color.BLUE);
                paintText.setTextSize(15);
                canvas.drawText(title, point.x - 30, point.y, paintText); // 繪製文本
            }
            super.draw(canvas, mapView, shadow);
            // 調整一個drawable邊界,使得(0,0)是這個drawable底部最後一行中心的一個像素
            boundCenterBottom(marker);
        }

        @Override
        protected OverlayItem createItem(int i) {
            // TODO Auto-generated method stub
            return o;
        }

        @Override
        public int size() {
            // TODO Auto-generated method stub
            return 1;
        }

        @Override
        // 處理當點擊事件
        protected boolean onTap(int i) {
            // 更新氣泡位置,並使之顯示
            return true;
        }

        @Override
        public boolean onTap(GeoPoint arg0, MapView arg1) {
            // TODO Auto-generated method stub
            // 消去彈出的氣泡
            // ItemizedOverlayDemo.mPopView.setVisibility(View.GONE);
            return super.onTap(arg0, arg1);
        }
    }

}

CustomOverlayItem代碼:

public class CustomOverlayItem extends ItemizedOverlay<OverlayItem> {
    // private List<OverlayItem> GeoList = new ArrayList<OverlayItem>();
    private Context mContext;
    private OverlayItem overlay;
    boolean showtext;
//    private String title;
    private Drawable marker;

    public CustomOverlayItem(Drawable marker, Context context, GeoPoint p,
            String title,String sinppet, boolean showtext) {
        super(boundCenterBottom(marker));
        this.mContext = context;
        // 用給定的經緯度構造GeoPoint,單位是微度 (度 * 1E6)
//        point = p;
        this.showtext = showtext;
//        this.title = title;
        this.marker = marker;
        overlay = new OverlayItem(p, title, sinppet);
        populate(); // createItem(int)方法構造item。一旦有了數據,在調用其它方法前,首先調用這個方法
    }

    @Override
    protected OverlayItem createItem(int i) {
        return overlay;
    }

    @Override
    public int size() {
        return 1;
    }

    @Override
    public void draw(Canvas canvas, MapView mapView, boolean arg2) {
        // TODO Auto-generated method stub
        super.draw(canvas, mapView, arg2);
        // Projection介面用於屏幕像素坐標和經緯度坐標之間的變換
        Projection projection = mapView.getProjection();
        String title = overlay.getTitle();
        // 把經緯度變換到相對於MapView左上角的屏幕像素坐標
        Point point = projection.toPixels(overlay.getPoint(), null);
        // 可在此處添加您的繪製代碼
        Paint paintText = new Paint();
        Paint paint = new Paint();
        paint.setAlpha(255);
        paint.setColor(Color.DKGRAY);
        paint.setStrokeWidth(5);
        paintText.setColor(Color.BLUE);
        paintText.setTextSize(15);
//        canvas.drawCircle(point.x, point.y, 100, paint);
        canvas.drawText(title, point.x-30, point.y-50, paintText); // 繪製文本
        // 調整一個drawable邊界,使得(0,0)是這個drawable底部最後一行中心的一個像素
        boundCenterBottom(marker);
    }

    @Override
    // 處理當點擊事件
    protected boolean onTap(int i) {
        if (showtext)
            Toast.makeText(this.mContext, overlay.getTitle(), Toast.LENGTH_SHORT).show();
        return true;
    }
}
四、編寫主程式BuslineSearch,擴展MapActivity,實現地圖信息的顯示
public class BuslineSearch extends MapActivity {
    Button mBtnSearch = null; // 搜索按鈕
    MapView mMapView = null; // 地圖View
    MKSearch mSearch = null; // 搜索模塊,也可去掉地圖模塊獨立使用
    String mCityName = null;
    LocationListener loc_listener;
    App app = null;
    static boolean flag = false;
    static Thread thread;
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.buslinesearch);

        app = (App) this.getApplication();
        if (app.mBMapMan == null) {
            app.mBMapMan = new BMapManager(getApplication());
            app.mBMapMan.init(app.mStrKey, new App.MyGeneralListener());
        }
        app.mBMapMan.start();
        // 如果使用地圖SDK,請初始化地圖Activity
        super.initMapActivity(app.mBMapMan);
        mMapView = (MapView) findViewById(R.id.bmapView);
        mMapView.setBuiltInZoomControls(true);
        // 設置在縮放動畫過程中也顯示overlay,預設為不繪製
        mMapView.setDrawOverlayWhenZooming(true);
        mMapView.setBuiltInZoomControls(true);
        // 初始化搜索模塊,註冊事件監聽
        MapController mMapController = mMapView.getController(); // 得到mMapView的控制權,可以用它控制和驅動平移和縮放
        GeoPoint point = new GeoPoint((int) (39.915 * 1E6),
                (int) (116.404 * 1E6)); // 用給定的經緯度構造一個GeoPoint,單位是微度 (度 * 1E6)
        mMapController.setCenter(point); // 設置地圖中心點
        mMapController.setZoom(15); // 設置地圖zoom級別
        mSearch = new MKSearch();
        mSearch.init(app.mBMapMan, new MKSearchListener() {
            public void onGetPoiResult(MKPoiResult res, int type, int error) {
                // 錯誤號可參考MKEvent中的定義
                if (error != 0 || res == null) {
                    Toast.makeText(BuslineSearch.this, "抱歉,未找到結果",
                            Toast.LENGTH_LONG).show();
                    return;
                }
                // System.out.println(res.toString());
                // 找到公交路線poi node
                MKPoiInfo curPoi = null;
                int totalPoiNum = res.getNumPois();
                for (int idx = 0; idx < totalPoiNum; idx++) {
                    Log.d("busline", "the busline is " + idx);
                    curPoi = res.getPoi(idx);
                    if (2 == curPoi.ePoiType) {
                        break;
                    }
                }
                mSearch.busLineSearch(mCityName, curPoi.uid);
            }

            public void onGetDrivingRouteResult(MKDrivingRouteResult res,
                    int error) {
            }

            public void onGetTransitRouteResult(MKTransitRouteResult res,
                    int error) {
                res.getPlan(0).getDistance();
            }

            public void onGetWalkingRouteResult(MKWalkingRouteResult res,
                    int error) {
            }

            public void onGetAddrResult(MKAddrInfo res, int error) {
            }

            public void onGetBusDetailResult(MKBusLineResult result, int iError) {
                if (iError != 0 || result == null) {
                    Toast.makeText(BuslineSearch.this, "抱歉,未找到結果",
                            Toast.LENGTH_LONG).show();
                    return;
                }
                // result.getBusRoute().get
                // result.getBusRoute().getStart().toString();
                CustomRouteOverLay routeOverlay = new CustomRouteOverLay(
                        BuslineSearch.this, mMapView);
            
                routeOverlay.setData(result.getBusRoute());
                mMapView.getOverlays().clear();
                System.out.println(mMapView.getOverlays().size());
                mMapView.getOverlays().add(routeOverlay);
                mMapView.invalidate();
                mMapView.getController().animateTo(
                        result.getBusRoute().getStart());
            }

            @Override
            public void onGetSuggestionResult(MKSuggestionResult res, int arg1) {
                // TODO Auto-generated method stub

            }
        });

        // mLocationManager.requestLocationUpdates(listener);
        // 註冊定位事件
        loc_listener = new LocationListener() {

            @Override
            public void onLocationChanged(Location location) {
                if (location != null) {
                    String strLog = String.format("您當前的位置:\r\n" + "緯度:%f\r\n"
                            + "經度:%f", location.getLongitude(),
                            location.getLatitude());
                    flag = true;
                    Drawable marker = getResources()
                            .getDrawable(R.drawable.ic_launcher);
                    final GeoPoint p = new GeoPoint(
                            (int) (location.getLatitude() * 1E6),
                            (int) (location.getLongitude() * 1E6));
                    CustomOverlayItem item = new CustomOverlayItem(marker,
                            BuslineSearch.this, p, "我的位置", "", false);
                    mMapView.getOverlays().add(item);
                    mMapView.getController().animateTo(p);
                }
            }
        };
        // 設定搜索按鈕的響應
        mBtnSearch = (Button) findViewById(R.id.search);

        OnClickListener clickListener = new OnClickListener() {
            public void onClick(View v) {
                SearchButtonProcess(v);
            }
        };

        mBtnSearch.setOnClickListener(clickListener);
    }

    void SearchButtonProcess(View v) {
        if (mBtnSearch.equals(v)) {
            mMapView.getOverlays().clear();
            mMapView.getOverlays().removeAll(mMapView.getOverlays());
            mMapView.invalidate();
            EditText editCity = (EditText) findViewById(R.id.city);
            EditText editSearchKey = (EditText) findViewById(R.id.searchkey);
            mCityName = editCity.getText().toString();
            mSearch.poiSearchInCity(mCityName, editSearchKey.getText()
                    .toString());
        }
    }

    @Override
    protected void onPause() {
        if (null == app)
            app = (App) this.getApplication();
        app.mBMapMan.getLocationManager().removeUpdates(loc_listener);
        app.mBMapMan.stop();
        super.onPause();
    }

    @Override
    protected void onResume() {
        if (null == app)
            app = (App) this.getApplication();
        app.mBMapMan.start();
        super.onResume();
        app.mBMapMan.getLocationManager().requestLocationUpdates(loc_listener);// 定位
    }

    @Override
    protected boolean isRouteDisplayed() {
        // TODO Auto-generated method stub
        return false;
    }

    @Override
    protected void onDestroy() {
        // TODO Auto-generated method stub
        super.onDestroy();
    }
}

 

作者:傑瑞教育
出處:http://www.cnblogs.com/jerehedu/ 
版權聲明:本文版權歸傑瑞教育技有限公司和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。
技術咨詢:JRedu技術交流
 
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 如何自定義屬性的特性? 用對象.屬性的特性和自定義的屬性的特性有什麼區別? 它的四大特性 writable enumerable configable 有什麼區別? 先預習一個用對象.屬性定義 ,屬性的四大特性是以什麼方式呈現的。 這時個屬性的三大特性預設值都為true。 代碼演示: 1 <scri
  • 針對兩列佈局案例(左邊一列寬度固定,右邊一列寬度自適應),從而講解float屬性
  • 數據屬性: 數據屬性包含一個數據值的位置,在這個位置可以讀取和寫入值。 4個描述的行為特性: writable 表示能否修改屬性的值。預設為true Enumerable 表示能否過過for in迴圈返回屬性是否可以枚舉。 configuralbe 表示是否能過來delete刪除屬性從來重新定義屬性
  • 出場: 首先我們來說說為什麼需要label標簽,雖然我們已經知道有break,continue跳出迴圈,但如果是多重迴圈那麼它們就顯的無能為力了,所以就出現了label這個標簽來為我們服務。 我們先來看看單獨使用break的情況 1 for(var i=0;i<4;i++){ 2 for(var j
  • [在此處輸入文章標題] // JScript 文件 /* ================================================================== JS 公共函數 080827 =======================================
  • 1 <!doctype html> 2 <html> 3 <head> 4 <meta charset="utf-8"> 5 <title>無標題文檔</title> 6 <style> 7 .fontSize22 8 { 9 font-size:22px; 10 } 11 .fontWeight
  • ajax在用於非同步交互以來,一直被廣泛使用,其使用語法格式基本如下: 基本格式為$.ajxa({ type:"",數據傳送類型POST,GET url:"",處理地址 data:"",傳送數據, dataType:"",返回數據類型, success:function(){},成功並接收返回值, e
  • 作者: "@gzdaijie" 本文為作者原創,轉載請註明出處:http://www.cnblogs.com/gzdaijie/p/5187171.html 。 1.寫在前面 之前使用過有道雲筆記和為知筆記,後來偶然喜歡上用Markdown寫文檔。被Markdown的簡潔與大氣所折服,因此拋棄了有道
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...