[关闭]
@kezhen 2015-09-18T01:20:01.000000Z 字数 6926 阅读 5893

基于百度地图SDK记录运动轨迹

运动轨迹


一、 前期准备工作

首先需要登录百度开放平台下载iOS地图SDK(内含开发者文档、框架以及Demo示例),推荐下载全新Framework形式的静态库,配置更加简单方便,具体看下图:
百度开放平台

framework静态库


1. 新建Xcode工程

File->New->Project->Single View Application,填写好相关信息完成工程建立。
新建工程


2. 获取Bundle Identifier

通过project->target->general可以看到本应用的Bundle Identifie,我们正是需要这串字符串去百度开发平台申请一个Key用于百度地图的调用。
Bundle Identifie


3. 申请key

登录百度开放平台后,点击右上角的API控制台进入申请key的界面,点击创建应用,在“安全码”处填入你的应用的Bundle Identifie,具体信息填写请看下图:
申请key信息填写图1
申请key信息填写图2


4. 导入框架配置工程

第一步 、引入BaiduMapAPI.framework
百度地图SDK提供了模拟器和真机两种环境所使用的framework,分别存放在libs/Release-iphonesimulator和libs/Release-iphoneos文件夹下,开发者可根据需要使用真机或模拟器的包,如果需同时使用真机和模拟器的包,可以使用lipo命令将设备和模拟器framwork包中的BaiduMapAPI文件合并成一个通用的文件,命令如下:

  1. lipo -create Release-iphoneos/BaiduMapAPI.framework/BaiduMapAPI Release-iphonesimulator/BaiduMapAPI.framework/BaiduMapAPI -output Release-iphoneos/BaiduMapAPI.framework/BaiduMapAPI

此时Release-iphoneos文件夹下的BaiduMapAPI.framework即可同时用于真机和模拟器。将所需的BaiduMapAPI.framework拷贝到工程所在文件夹下。在TARGETS->Build Phases-> Link Binary With Libaries中点击+按钮,在弹出的窗口中点击“Add Other”按钮,选择BaiduMapAPI.framework文件添加到工程中。
注:静态库中采用ObjectC++实现,因此需要您保证您工程中至少有一个.mm后缀的源文件(您可以将任意一个.m后缀的文件改名为.mm),或者在工程属性中指定编译方式,即将Xcode的Project -> Edit Active Target -> Build -> GCC4.2 - Language -> Compile Sources As设置为Objective-C++

第二步、引入所需的系统库
百度地图SDK中提供了定位功能和动画效果,v2.0.0版本开始使用OpenGL渲染,因此您需要在您的Xcode工程中引入CoreLocation.frameworkQuartzCore.frameworkOpenGLES.framework
SystemConfiguration.frameworkCoreGraphics.framework
Security.framework。添加方式:在Xcode的Project -> Active Target ->Build Phases ->Link Binary With Libraries,添加这几个framework即可。

第三步、环境配置
TARGETS->Build Settings->Other Linker Flags中添加-ObjC

第四步、引入mapapi.bundle资源文件
如果使用了基础地图功能,需要添加该资源,否则地图不能正常显示
mapapi.bundle中存储了定位、默认大头针标注View及路线关键点的资源图片,还存储了矢量地图绘制必需的资源文件。如果您不需要使用内置的图片显示功能,则可以删除bundle文件中的image文件夹。您也可以根据具体需求任意替换或删除该bundle中image文件夹的图片文件。
方法:选中工程名,在右键菜单中选择Add Files to “工程名”…,从BaiduMapAPI.framework||Resources文件中选择mapapi.bundle文件,并勾选“Copy items if needed”复选框,单击Add按钮,将资源文件添加到工程中。

第五步、引入头文件
在使用SDK的类引入头文件:

  1. #import <BaiduMapAPI/BMapKit.h>//引入所有的头文件
  2. #import <BaiduMapAPI/BMKMapView.h>//只引入所需的单个头文件

--引用自百度开放平台iOS SDK环境配置


5. 初始化 BMKMapManager

  1. @interface AppDelegate ()<BMKGeneralDelegate>
  2. @property (nonatomic,strong) BMKMapManager *mapManager;
  3. @end
  1. - (void)onGetNetworkState:(int)iError
  2. {
  3. if (0 == iError) {
  4. NSLog(@"联网成功");
  5. }
  6. else{
  7. NSLog(@"onGetNetworkState %d",iError);
  8. }
  9. }
  10. - (void)onGetPermissionState:(int)iError
  11. {
  12. if (0 == iError) {
  13. NSLog(@"授权成功");
  14. }
  15. else {
  16. NSLog(@"onGetPermissionState %d",iError);
  17. }
  18. }

BMKGeneralDelegate.h

  1. - (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
  2. // 要使用百度地图先实例化 BMKMapManager
  3. self.mapManager = [[BMKMapManager alloc]init];
  4. // 如果要关注网络及授权验证事件,请设定 generalDelegate 参数
  5. BOOL ret = [self.mapManager start:@"OjYbYha0YULmuLPaHT9wxxx" generalDelegate:self];
  6. if (!ret) {
  7. NSLog(@"manager start failed");
  8. }
  9. return YES;
  10. }

二、实战记录运动轨迹

一条完整的运动轨迹是由一组地理位置坐标数组描点连线构成的,我们需要实时监测用户位置的变更,将最新的符合限定条件的地位位置数据存放到数据中,调用SDK中的画折线方法绘制运动轨迹。


1. 初始化工作

  1. @interface ViewController () <BMKMapViewDelegate, BMKLocationServiceDelegate>
  2. /** 记录上一次的位置 */
  3. @property (nonatomic, strong) CLLocation *preLocation;
  4. /** 位置数组 */
  5. @property (nonatomic, strong) NSMutableArray *locationArrayM;
  6. /** 轨迹线 */
  7. @property (nonatomic, strong) BMKPolyline *polyLine;
  8. /** 百度地图View */
  9. @property (nonatomic,strong) BMKMapView *mapView;
  10. /** 百度定位地图服务 */
  11. @property (nonatomic, strong) BMKLocationService *bmkLocationService;
  12. @end
  13. @implementation ViewController
  14. - (void)viewDidLoad {
  15. [super viewDidLoad];
  16. // 初始化百度位置服务
  17. [self initBMLocationService];
  18. // 初始化地图窗口
  19. self.mapView = [[BMKMapView alloc]initWithFrame:self.view.bounds];
  20. // 设置MapView的一些属性
  21. [self setMapViewProperty];
  22. [self.view addSubview:self.mapView];
  23. }
  24. @end
  1. /**
  2. * 设置 百度MapView的一些属性
  3. */
  4. - (void)setMapViewProperty
  5. {
  6. // 显示定位图层
  7. self.mapView.showsUserLocation = YES;
  8. // 设置定位模式
  9. self.mapView.userTrackingMode = BMKUserTrackingModeNone;
  10. // 允许旋转地图
  11. self.mapView.rotateEnabled = YES;
  12. // 显示比例尺
  13. self.bmkMapView.showMapScaleBar = YES;
  14. self.bmkMapView.mapScaleBarPosition = CGPointMake(self.view.frame.size.width - 50, self.view.frame.size.height - 50);
  15. // 定位图层自定义样式参数
  16. BMKLocationViewDisplayParam *displayParam = [[BMKLocationViewDisplayParam alloc]init];
  17. displayParam.isRotateAngleValid = NO;//跟随态旋转角度是否生效
  18. displayParam.isAccuracyCircleShow = NO;//精度圈是否显示
  19. displayParam.locationViewOffsetX = 0;//定位偏移量(经度)
  20. displayParam.locationViewOffsetY = 0;//定位偏移量(纬度)
  21. displayParam.locationViewImgName = @"walk";
  22. [self.mapView updateLocationViewWithParam:displayParam];
  23. }
  1. /**
  2. * 初始化百度位置服务
  3. */
  4. - (void)initBMLocationService
  5. {
  6. // 初始化位置百度位置服务
  7. self.bmkLocationService = [[BMKLocationService alloc] init];
  8. // 设置距离过滤,表示每移动10更新一次位置
  9. [BMKLocationService setLocationDistanceFilter:10];
  10. // 设置定位精度
  11. [BMKLocationService setLocationDesiredAccuracy:kCLLocationAccuracyBest];
  12. }

2. 开始定位

  1. // 打开定位服务
  2. [self.bmkLocationService startUserLocationService];
  3. // 设置当前地图的显示范围,直接显示到用户位置
  4. BMKCoordinateRegion adjustRegion = [self.mapView regionThatFits:BMKCoordinateRegionMake(self.bmkLocationService.userLocation.location.coordinate, BMKCoordinateSpanMake(0.02f,0.02f))];
  5. [self.mapView setRegion:adjustRegion animated:YES];

只要遵守了BMKLocationServiceDelegate协议就可以获知位置更新的情况,需要实现下面几个代理方法:

  1. /**
  2. * 定位失败会调用该方法
  3. *
  4. * @param error 错误信息
  5. */
  6. - (void)didFailToLocateUserWithError:(NSError *)error
  7. {
  8. NSLog(@"did failed locate,error is %@",[error localizedDescription]);
  9. }
  10. /**
  11. * 用户位置更新后,会调用此函数
  12. * @param userLocation 新的用户位置
  13. */
  14. - (void)didUpdateBMKUserLocation:(BMKUserLocation *)userLocation
  15. {
  16. // 如果此时位置更新的水平精准度大于10米,直接返回该方法
  17. // 可以用来简单判断GPS的信号强度
  18. if (userLocation.location.horizontalAccuracy > kCLLocationAccuracyNearestTenMeters) {
  19. return;
  20. }
  21. }
  22. /**
  23. * 用户方向更新后,会调用此函数
  24. * @param userLocation 新的用户位置
  25. */
  26. - (void)didUpdateUserHeading:(BMKUserLocation *)userLocation
  27. {
  28. // 动态更新我的位置数据
  29. [self.mapView updateLocationData:userLocation];
  30. }

3. 存储更新的用户地理位置

  1. /**
  2. * 开始记录轨迹
  3. *
  4. * @param userLocation 实时更新的位置信息
  5. */
  6. - (void)recordTrackingWithUserLocation:(BMKUserLocation *)userLocation
  7. {
  8. if (self.preLocation) {
  9. // 计算本次定位数据与上次定位数据之间的距离
  10. CGFloat distance = [userLocation.location distanceFromLocation:self.preLocation];
  11. self.statusView.distanceWithPreLoc.text = [NSString stringWithFormat:@"%.3f",distance];
  12. NSLog(@"与上一位置点的距离为:%f",distance);
  13. // (5米门限值,存储数组画线) 如果距离少于 5 米,则忽略本次数据直接返回方法
  14. if (distance < 5) {
  15. return;
  16. }
  17. }
  18. // 2. 将符合的位置点存储到数组中(第一直接来到这里)
  19. [self.locationArrayM addObject:userLocation.location];
  20. self.preLocation = userLocation.location;
  21. // 3. 绘图
  22. [self drawWalkPolyline];
  23. }

4. 绘制轨迹线

  1. /**
  2. * 绘制轨迹路线
  3. */
  4. - (void)drawWalkPolyline
  5. {
  6. // 轨迹点数组个数
  7. NSUInteger count = self.locationArrayM.count;
  8. // 动态分配存储空间
  9. // BMKMapPoint是个结构体:地理坐标点,用直角地理坐标表示 X:横坐标 Y:纵坐标
  10. BMKMapPoint *tempPoints = new BMKMapPoint[count];
  11. // 遍历数组
  12. [self.locationArrayM enumerateObjectsUsingBlock:^(CLLocation *location, NSUInteger idx, BOOL *stop) {
  13. BMKMapPoint locationPoint = BMKMapPointForCoordinate(location.coordinate);
  14. tempPoints[idx] = locationPoint;
  15. }
  16. }];
  17. //移除原有的绘图,避免在原来轨迹上重画
  18. if (self.polyLine) {
  19. [self.mapView removeOverlay:self.polyLine];
  20. }
  21. // 通过points构建BMKPolyline
  22. self.polyLine = [BMKPolyline polylineWithPoints:tempPoints count:count];
  23. //添加路线,绘图
  24. if (self.polyLine) {
  25. [self.mapView addOverlay:self.polyLine];
  26. }
  27. // 清空 tempPoints 临时数组
  28. delete []tempPoints;
  29. // 根据polyline设置地图范围
  30. [self mapViewFitPolyLine:self.polyLine];
  31. }
添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注