一、问题描述
基于百度地图实现检索指定城市指定公交的交通路线图,效果如图所示
二、通用组件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(); } }
以上代码内容是针对Android百度地图自定义公交路线导航的相关知识,希望对大家有所帮助。
本文向大家介绍百度地图自定义控件分享,包括了百度地图自定义控件分享的使用技巧和注意事项,需要的朋友参考一下 废话不多说了,直接奉上代码; 上面代码就是个人正在使用的百度地图的自定义控件,希望大家能够喜欢。
我正在为Android制作基于谷歌地图服务的公共交通地图。地图应该包含很多标记(超过300个),它们应该在地图放大和缩小(缩放)时调整大小。现在标记只是互相重叠,有没有办法创建像这样的自定义标记? 我自己也试过,但没有成功。使用android-map-utils库(https://github.com/googlemaps/android-maps-utils)标记现在看起来更好了,但它们不能调整
本文向大家介绍Android自定义View弧线进度控件,包括了Android自定义View弧线进度控件的使用技巧和注意事项,需要的朋友参考一下 这个是一个以弧线为依托的进度控件,主要包括了两个圆弧、一个圆、一个文本。 当我们点击开始按钮的时候,会出现一个动画,逐渐的出现进度,好了,下面开始我们的编码。 新建一个类,继承自View,实现三个构造方法,接着定义变量,初始化变量的数据。代码如下: 可
本文向大家介绍IOS实现百度地图自定义大头针和气泡样式,包括了IOS实现百度地图自定义大头针和气泡样式的使用技巧和注意事项,需要的朋友参考一下 一、自定义大头针和气泡 二、气泡自定义内容 最简单,最直接的方法。。。 自定义一个 UIView 核心代码如下: 三、添加标注自定义气泡 1.首先实现添加多个标注和自定义气泡 添加自定义标注 arry 中放入标注(BMKPointAnnotation
我有一个(由Google Maps Android API v2提供),我试图实现的应该足够简单,就是简单地绘制一条弯曲的。 具体地说,我有一个s数组,我不想让它们以锐角连接,而是想让路线很好地圆滑,这样穿过点的线会沿着曲线,而不是直线和锐角。 现在,在旧的Google Maps API中,可以通过创建自定义的,覆盖,然后手动绘制到地图上(例如,使用自定义的和,以及所需的设置)。 不幸的是,在v2
纪念一下今年第三次面百度自驾岗失败🤣 -------- 大概 45 分钟左右,主要内容 前 25 分钟:实习经历,蔚来和 mmt 感知具体做的啥 后 20 分钟:基本都是八股 batch normalization 是否了解? 什么是过拟合,怎么解决? C++ map 和 unordered_map 底层,什么时候用? 传统的 lidar 点云处理算法有哪些?(这道没答好,只说了聚类) 平时 C