系列文章
提示:转到安卓学习专栏,观看更多内容!
点我直达–>安卓学习专栏
前言
新建一个项目,项目名为LBSTest。
百度地图(6)移动到我的位置(图文+代码)
地图是成功显示出来了,但也许这并不是你想要的。因为这是一张默认
的地图,显示的是北京市中心的位置,而你可能希望看到更加精细的地
图信息,比如说自己所在位置的周边环境。显然,通过缩放和移动的方
式来慢慢找到自己的位置是一种很愚蠢的做法。那么本小节我们就来学
习一下,如何才能在地图中快速移动到自己的位置。
介绍api
百度LBS SDK的API中提供了一个BaiduMap 类,它是地图的总控制器,调用MapView的getMap()方法就能获取到BaiduMap 的实例。
BaiduMap baiduMap = mapView.getMap();
有了BaiduMap后,我们就能对地图进行各种各样的操作了,比如设置
地图的缩放级别以及将地图移动到某一个经纬度上。百度地图将缩放级别的取值范围限定在3到19之间,其中小数点位的值也是可以取的,值越大,地图显示的信息就越精细。比如我们想要将缩
放级别设置成12.5:
MapStatusUpdateFactory的zoomTo() 方法接收一个float 型的参
数,就是用于设置缩放级别的,这里我们传入12.5f。zoomTo() 方法返
回一个MapStatusUpdate 对象,我们把这个对象传入BaiduMap的
animateMapStatus() 方法当中即可完成缩放功能。
MapStatusUpdate update = MapStatusUpdateFactory.zoomTo(12.5f);
baiduMap.animateMapStatus(update);
让地图移动到某一个经纬度需要借助LatLng类了。其实LatLng存放经纬度值的,它的构造方法接收两个参数,第一个参数是纬度值,第二个参数是经度值。之后调用MapStatusUpdateFactory的newLatLng() 方法将LatLng 对象传入,newLatLng() 方法返回的也是一个MapStatusUpdate 对象,我们再把这个对象传入BaiduMap的animateMapStatus() 方法当中,就可以将地图移动到指定的经纬度上,实现将地图移动到北纬39.915度、东经116.404度这个位置的代码:
LatLng ll = new LatLng(39.915, 116.404);
MapStatusUpdate update = MapStatusUpdateFactory.newLatLng(ll);
baiduMap.animateMapStatus(update);
实现效果
activity_main.xml
- 布局文件中的内容非常简单,只有一个TextView控件,用于稍后显示当
前位置的经纬度信息。 - 在布局文件中新放置了一个MapView控件,并让它填充满整个屏
幕。这个MapView是由百度提供的自定义控件,所以在使用它的时候需
要将完整的包名加上。另外,之前用于显示定位信息的TextView现在暂
时用不到了,我们将它的visibility 属性指定成gone ,让它在界面上
隐藏起来。
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent" >
<TextView
android:id="@+id/position_text_view"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:visibility="gone" />
<com.baidu.mapapi.map.MapView
android:id="@+id/bmapView"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clickable="true" />
</LinearLayout>
AndroidManifest.xml
- AndroidManifest.xml文件改动比较多,我们来仔细阅读一下。可以看到,这里首先添加了很多行权限声明,每一个权限都是百度LBS SDK内部要用到的。然后在 标签的内部添加了一个<metadata> 标签,这个标签的android:name 部分是固定的,必须com.baidu.lbsapi.API_KEY ,android:value 部分则应该填入我们在前面的文章
安卓学习专栏——安卓获取SHA1指纹申请百度地图API Key使用百度LBS定位功能(图文保姆级)
申请到的API Key。 - 如果你的“android:name=“com.baidu.location.f””飘红说明前面的jar配置有问题
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.lbstest">
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/Theme.AppCompat">
<meta-data
android:name="com.baidu.lbsapi.API_KEY"
android:value="填写你前面申请到的api key" />
<activity
android:name=".MainActivity"
android:exported="true">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<service
android:name="com.baidu.location.f"
android:enabled="true"
android:process=":remote">
</service>
</application>
</manifest>
MainActivity.java(修改)
加入了一个navigateTo() 方法。将BDLocation 对象中的地理位置信息取出并封装到LatLng 对象中,然后调用MapStatusUpdateFactory的newLatLng() 方法并将LatLng 对象传入,接着将返回的MapStatusUpdate 对象作为参数传入到BaiduMap的animateMapStatus() 方法当中,和上面介绍的用法是一模一样的。并且这里为了让地图信息可以显示得更加丰富一些,我们将缩放级别设置
成了16。
上述代码当中我们使用了一个isFirstLocate 变量,这个变量的作用是为了防止多次调用animateMapStatus() 方法,因为将地图移动到我们当前的位置只需
要在程序第一次定位的时候调用一次就可以了。
写了navigateTo() 方法之后,剩下的事情就简单了,当定位到设备当前位置的时候,我们在onReceiveLocation() 方法中直接把BDLocation 对象传给navigateTo() 方法,这样就能够让地图移动到设备所在的位置了。
package com.example.lbstest;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.core.content.ContextCompat;
import android.Manifest;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.widget.TextView;
import android.widget.Toast;
import com.baidu.location.BDLocation;
import com.baidu.location.BDLocationListener;
import com.baidu.location.LocationClient;
import com.baidu.location.LocationClientOption;
import com.baidu.mapapi.SDKInitializer;
import com.baidu.mapapi.map.BaiduMap;
import com.baidu.mapapi.map.MapStatusUpdate;
import com.baidu.mapapi.map.MapStatusUpdateFactory;
import com.baidu.mapapi.map.MapView;
import com.baidu.mapapi.model.LatLng;
import java.util.ArrayList;
import java.util.List;
public class MainActivity extends AppCompatActivity {
public LocationClient mLocationClient;
private TextView positionText;
private MapView mapView;
//修改内容开始
private BaiduMap baiduMap;
private boolean isFirstLocate = true;
//修改内容结束
@Override
protected void onCreate(Bundle savedInstanceState) {
//修改内容开始
//首先需要调用SDKInitializer的
//initialize() 方法来进行初始化操作,initialize() 方法接收一
//个Context 参数,这里我们调用getApplicationContext() 方法来获
//取一个全局的Context 参数并传入。注意初始化操作一定要
//在setContentView() 方法前调用,不然的话就会出错。接下来我们调
//用findViewById() 方法获取到了MapView的实例
super.onCreate(savedInstanceState);
mLocationClient = new LocationClient(getApplicationContext());
mLocationClient.registerLocationListener(new MyLocationListener());
SDKInitializer.initialize(getApplicationContext());
setContentView(R.layout.activity_main);
mapView = (MapView) findViewById(R.id.bmapView);
baiduMap = mapView.getMap();
//修改内容结束
positionText = (TextView) findViewById(R.id.position_text_view);
List<String> permissionList = new ArrayList<>();
if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.
permission.ACCESS_FINE_LOCATION) != PackageManager.PERMISSION_GRANTED) {
permissionList.add(Manifest.permission.ACCESS_FINE_LOCATION);
}
if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.
permission.READ_PHONE_STATE) != PackageManager.PERMISSION_GRANTED) {
permissionList.add(Manifest.permission.READ_PHONE_STATE);
}
if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.
permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
permissionList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
}
if (!permissionList.isEmpty()) {
String[] permissions = permissionList.toArray(new String[permissionList.
size()]);
ActivityCompat.requestPermissions(MainActivity.this, permissions, 1);
} else {
requestLocation();
}
}
private void navigateTo(BDLocation location) {
if (isFirstLocate) {
LatLng ll = new LatLng(location.getLatitude(), location.getLongitude());
MapStatusUpdate update = MapStatusUpdateFactory.newLatLng(ll);
baiduMap.animateMapStatus(update);
update = MapStatusUpdateFactory.zoomTo(16f);
baiduMap.animateMapStatus(update);
isFirstLocate = false;
}
}
public class MyLocationListener implements BDLocationListener {
@Override
public void onReceiveLocation(BDLocation location) {
if (location.getLocType() == BDLocation.TypeGpsLocation
|| location.getLocType() == BDLocation.TypeNetWorkLocation) {
navigateTo(location);
}
}
}
private void requestLocation() {
initLocation();
mLocationClient.start();
}
private void initLocation() {
LocationClientOption option = new LocationClientOption();
option.setScanSpan(5000);
option.setIsNeedAddress(true);
mLocationClient.setLocOption(option);
}
//要重写onResume() 、onPause() 和onDestroy() 这3个方
//法,在这里对MapView进行管理,以保证资源能够及时地得到释放。
@Override
protected void onResume() {
super.onResume();
mapView.onResume();
}
@Override
protected void onPause() {
super.onPause();
mapView.onPause();
}
@Override
protected void onDestroy() {
super.onDestroy();
mLocationClient.stop();
mapView.onDestroy();
}
//新修改的内容实时更新位置结束
@Override
public void onRequestPermissionsResult(int requestCode, String[] permissions,
int[] grantResults) {
switch (requestCode) {
case 1:
if (grantResults.length > 0) {
for (int result : grantResults) {
if (result != PackageManager.PERMISSION_GRANTED) {
Toast.makeText(this, "必须同意所有权限才能使用本程序",
Toast.LENGTH_SHORT).show();
finish();
return;
}
}
requestLocation();
} else {
Toast.makeText(this, "发生未知错误", Toast.LENGTH_SHORT).show();
finish();
}
break;
default:
}
}
}
附录.参考资料
下载资源
总结
大家喜欢的话,给个👍,点个关注!继续跟大家分享敲代码过程中遇到的问题!