使用过百度地图的同学知道,它有个街景功能,可以看到许多地方的实景。这里就其街景内容的实现,进行下学习。

在百度地图SDK的官网上可以看到,百度对开发者提供了很多相关的内容,方便我们进行学习。关于SDK的使用方法,包括jar包导入,*.so 动态库的添加位置及AndroidManifest文件的配置不做为我们这里讨论的内容,官方文档已经介绍的很详细,不做无聊的搬运工。

效果图##

这里我们首先预览下,今天最终要实现的效果图

静态图1

静态图2

效果图

如图所示,我们这里的实现,就是两个页面的内容,一个是基础的地图MapView,一个是街景地图PanoView。接下来,就这两个页面(Activity)分别展开来说。(由于GIF图片大小限制,效果不是很理想,文章结尾有源码地址,可以自己跑一下看一下效果先)

地图MapView实现##

地图MapView的简单显示###

布局文件####

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical">

android:id="@+id/bmapView"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:clickable="true" />

Application####

一般情况下,我们的应用程序都会有一个继承自Application的类,用于实现一些初始化的方法,这里可以在Application里执行一些百度地图初始化的工作,这也是官方提倡的方式。

public void onCreate() {

super.onCreate();

mInstance = this;

SDKInitializer.initialize(getApplicationContext());

}

Activity####

在Activity的OnCreate方法中实现

setContentView(R.layout.activity_mapview);

mMapView = (MapView) findViewById(R.id.bmapView);

上面这样一段简单的代码,就可以在Activity中显示出一个MapView,也就是我们最熟悉的地图页面,是不是很简单,就像我们显示一个TextView一样。

这里说明写一下,按照官方的API指导文档,使用MapView等百度地图SDK所提供的各种实现,是需要去申请相关的key的,申请的方法在官网有着详细的介绍,这里就不再粘贴复制了;很多同学在使用MapView的时候发现,程序运行后地图没有显示,显示的都是一些方格子,这往往是由于key没有申请,或申请的方式不当造成的

MapView显示到当前位置###

每次打开百度地图,都会自动定位到我们当前所在的位置,或者是我们搜索某个特定的地方作为新的位置,整个地图所呈现的区域都是新位置周边的环境。这里,关于地图的定位和搜索的相关实现内容,就不展开来说,不当做此次的重点。

假设我们已通过定位(或者是搜索),定位了到了一个位置

**

* 假设我们当前的位置在此

*/

private final double latitude = 39.963175;

private final double longitude = 116.400244;

这个位置按照新闻里常听到的说法就是,东经116.40度,北纬39.96度,位于北京市东城区旧鼓楼大街丙1号。

接下来,我们要做的就是将MapView的视图更新到我们“定位”的位置,这个位置周边的地图才是我们关心的。

mBaiduMap = mMapView.getMap();

//定义Maker坐标点

point = new LatLng(latitude, longitude);

//定义地图状态

final MapStatus mMapStatus = new MapStatus.Builder()

.target(point)

.zoom(18)

.build();

//定义MapStatusUpdate对象,以便描述地图状态将要发生的变化

MapStatusUpdate mMapStatusUpdate =

MapStatusUpdateFactory.newMapStatus(mMapStatus);

//改变地图状态

mBaiduMap.setMapStatus(mMapStatusUpdate);

这里的mBaiduMap 是一个BaiduMap的实例,通过MapView的getMap方法即可获得。我们对地图的各种操作,设置属性都是基于这个实例进行。

通过上面的代码,我们就可以将MapView的视图更新到我们所想要的位置了。

添加View到MapView###

添加Marker####

按照百度地图API的说法,我们添加到地图上的小图标统一称为Marker。

//构建Marker图标

bitmap = BitmapDescriptorFactory

.fromResource(R.drawable.icon_markc);

//构建MarkerOption,用于在地图上添加Marker

option = new MarkerOptions()

.position(point)

.icon(bitmap);

//在地图上添加Marker,并显示

mBaiduMap.addOverlay(option);

通过上面的实现,我们就可以将一个小图标添加到地图层,作为标记。我们日常使用地图时,所搜周边后呈现的一系列小圆点就是如此(如下图)

marker示意图

ShowInfoWindow使用####

最后一步,实现显示街景缩略图的那个小弹框。

这里首先自定义一下我们要添加到地图层的View。

view = LayoutInflater.from(mContext).inflate(R.layout.pano_overlay, null);

pic = (ImageView) view.findViewById(R.id.panoImageView);

view.setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

Intent intent = new Intent(mContext, PanoDemoMain.class);

intent.putExtra("latitude", latitude);

intent.putExtra("longitude", longitude);

startActivity(intent);

}

});

这里pic这个ImageView用于显示我们要展示的街景缩略图。pano_overlay是整个弹框的布局,很简单,这里就不贴代码了。

同时,我们为这个自定义View设置点击事件,方便我们跳转到PanoView街景地图页面,并且将当前位置传递过去。

由于祖国地大物博,所以街景的覆盖并非百分之百,所以说,不是每个地方都有街景可以显示,有些鸟不拉屎的地方是看不到的。那我们怎么知道什么地方有街景呢?API为我们提供了很好的检测方法

new Thread(new Runnable() {

@Override

public void run() {

PanoramaRequest request =

PanoramaRequest.getInstance(mContext);

BaiduPanoData locationPanoData =

request.getPanoramaInfoByLatLon(longitude, latitude);

//开发者可以判断是否有外景(街景)

if (locationPanoData.hasStreetPano()) {

String url = baseUrl + locationPanoData.getPid();

Message message = new Message();

message.what = 0x01;

message.obj = url;

handler.sendMessage(message);

}

}

}).start();

这样,我们就可以根据当前位置,先检测一下是否有街景可以显示。这里,如果当前位置有街景,我们就通过Handler通知主线程去更新UI

private class myHandler extends Handler {

@Override

public void handleMessage(Message msg) {

super.handleMessage(msg);

if (msg.what == 0x01) {

String url = (String) msg.obj;

Glide.with(mContext).load(url).into(pic);

InfoWindow mInfoWindow = new InfoWindow(view, point, -57);

//显示InfoWindow

mBaiduMap.showInfoWindow(mInfoWindow);

}

}

}

这里看一下,InfoWindow的说明及其构造函数

public class InfoWindow extends java.lang.Object

在地图中显示一个信息窗口,可以设置一个View作为该窗口的内容,也可以设置一个 BitmapDescriptor 作为该窗口的内容。

public InfoWindow(View view, LatLng position, int yOffset)

/**

通过传入的 view 构造一个 InfoWindow, 此时只是利用该view

生成一个Bitmap绘制在地图中,监听事件由开发者实现。

Parameters:

view - InfoWindow 展示的 view

position - InfoWindow 显示的地理位置

yOffset - InfoWindow Y 轴偏移量

*/

在Handler的handleMessage方法中,我们通过返回的url加载图片,并将自定义的弹框View显示到之前一步添加的marker偏上一点的地方(这就是InfoWindow的构造函数中-57的意义)

关于这个加载图片的URL,可以参考这里静态图API。

这样,就实现了MapView页面所有的内容。通过点击InfoWindow,就可以跳转到PanoView所在的界面去查看街景地图。

接下来,我们将介绍PanoView街景地图的实现。

街景地图PanoViewActivity实现##

街景地图PanoView基础###

街景地图PanoView的显示和基础地图MapView十分相似

首先是在布局文件中定义view

android:id="@+id/panorama"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:clickable="true"

android:visibility="visible" />

在Activity的OnCreate方法中

PanoDemoApplication app = (PanoDemoApplication) this.getApplication();

if (app.mBMapManager == null) {

app.mBMapManager = new BMapManager(app);

app.mBMapManager.init(new

PanoDemoApplication.MyGeneralListener());

}

mPanoView = (PanoramaView) findViewById(R.id.panorama);

mPanoView.setPanorama(longitude, latitude);

这里同样需要的是在Application类中做一些初始化工作;对我们所使用key的有效性进行检测。

public void initEngineManager(Context context) {

if (mBMapManager == null) {

mBMapManager = new BMapManager(context);

}

if (!mBMapManager.init(new MyGeneralListener())) {

Toast.makeText(PanoDemoApplication.getInstance()

.getApplicationContext(), "BMapManager 初始化错误!",

Toast.LENGTH_LONG).show();

}

}

// 常用事件监听,用来处理通常的网络错误,授权验证错误等

static class MyGeneralListener implements MKGeneralListener {

@Override

public void onGetPermissionState(int iError) {

// 非零值表示key验证未通过

if (iError != 0) {

// 授权Key错误:

Toast.makeText(PanoDemoApplication.getInstance()

.getApplicationContext(),

"请在AndoridManifest.xml中输入正确的授权Key,并检查您的网络连接是否正常!error: " + iError, Toast.LENGTH_LONG).show();

} else {

Toast.makeText(PanoDemoApplication.getInstance()

.getApplicationContext(), "key认证成功", Toast.LENGTH_LONG)

.show();

}

}

}

同时在Activity里也需要做一些初始化的工作,最后就是通过PanoView的setPanorama()方法实现街景的显示。

关于这里用到的setPanorama(),根据API我们可以看到

public void setPanorama(java.lang.String pid)

//根据全景pid值切换全景场景

public void setPanorama(int x,int y)

//根据百度墨卡托投影坐标切换全景场景

public void setPanorama(double longitude,double latitude)

//根据百度经纬度坐标切换全景场景

public void setPanoramaByUid(java.lang.String uid,

int panoType)

//根据uid值切换全景场景

也就是说,不仅通过经纬度,而且可以通过别的方式实现街景地图的功能,甚至室内景的实现。这里我们就使用了大家最熟悉的经纬度,对于别的实现方式有兴趣的同学,可以自己去探索一下。

将地图MapView展示在街景PanoView上面###

如图所示,将一个MapView显示在PanoView之上;很自然的我们会写出下面的布局方式:

android:layout_width="match_parent"

android:layout_height="match_parent">

android:id="@+id/panorama"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:clickable="true"

android:visibility="visible" />

android:layout_width="60dp"

android:layout_height="60dp"

android:layout_alignParentBottom="true"

android:layout_alignParentLeft="true"

android:layout_marginBottom="8dp"

android:layout_marginLeft="8dp"

android:background="#00ffffff">

android:id="@+id/bmapView"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:clickable="true"

android:padding="20dp" />

这样,我们在整个PanoView的左下角定义一个60x60大小的view用于显示一个MapView。

实现对街景视图操作的监听###

街景SDK为我们提供了PanoramaViewListener这个接口,可以实现对从街景视图开始绘制到完成绘制,对街景地图的操作(如点击,旋转)的监听。

API

这里我们重点看一下onMessage(String msgName, int msgType)这个回调方法。

public void onMessage(String msgName, int msgType) {

Log.e(LTAG, "msgName--->" + msgName + ",

msgType--->" + msgType);

switch (msgType) {

case 8213:

//旋转

Log.e(PanoViewActivity.class.getSimpleName(),

"now,the heading is " + mPanoView.getPanoramaHeading());

Message message = new Message();

message.what = ACTION_DRAG;

message.arg1 = (int) mPanoView.getPanoramaHeading();

handler.sendMessage(message);

break;

case 12302:

//点击

Log.e(PanoViewActivity.class.getSimpleName(),

"clicked");

Message msg = new Message();

msg.what = ACTION_CLICK;

handler.sendMessage(msg);

break;

default:

break;

}

}

这里不得不吐槽一下,官方所提供的API文档,对这个onMessage回调方法中的参数居然没有任何有价值的解释。这里的8213及12302完全是通过打印日志自己总结出的规律。

这样,我们对于不同的操作,就可以通过Handler实现不同的UI效果。我们看一下handler的实现:

private class MyHandler extends Handler {

@Override

public void handleMessage(Message msg) {

super.handleMessage(msg);

switch (msg.what) {

case ACTION_CLICK:

if (titleVisible) {

titleVisible = false;

textTitle.startAnimation(animationHide);

textTitle.setVisibility(View.GONE);

sv.setVisibility(View.GONE);

} else {

titleVisible = true;

textTitle.startAnimation(animationShow);

textTitle.setVisibility(View.VISIBLE);

sv.setVisibility(View.VISIBLE);

}

break;

case ACTION_DRAG:

float heading = (float) msg.arg1;

mBaiduMap.clear();

//构建MarkerOption,用于在地图上添加Marker

option = new MarkerOptions()

.position(point)

.rotate(360-heading)

.icon(bitmap);

//在地图上添加Marker,并显示

mBaiduMap.addOverlay(option);

break;

default:

break;

}

}

}

这里的处理就分两种情况:

点击事件

我们仿照百度地图的样式,实现标题栏及MapView的隐藏,并添加动画,这样可以方便用户全屏更清晰的观察街景内容。

旋转事件

上面我们说过对MapView添加Marker的方法,这里就派上用场了。随着我们对PanoView的不断拖拽旋转,通过其getPanoramaHeading() 可以得到当前视角的偏航角。

在UI线程中,我们可以通过不断移除和添加Marker,并设置不同的marker的偏转角度,从而实现一种在左下方小地图上呈现我们当前视角的效果。

好了,这样就简单模仿了一下百度地图街景的部分实现功能,由于UI资源所限制,部分效果并非完全一致,这里只是学习下而已。

代码已上传至github,点这里即可查看。

仿百度地图 android,仿百度地图街景实现相关推荐

  1. 百度地图 Android SDK - 个性化地图

    什么是百度个性化地图Android SDK? 百度个性化地图Android SDK是一套基于Android 2.2及以上版本号设备的应用程序接口,您能够通过该套接口实现主要的地图功能,而且能够定制地图 ...

  2. android仿ios菊花,android仿IOS 旋转菊花Loading、等待Loading、花瓣

    好久没写博客了,每天都在不停的进步着,今天给大家带来的是仿菊花loading,在百度看了许多,大部分都是通过图片的方式实现的,那么我现在是通过自定义view的方式实现的,废话也不多说了,就在代码中注释 ...

  3. android 仿搜索动画,Android仿京东顶部搜索框滑动伸缩动画效果

    最近使用京东发现,京东顶部的搜索框有一个新的伸缩效果,根据用户的手势滑动,伸缩搜索框.觉得效果还不错,就看了下其他的应用有没有这种伸缩的效果,发现安居客也使用了类似的一种效果,然后就想着实现这样的一种 ...

  4. 基于android的高仿抖音,Android仿抖音列表效果

    本文实例为大家分享了Android仿抖音列表效果的具体代码,供大家参考,具体内容如下 当下抖音非常火热,是不是也很心动做一个类似的app吗? 那我们就用RecyclerView实现这个功能吧,关于内存 ...

  5. android高仿微信拍摄,Android 仿微信视频拍摄 支持触摸拍摄 长按拍摄

    JCamera This is Android CameraActivity,Imitation WeChat Camera Android 仿微信视频拍摄 支持触摸拍摄 长按拍摄,采用camera2 ...

  6. android高仿京东秒杀,Android仿京东首页秒杀倒计时

    本文实例为大家分享了Android仿京东首页秒杀倒计时的具体代码,供大家参考,具体内容如下 xml配置 android:layout_width="wrap_content" an ...

  7. android 仿360浮动,Android仿360悬浮小球自定义view实现示例

    Android仿360悬浮小球自定义view实现示例 效果图如下: 实现当前这种类似的效果 和360小球 悬浮桌面差不错类似.这种效果是如何实现的呢.废话不多说 ,直接上代码. 1.新建工程,添加悬浮 ...

  8. android仿iphone页面,Android仿苹果关机界面实现代码

    本文实例为大家分享了Android仿苹果关机界面的具体代码,供大家参考,具体内容如下 主class 用来控制viewdialog的显示 package com.android.server.polic ...

  9. android 仿iphone滚轮,android仿iphone滚轮控件显示效果

    android仿iphone滚轮控件显示效果,供大家参考,具体内容如下 在论坛里看到的,自己弄个效果: 这个滚动的WheelView /* * Android Wheel Control. * htt ...

最新文章

  1. Pig的安装和简单使用
  2. OpenCV HDF和建立群组Group
  3. 数据结构 - 单链表(百度面试题单链表的倒序打印)
  4. bdc注意清空bdcdata【否则bdc可能…
  5. DEVO 7E遥控器配对
  6. MachineLearning:UFLDL学习笔记 ---- 主成分分析与白化
  7. Java基本开发环境搭建(适合第一次使用)
  8. spring5.0学习笔记7
  9. Note8 android 9 root,三星Note8 国行 root N9500 9.0 root N9500ZCS6DTC1 root
  10. 【CANdelaStudio编辑CDD】-0.1-如何对比两个CDD诊断描述文件
  11. gRPC 使用自定义的 NameResolver
  12. 青岛科技大学软件测试专业如何,青岛科技大学优势专业排名
  13. 浏览器访问jsp页面
  14. 【公益创投】“心健康护成长”来穗青少年心理健康项目联合琶洲街“丝路红星,与法同行”来穗与本土青少年融合社区教育服务项目启动仪式
  15. 一步步学习zynq软硬件协同开发(AX7010/20)【FPGA+ReWorks】:创建自定义IP实现rtc读写
  16. pixabay注册失败原因以及解决办法
  17. 基于STM32F1-C8T6无人机(二)——舵机/电调/空心杯电机/飞控/机架/subs接收机/充电器和电池(给出链接和思考)
  18. 给UMEditor 增加placeholder
  19. 2022.3 C++植物大战僵尸项目总结
  20. C语言练习——单词本管理

热门文章

  1. 智能驾驶板块_无人驾驶受益股名单有哪些?无人驾驶板块龙头股票一览表2020...
  2. 收集的几个存在漏洞的程序
  3. NCBI-genebank注释说明
  4. golang后台管理系统GoFrame+Vue+ElementUI框架搭建教程
  5. python - 饮料机,if elif与列表法
  6. 请写出sfr和sbit的语句格式_根据给水阀门井图集07MS101-2第25页阀门井,进行清单组价,截图上传并上传计价软件版本附件...
  7. 乘“云”而上,有孚云为教育行业插上翅膀
  8. 公司来了个北大妹子,一人写了个操作系统,折服了。。。
  9. 给海归技术创业兄弟的九个忠告(转载)
  10. 2012虚拟机配置服务器配置,Windows Server 2012 R2 NAP服务器安装与配置03之 NPS服务器DHCP访问控制配置...