Android实现高德地图轨迹回放
写在前面
本篇文章是对近期工作项目中集成高德地图轨迹回放和单/多点标记功能的一个总结。方便以后使用时翻阅,也方便大家,避免跟我踩一样的坑
欢迎指正,不喜勿喷,请你善良,谢谢!!!
准备
官方文档解读
首先去[官网](https://lbs.amap.com/api/android-sdk/summary)查看开发指南进行项目中需要的内容准备。
创建应用:
左上角头像位置指针划过出现如图所示内容
后点击应用管理->我的应用,如下图
上图中创建新应用后进入如下界面
上图两个步骤完成后点击新建,新建后就会在我的应用中显示你自己创建的这个应用,我用的是test作为演示。新建完成后如下图
点击添加展示图如下:
将上图中内容编辑完成后提交,就会在应用列表中将key展示出来,这里我就不做展示了。
以上的内容就是官方文档中获取key的内容。相对来说还是比较简单的。
地图api引入:
//定位
implementation 'com.amap.api:location:5.1.0'
//搜索
implementation 'com.amap.api:search:7.3.0'
//导航
implementation 'com.amap.api:navi-3dmap:7.6.0_3dmap7.6.0'
权限添加
<!--允许程序打开网络套接字-->
<uses-permission android:name="android.permission.INTERNET" />
<!--允许程序设置内置sd卡的写权限-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<!--允许程序获取网络状态-->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!--允许程序访问WiFi网络信息-->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<!--允许程序读写手机状态和身份-->
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<!--用于进行网络定位-->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<!--用于访问GPS定位-->
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<!--用于获取wifi的获取权限,wifi信息会用来进行网络定位-->
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE"/>
<!--用于读取手机当前的状态-->
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
<!--用于申请调用A-GPS模块-->
<uses-permission android:name="android.permission.ACCESS_LOCATION_EXTRA_COMMANDS"/>
<!--获取运营商信息,用于支持提供运营商信息相关的接口-->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<!--用于访问wifi网络信息,wifi信息会用于进行网络定位-->
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<!--这个权限用于获取wifi的获取权限,wifi信息会用来进行网络定位-->
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<!--用于访问网络,网络定位需要上网-->
<uses-permission android:name="android.permission.INTERNET" />
<!--用于读取手机当前的状态-->
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<!--写入扩展存储,向扩展卡写入数据,用于写入缓存定位数据-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
并且要在AndroidManifest.xml的application中添加如下代码:
<meta-data
android:name="com.amap.api.v2.apikey"
android:value="创建应用中你自己生成的key" />
<service android:name="com.amap.api.location.APSService" />
到这里准备工作就完成了。我先上个效果图展示一下。
效果展示
有一个点呼吸的效果是我今天早上才加进去的,所以展示效果上面没有体现出来呢,不过这个功能也不是本次项目的重点
过程实现
地图初始化
mMapView.onCreate(savedInstanceState);
if (aMap == null) {
aMap = mMapView.getMap();
myLocationStyle = new MyLocationStyle();
myLocationStyle.myLocationType(MyLocationStyle.LOCATION_TYPE_LOCATION_ROTATE_NO_CENTER);
// 设置圆形的边框颜色
myLocationStyle.strokeColor(Color.argb(50, 30, 150, 180));
// 设置圆形的填充颜色
myLocationStyle.radiusFillColor(Color.argb(50, 30, 150, 180));
// 设置圆形的边框粗细
myLocationStyle.strokeWidth(1.0f);
//设置显示定位按钮 并且可以点击
UiSettings settings = aMap.getUiSettings();
//设置了定位的监听
aMap.setLocationSource(this);
// 是否显示定位按钮
settings.setMyLocationButtonEnabled(true);
//显示定位层并且可以触发定位,默认是flase
aMap.setMyLocationEnabled(true);
// 定位、且将视角移动到地图中心点,定位点依照设备方向旋转,并且会跟随设备移动。
}
定位
//初始化定位
mlocationClient = new AMapLocationClient(getApplicationContext());
//设置定位回调监听
mlocationClient.setLocationListener(this);
//初始化定位参数
mLocationOption = new AMapLocationClientOption();
//设置定位模式为Hight_Accuracy高精度模式,Battery_Saving为低功耗模式,Device_Sensors是仅设备模式
mLocationOption.setLocationMode(AMapLocationClientOption.AMapLocationMode.Hight_Accuracy);
//设置是否返回地址信息(默认返回地址信息)
mLocationOption.setNeedAddress(true);
//设置是否只定位一次,默认为false
mLocationOption.setOnceLocation(false);
//设置是否强制刷新WIFI,默认为强制刷新
mLocationOption.setWifiActiveScan(true);
//设置是否允许模拟位置,默认为false,不允许模拟位置
mLocationOption.setMockEnable(false);
//设置定位间隔,单位毫秒,默认为2000ms
mLocationOption.setInterval(2000);
//给定位客户端对象设置定位参数
mlocationClient.setLocationOption(mLocationOption);
//启动定位
mlocationClient.startLocation();
/**
* 定位回调
* /
@Override
public void onLocationChanged(AMapLocation aMapLocation) {
if (mListener != null && aMapLocation != null) {
if (aMapLocation != null && aMapLocation.getErrorCode() == 0) {
//定位成功回调信息,设置相关消息
aMapLocation.getLocationType();//获取当前定位结果来源,如网络定位结果,详见官方定位类型表
aMapLocation.getLatitude();//获取纬度
aMapLocation.getLongitude();//获取经度
aMapLocation.getAccuracy();//获取精度信息
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date date = new Date(aMapLocation.getTime());
df.format(date);//定位时间
aMapLocation.getAddress();//地址,如果option中设置isNeedAddress为false,则没有此结果,网络定位结果中会有地址信息,GPS定位不返回地址信息。
aMapLocation.getCountry();//国家信息
aMapLocation.getProvince();//省信息
aMapLocation.getCity();//城市信息
aMapLocation.getDistrict();//城区信息
aMapLocation.getStreet();//街道信息
aMapLocation.getStreetNum();//街道门牌号信息
aMapLocation.getCityCode();//城市编码
aMapLocation.getAdCode();//地区编码
// 如果不设置标志位,此时再拖动地图时,它会不断将地图移动到当前的位置
if (isFirstLoc) {
//设置缩放级别
aMap.moveCamera(CameraUpdateFactory.zoomTo(18));
//将地图移动到定位点
aMap.moveCamera(CameraUpdateFactory.changeLatLng(new LatLng(aMapLocation.getLatitude(), aMapLocation.getLongitude())));
//点击定位按钮 能够将地图的中心移动到定位点
mListener.onLocationChanged(aMapLocation);
//添加图钉
// aMap.addMarker(getMarkerOptions(amapLocation));
//获取定位信息
StringBuffer buffer = new StringBuffer();
buffer.append(aMapLocation.getCountry() + ""
+ aMapLocation.getProvince() + ""
+ aMapLocation.getCity() + ""
+ aMapLocation.getProvince() + ""
+ aMapLocation.getDistrict() + ""
+ aMapLocation.getStreet() + ""
+ aMapLocation.getStreetNum());
Toast.makeText(getApplicationContext(), buffer.toString(), Toast.LENGTH_LONG).show();
isFirstLoc = false;
// 显示系统小蓝点
mListener.onLocationChanged(aMapLocation);
}
} else {
Log.e("AmapError", "location Error, ErrCode:"
+ aMapLocation.getErrorCode() + ", errInfo:"
+ aMapLocation.getErrorInfo());
Toast.makeText(getApplicationContext(), "定位失败", Toast.LENGTH_LONG).show();
}
}
}
显示标记点
ArrayList<BitmapDescriptor> normalGiflist = new ArrayList<>();
normalGiflist.add(BitmapDescriptorFactory.fromResource(R.drawable.icon_normal1));
normalGiflist.add(BitmapDescriptorFactory.fromResource(R.drawable.icon_normal17));
normalGiflist.add(BitmapDescriptorFactory.fromResource(R.drawable.icon_normal38));
for (int i = 0; i < coords.length; i += 2) {
Marker marker = aMap.addMarker(new MarkerOptions().position(new LatLng(coords[i + 1], coords[i])).anchor(0.5f, 0.5f).icons(normalGiflist).period(15));
markerList.add(marker);
}
注意:这里要说一下关于点标记删除的问题,删除点要将点遍历后通过remove移除,虽然map.clear()方法也可以使用,但是如果同时有很多中marker在显示的话就不建议使用了
我就死在了这个上面纠结了好几天,导致各种bug频现,唉,说多了全是泪啊????
点平滑移动
public void startMove() {
if (mPolyline == null) {
Toast.makeText(this, "请先设置路线", Toast.LENGTH_SHORT).show();
return;
}
List<LatLng> points = readLatLngs();
// 构建 轨迹的显示区域
LatLngBounds.Builder builder = new LatLngBounds.Builder();
builder.include(points.get(0));
builder.include(points.get(points.size() - 2));
aMap.animateCamera(CameraUpdateFactory.newLatLngBounds(builder.build(), 50));
// 实例 MovingPointOverlay 对象
if (smoothMarker == null) {
// 设置 平滑移动的 图标
marker = aMap.addMarker(new MarkerOptions().icon(BitmapDescriptorFactory.fromResource(R.mipmap.ic_launcher)));
smoothMarker = new MovingPointOverlay(aMap, marker);
}
// 取轨迹点的第一个点 作为 平滑移动的启动
LatLng drivePoint = points.get(0);
Pair<Integer, LatLng> pair = SpatialRelationUtil.calShortestDistancePoint(points, drivePoint);
points.set(pair.first, drivePoint);
List<LatLng> subList = points.subList(pair.first, points.size());
// 设置轨迹点
smoothMarker.setPoints(subList);
// 设置平滑移动的总时间 单位 秒
smoothMarker.setTotalDuration(40);
// 设置 自定义的InfoWindow 适配器
aMap.setInfoWindowAdapter(infoWindowAdapter);
// 显示 infowindow
marker.showInfoWindow();
// 设置移动的监听事件 返回 距终点的距离 单位 米
smoothMarker.setMoveListener(new MovingPointOverlay.MoveListener() {
@Override
public void move(final double distance) {
try {
runOnUiThread(new Runnable() {
@Override
public void run() {
if (infoWindowLayout != null && title != null) {
title.setText("距离终点还有: " + (int) distance + "米");
}
}
});
} catch (Throwable e) {
e.printStackTrace();
}
}
});
// 开始移动
smoothMarker.startSmoothMove();
}
添加呼吸点
noLineMaker = aMap.addMarker(new MarkerOptions().position(new LatLng(39.97693511, 116.34892166))
.zIndex(1).anchor(0.5f, 0.5f).icon(BitmapDescriptorFactory.fromResource(R.drawable.icon_no_line1)));
//呼吸点中心点
aMap.addMarker(new MarkerOptions().position(new LatLng(39.97693511, 116.34892166)).zIndex(2)
.anchor(0.5f, 0.5f).icon(BitmapDescriptorFactory.fromResource(R.drawable.icon_no_line1)));
AnimationSet animationSet = new AnimationSet(true);
AlphaAnimation alphaAnimation = new AlphaAnimation(0.5f, 0f);
alphaAnimation.setDuration(2000);
alphaAnimation.setRepeatCount(Animation.INFINITE);
ScaleAnimation scaleAnimation = new ScaleAnimation(1, 3.5f, 1, 3.5f);
scaleAnimation.setDuration(2000);
scaleAnimation.setRepeatCount(Animation.INFINITE);
animationSet.addAnimation(alphaAnimation);
animationSet.addAnimation(scaleAnimation);
animationSet.setInterpolator(new LinearInterpolator());
noLineMaker.setAnimation(animationSet);
noLineMaker.startAnimation();
到这里所有的内容就都结束了。
写在结尾
这篇文章本来应该是在前天下午就应该出现的,但是由于不会弄那个gif图,导致今天才完成了这篇文章。虽然文章写晚了,但是还是有一个小功能添加的,所以也不亏的哦。按照本文章的话还是可以顺利完成这些功能的,我也是找了很多的文章之后才实现这些功能的,中间还是踩了很多坑的,现在回想一下,好像也没有什么。所以,加油,打工人!我们都是最棒的。
本篇文章相关demo请关注微信公众号”只会写bug的咸鱼“ 后台回复”轨迹回放“获取
创作不易,不喜勿喷,欢迎指正!
最后的最后,我要说一句,周末愉快,打工人
本文地址:https://blog.csdn.net/lzllzllhl/article/details/109380951