今天咱们就用用第三个传感器–磁力计–来做一个AR的场景。说到AR这个词,请大家不要喷我哈,并没有用到WWDC刚出的ARKit。而且今天这个例子重点是学习使用磁力计,本质上来讲和AR关系并不大。
磁力计跟前面的加速计、陀螺仪,都是用到了上次说的iOS当中的那个核心运动框架CoreMotion
, 也都用了CMMotionManager
。
完成后的效果,能看到在视频输出的下面会有一个随着屏幕移动的天空星辰背景图,同时屏幕左上角会实时打印当前的方向信息、地理信息。
1. 磁力计的介绍
磁强计指的是各种用于测量磁场的仪器,也称磁力仪、高斯计。它可以感应地球的磁场,获得方向信息。
1.1 应用场景
那显而易见,典型的应用场景就是用在电子罗盘和导航上面。
之前看到过某个大神用磁力计简直玩出了花儿,隔空抓牛的感觉。利用iPhone上磁力计、加速计和麦克风实现平面和三维上的磁铁追踪,并能实时的反馈在iPhone 屏幕上。
1.2 需要了解的基本概念
要用到磁力计,经常会听到有人说到“磁北”、“真北”这两个高频词,CoreMotion也会给我们返回这两个数值。是什么意思呐?
- 真北:指的是地理的北极
- 磁北:指的是磁场北极
纳尼?这是什么鬼?来来来,咱们科普一下。
- 磁北
磁北是以大地磁场为基准的,通过各种传感器传送的方位都是以磁北为基准的。BUT!!!!敲黑板!!!!!磁北的具体位置是随着时间而改变的。
也就是说咱们随着地球的旋转,咱们除了有一年四季的变化、时间的变化,连磁场都会发生改变。嗯,是这样的。 - 真北
由于磁北是会变化的,那我们怎么用?不可能还要计算地球自转轴、考虑时间因素吧。所以才有了真北这个概念。
真北是地球自转的地理北极,这个就是考虑到了各种因素,是一个固定的位置。所以咱们电子罗盘神马的所指的北,说的是这个真北。通常情况下,方位都是需要矫正到真北的。苹果很贴心啊,真北就不用自己算了,直接也会有返回的数值。
剩下的还有磁偏角校正、网络北、网络北校正、收敛角等等学术概念。
那岂不是电子罗盘上面的北和指南针上面的北不一致啊?
问这个问题的童鞋那是相当的聪明呀,那肯定是不一致的。不过误差也是在可感官接受的范围内。在等会儿的例子里面,咱们把这两个数值都打印出来,自己看看。
2. 磁力计的使用
2.1 使用步骤
磁力计同样也是通过CoreMotion
这个框架来管理的,所以和前面两个传感器一样,四个标准步骤:
- 初始化CMMotionManager管理对象;
- 调用管理对象的对象方法获取数据;
- 处理数据;
- 当不需要使用的时候,停止获取数据。
2.2 磁力计数据获取的两种方法
CoreMotion
中有2种获取数据方式,一种叫做PUSH的方式,一种叫做PULL的方式。顾名思义,PUSH就是被动的获取。设定完了之后,线程定时把获取到的数据推送回来。可想而知,对于资源的消耗是会稍微大一点的。PULL,就是要去索取。拉一下才会获取到数据。不要不给。上一次代码是Swift的,这一次咱们就使用OC啦。
2.2.1 PULL的方式
//PULL的方法获取数据 - (void)pullMagnetometer { // 判断磁力计是否可用 if (self.manager.magnetometerAvailable) { // 设置磁力计采样频率 self.manager.magnetometerUpdateInterval = 0.1; //开始更新,后台线程开始运行。这是Pull方式。 [self.manager startMagnetometerUpdates]; NSLog(@"X = %f,Y = %f,Z = %f",self.manager.magnetometerData.magneticField.x,self.manager.magnetometerData.magneticField.y,self.manager.magnetometerData.magneticField.z); } else { NSLog(@"It cannot be used!"); } }
2.2.2 PUSH的方式
//PUSH的方法获取数据 - (void)pushMagnetometer { // 判断磁力计是否可用 if (self.manager.magnetometerAvailable) { // 设置磁力计采样频率 self.manager.magnetometerUpdateInterval = 0.1; //Push方式获取和处理数据,这里我们一样只是做了简单的打印。把采样的工作放在了主线程中。 [self.manager startMagnetometerUpdatesToQueue:[NSOperationQueue mainQueue] withHandler:^(CMMagnetometerData * _Nullable magnetometerData, NSError * _Nullable error) { NSLog(@"X = %f,Y = %f,Z = %f",magnetometerData.magneticField.x,magnetometerData.magneticField.y,magnetometerData.magneticField.z); }]; } else { NSLog(@"It cannot be used!"); } }
3. 开始我们的小案例
写这个案例的时候发现要实现一个比较逼真的AR,咱们有好多东东都没有分享过。所以例子写完之后,写这篇文章的时候又对这个例子做了一些调整。大幅简化删减了了好多需求。但是,最后还是使用了相机、百度地图,如果这两个都不用,那真的一点都不能算是AR了。
完成后的效果,能看到在视频输出的下面会有一个随着屏幕移动的天空星辰背景图,同时屏幕左上角会实时打印当前的方向信息、地理信息。
小案例里面的相机不用紧张,咱们后面也还是会分享的。还有一个之前说过的,多线程也记得的哈,下一个系列就来补。
3.1 思维导图
3.2 准备工作
3.2.1 SDK的导入
这个例子里面咱们用了百度地图,所以需要导入百度地图的SDK。因为咱们没有分享过如何使用第三方库,可以看看这篇文章iOS·采用第三方(百度地图SDK)实现定位等功能开发
3.2.2 相机、定位权限的索取
Phone对于APP使用用户的隐私权限做了很严格的规定,每个APP使用用户隐私之前必须要让用户知道并且同意。大概也正是因为这点,本宅胖才这么爱iPhone吧。虽然开发的时候就面临着很多问题,但至少产品始终是站在用户的角度考虑问题的。
在Info.plist中向用户索取相机和地理位置信息的权限。
- Privacy – Camera Usage Description
- Privacy – Location When In Use Usage Description
麦克风、媒体库的权限就不需要了。之前没有删减的那个案例里面用到了这个。说起来好心疼~~~
3.2.3 相机使用
相机在这个案例里面,使用的是AVFoundation
框架。也是很心痛,这部分之前没有分享过。所以如果等不及俺的分享,可以先看看这个。Objc的第21期内容:iOS上的相机捕捉
别忘了在头文件<AVFoundation/AVFoundation.h>
,同时遵守代理协议AVCaptureVideoDataOutputSampleBufferDelegate
。
3.3 创建动态活动的星空背景
从网上找到的星空图是4000*2800的大小,要让它完全超出屏幕。这样才能根据手机的移动进行活动。
同样的,为了能够明显的看到效果,在从陀螺仪获取到的数值之后,添加了一个放大倍数。这个小例子里面咱们使用的是5。
3.3.1 使用陀螺仪进行防抖
如果陀螺仪返回的数据在某个特定小范围内,我们就是视同只是手抖,不对图片本身进行处理。这样就看不到背景图片明显抖动的感觉了。
- 注意:陀螺仪返回的各轴旋转角度是有可能为负数的,所以别忘了用绝对值进行判断。
// 做一下防抖动的处理,如果手机旋转的不太大,就不执行操作 if (fabs(gyroData.rotationRate.x) * multiplier < 0.2 && fabs(gyroData.rotationRate.y) * multiplier < 0.2) { return ; }
3.3.2 让背景星空图随着屏幕进行运行
直接修改背景图的center就好了,让原center添加上需要进行的位移量就可以实现了。
这里需要注意的是,需要对边界值进行处理。如果屏幕旋转的乱七八糟,我们要让视频输出层下面始终有一个背景存在。
// 因为背景图的大小事屏幕宽度的三倍,高度的两倍。为了防止超出边界,进行限制 if (imageRotationX > self.view.frame.size.width * 1.5) { imageRotationX = self.view.frame.size.width * 1.5; } if(imageRotationX < (- self.view.frame.size.width * 0.5)){ imageRotationX=(- self.view.frame.size.width * 0.5); } if (imageRotationY > self.view.frame.size.height) { imageRotationY = self.view.frame.size.height; } if (imageRotationY < 0) { imageRotationY = 0; }
3.4 利用百度地图输出磁力计信息、经纬度、高度
3.4.1 输出磁力计信息
根据百度地图SDK的文档,在用户的方向信息放生变化之后,会调用以下的方法。
这里我们没有做任何特殊的处理,就只是简单的打印出来了磁北、真北、三轴的偏移量。
等会儿运行的时候大家就能看到之前的问题,到底磁北、真北之间相差多少。
/** *用户方向更新后,会调用此函数 *@param userLocation 新的用户位置 */ - (void)didUpdateUserHeading:(BMKUserLocation *)userLocation { self.magnetometerInfo.numberOfLines = 0; self.magnetometerInfo.text = [NSString stringWithFormat:@"磁北:%.0f,真北:%.0f \n偏移:%.0f \nx:%.1f y:%.1f z:%.1f", userLocation.heading.magneticHeading,userLocation.heading.trueHeading,userLocation.heading.headingAccuracy,userLocation.heading.x,userLocation.heading.y,userLocation.heading.z]; }
3.4.2 输出用户位置信息:经纬度、高度
/** *用户位置更新后,会调用此函数 *@param userLocation 新的用户位置 */ - (void)didUpdateBMKUserLocation:(BMKUserLocation *)userLocation { self.physicalLocation.numberOfLines = 0; self.physicalLocation.text = [NSString stringWithFormat:@"经度:%f \n纬度:%f \n高度:%f",userLocation.location.coordinate.longitude,userLocation.location.coordinate.latitude,userLocation.location.altitude]; }