时间:2021-05-16 09:38:58 | 栏目:iOS代码 | 点击:次
最近要在 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>