最近要在 iOS 应用中添加位置信息, 需要满足的需求如下:
接下来逐步实现这三个需求。
在 iOS 上获取位置信息是很容易的, 网上的资料也很多, 我的代码如下:
// make sure location service is enabled.
if (!CLLocationManager.LocationServicesEnabled) {
return;
}
// create a new location manager
CLLocationManager locationManager = new CLLocationManager {
DistanceFilter = CLLocationDistance.FilterNone,
DesiredAccuracy = 1000
};
// check to work with both ios 6 and older.
if (UIDevice.CurrentDevice.CheckSystemVersion(6, 0)) {
locationManager.LocationsUpdated += OnLocationsUpdated;
}
else {
locationManager.UpdatedLocation += OnLocationUpdated;
}
locationManager.StartUpdatingLocation();
第一次运行时, 系统会弹出应用要使用位置的对话框, 点击确认即可。
当程序运行时, 用户直接锁定屏幕, 会调用 AppDelegate 的 DidEnterBackground 方法, 但是对于 iOS 应用来说, 用户切换程序或者点击 Home 按钮是, 也会调用这个函数, 这两种操作的区别在当前应用实例(UIApplication)的状态, 当用户切换程序或者点击 Home 按钮时, 应用的状态是 UIApplicationStateBackground , 而锁定屏幕时, 应用状态是 UIApplicationStateInactive , 使用下面的代码可以区分这两种情况:
public override void DidEnterBackground(UIApplication application) {
if (application.ApplicationState == UIApplicationState.Background) {
Log.Debug("App send to background by home button/switching to other app, stop upload location.");
}
else if (application.ApplicationState == UIApplicationState.Inactive) {
Log.Debug("App send to background by locking screen, contine upload location, but change mode to powersave mode");
}
}
对于锁屏情况下继续更新位置信息的需求, 就很容易达到了, 只要在 DidEnterBackground 函数中检查应用的状态, 当状态为 UIApplicationBackground 是才停止更新位置, 代码如下:
public override void DidEnterBackground(UIApplication application) {
// switch to other app or click home button, stop update location
if (application.ApplicationState == UIApplicationState.Background) {
locationManager.StopUpdatingLocation();
}
}
注意: 根据 Apple 的文档, DidEnterBackground 函数大约有 5 秒钟的执行时间, 如果超过 5 秒钟还没有从这个函数返回, 应用将会异常退出, 因此不要再这个函数中进行耗时的操作。
当应用切换到前台或者解锁时, 会调用 AppDelegate 的 WillEnterBackground 方法, 只要在这个方法中继续更新位置即可:
public override void WillEnterForeground(UIApplication application) {
// app switch to foreground, continue to update location.
locationManager.StartUpdatingLocation();
}
注意: 同样, WillEnterForeground 大约有 5 秒钟的执行时间, 因此也不要在这个函数中进行耗时的操作。
要真正能在锁屏状态下继续更新位置, 需要修改 info.plist ,增加后台位置权限。 找到项目中的 info.plist 文件, 双击打开, 并切换到 Source 标签, 如下图所示:
然后添加 Required background modes 属性项, 并将这个属性类型设置为 Array , 并为这个属性添加一个子项 Location-based information , 最终如下图所示:
也可以直接把下面的代码添加到 info.plist 文件中, 效果是一样的。
<key>UIBackgroundModes</key>
<array>
<string>location</string>
</array>