文章目录
- 前言
- I、向用户申请临时开启一次精确位置权限:不同产品场景可以定义不同的purposeKey
-
- 1.2.1 通过API设置不同的定位精确度
- 1.2.2 通过info.plist关闭精确定位权限的方法
- 1.1 注意事项
- 1.2 、控制授予APP的定位精度等级:根据不同的需求设置不同的定位精确度
- II、 高德定位SDK适配:“模糊定位”权限下的兼容策略
-
- 2.1、plist配置NSLocationTemporaryUsageDescriptionDictionary
- 2.2、locationAccuracyMode设置为AMapLocationFullAndReduceAccuracy/AMapLocationFullAccuracy
- 2. 3实现代理方法
- 2.4 处理定位权限状态改变的回调函数
前言
iOS14新增了 精确定位 和 模糊定位 的概念,用户可以手动选择,模糊定位的误差约 500m 。可以根据实际功能判断是否可以接受用户选择模糊定位。
- 在 iOS13 及以前,App 请求用户定位授权时为如下形态:一旦用户同意应用获取定位信息,当前应用就可以获取到用户的精确定位。
- iOS14 新增用户大致位置选项可供用户选择(原因是大多数 App 实际上并不需要获取用户到用户最准确的定位信息。)
iOS14 授权弹窗新增的 Precise的开关默认会选中精确位置。用户通过这个开关可以进行更改,当把这个值设为 On 时,地图上会显示精确位置;切换为Off时,将显示用户的大致位置。
- 对于对用户位置敏感度不高的 App 来说,这个似乎无影响,但是对于强依赖精确位置的 App 适配工作就显得非常重要了。
用户可以通过在 “隐私设置” 中设置来开启精确定位,但是若用户不愿意开启。这个时候,iOS14 在 CLLocationManager 新增两个方法用于向用户申请临时开启一次精确位置权限。
I、向用户申请临时开启一次精确位置权限:不同产品场景可以定义不同的purposeKey
申请的单次精准定位本次APP生命周期内有效,APP一个生命周期内可以多次申请。任何一次允许后则可正常获取精准定位。申请方式:需要在info.plist中配置NSLocationTemporaryUsageDescriptionDictionary
:
NSLocationTemporaryUsageDescriptionDictionary的value是字典;
其中中需要配置 key 和 value 表明使用位置的原因,以及具体的描述。
不同产品场景可以定义不同的purposeKey。
<key>NSLocationTemporaryUsageDescriptionDictionary</key>
<dict>
<key> purposeKey </key>
<string>This app needs accurate location so it can verify that you're in a supported region. </string>
<key> AnotherPurposeKey </key>
<string>This app needs accurate location so it can show you relevant results.</string>
</dict>
- 在需要的时候调用
requestTemporaryFullAccuracyAuthorizationWithPurposeKey
单独请求一次精确定位
//参数 purposeKey 是在 info.plist 中设置NSLocationTemporaryUsageDescriptionDictionary 字典时候的key `
CLLocationManager *manager = [[CLLocationManager alloc] init];
manager.desiredAccuracy = kCLLocationAccuracyBest;
// 通过字段判断可以明确当前用户开启的定位,如果是模糊定位
// 申请精准定位
if (@available(iOS 14.0, *)) {
// -1
if( manager.accuracyAuthorization == CLAccuracyAuthorizationReducedAccuracy){
if (@available(iOS 14.0, *)) {
[manager requestTemporaryFullAccuracyAuthorizationWithPurposeKey:@"purposeKey4changeInfo" completion:^(NSError * _Nullable error) {
NSLog(@"%@", [NSString stringWithFormat:@"requestTemporaryFullAccuracyAuthorizationWithPurposeKey error%@",error]);
}];
} else {
// Fallback on earlier versions
}
}
} else {
// Fallback on earlier versions
}
这样app会获取到临时的精确位置权限直至下次冷启动。当然这个授权也可能被用户无情拒绝。
当然如果你使用第三方SDK进行定位的话,请升级版本即可
- 效果
1.1 注意事项
- 判断通过字段判断可以明确当前用户开启的定位,如果是模糊定位申请精准定位
- purposeKey 注意是否包含空格,因为一旦包含了空格,你的代码没有加空格的话,是不会弹授权框的。
1.2 、控制授予APP的定位精度等级:根据不同的需求设置不同的定位精确度
1.2.1 通过API设置不同的定位精确度
kCLLocationAccuracyReduced
是iOS14才支持的
*
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyBestForNavigation API_AVAILABLE(ios(4.0), macos(10.7));
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyBest;
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyNearestTenMeters;
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyHundredMeters;
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyKilometer;
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyThreeKilometers;
/*
* kCLLocationAccuracyReduced
*
* Discussion:
* If you set your CLLocationManager's desiredAccuracy property to
* this value, locations delivered to your delegate in response to
* startUpdatingLocation or requestLocation will have their
* accuracy reduced. The locations you receive will match the
* locations your app would have received if the user had decided
* not to grant your app authorization for precise location.
*/
CL_EXTERN const CLLocationAccuracy kCLLocationAccuracyReduced API_AVAILABLE(ios(14.0), macos(11.0), watchos(7.0), tvos(14.0));
locationManager = [[CLLocationManager alloc] init];
locationManager.desiredAccuracy = kCLLocationAccuracyReduced;
1.2.2 通过info.plist关闭精确定位权限的方法
✎ 对于地理位置不敏感的App 来说,iOS14 也可以通过直接在 info.plist 中添加 NSLocationDefaultAccuracyReduced 为 true 默认请求大概位置。
设置之后,即使用户想要为该 App 开启精确定位权限,也无法开启。
- 注意:当 App 在 Background 模式下,如果并未获得精确位置授权,那么 Beacon 及其他位置敏感功能都将受到限制。
II、 高德定位SDK适配:“模糊定位”权限下的兼容策略
代码语言:javascript复制 pod 'AMapLocation', '2.6.7'
2.1、plist配置NSLocationTemporaryUsageDescriptionDictionary
代码语言:javascript复制 <key>NSLocationTemporaryUsageDescriptionDictionary</key>
<dict>
<key>purposeKey4changeInfo</key>
<string>此app需要精确的定位信息,以便于更好的为你服务. </string>
</dict>
2.2、locationAccuracyMode设置为AMapLocationFullAndReduceAccuracy/AMapLocationFullAccuracy
代码语言:javascript复制if (@available(iOS 14.0, *)) {
self.location.locationAccuracyMode = AMapLocationFullAndReduceAccuracy;
} else {
// Fallback on earlier versions
}
2. 3实现代理方法
代码语言:javascript复制/**
* @brief 当plist配置NSLocationTemporaryUsageDescriptionDictionary且desiredAccuracyMode设置CLAccuracyAuthorizationFullAccuracy精确定位模式时,如果用户只授权模糊定位,会调用代理的此方法。此方法实现调用申请临时精确定位权限API即可:
* [manager requestTemporaryFullAccuracyAuthorizationWithPurposeKey:@"PurposeKey" completion:^(NSError *error){
* if(completion){
* completion(error);
* }
* }]; (必须调用,不然无法正常获取临时精确定位权限)
* @param manager 定位 AMapLocationManager 类。
* @param locationManager 需要申请临时精确定位权限的locationManager。
* @param completion 临时精确定位权限API回调结果,error: 直接返回系统error即可。
* @since 2.6.7
*/
- (void)amapLocationManager:(AMapLocationManager *)manager doRequireTemporaryFullAccuracyAuth:(CLLocationManager*)locationManager completion:(void(^)(NSError *error))completion;
{
if(@available(iOS 14.0,*)){
[locationManager requestTemporaryFullAccuracyAuthorizationWithPurposeKey:@"purposeKey4changeInfo" completion:^(NSError * _Nullable error) {
if(completion){
completion(error);
}
}];
}
}
2.4 处理定位权限状态改变的回调函数
- 如果定位精度权限变更为精确的时候,再次更新定位信息
/**
记录当前是否向用户申请临时开启一次精确位置权限,用于【如果定位精度权限变更为精确的时候,再次更新定位信息】
*/
@property (assign, nonatomic) BOOL isrequestTemporaryFullAccuracyAuthorizationWithPurposeKey;
/**
存储获取定位信息的回调
*/
@property (nonatomic, copy) AMapLocatingCompletionBlock block;
/**
* @brief 定位权限状态改变时回调函数。注意:iOS14及之后版本回调
* @param manager 定位 AMapLocationManager 类。
* @param locationManager 定位CLLocationManager类,可通过locationManager.authorizationStatus获取定位权限,通过locationManager.accuracyAuthorization获取定位精度权限
*/
- (void)amapLocationManager:(AMapLocationManager *)manager locationManagerDidChangeAuthorization:(CLLocationManager*)locationManager{
//
// - 如果定位精度权限变更为精确的时候,再次更新定位信息
if (@available(iOS 14.0, *)) {
if( locationManager.accuracyAuthorization == CLAccuracyAuthorizationFullAccuracy){
if( self.isrequestTemporaryFullAccuracyAuthorizationWithPurposeKey == YES){
self.isrequestTemporaryFullAccuracyAuthorizationWithPurposeKey = NO;
[self.location requestLocationWithReGeocode:YES completionBlock:^(CLLocation *location, AMapLocationReGeocode *regeocode, NSError *error) {
[SVProgressHUD dismiss];
if(self.block){
self.block(location,regeocode,error);
}
}];
}
}
} else {
// Fallback on earlier versions
}
}
- 效果
- 定位SDK适配文档详情请见:
在iOS14之后的SDK新增的API
- 高德
- 百度iOS_14_beta版适配说明
-> Installing AMapLocation 2.6.7 (was 2.6.0)
> Http download
$ /usr/bin/curl -f -L -o /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/file.zip
https://amappc.oss-cn-zhangjiakou.aliyuncs.com/lbs/static/zip/AMap_iOS_Loc_Lib_V2.6.7.zip --create-dirs --netrc-optional --retry 2
% Total % Received % Xferd Average Speed Time Time Time Current
Dload Upload Total Spent Left Speed
100 2165k 100 2165k 0 0 553k 0 0:00:03 0:00:03 --:--:-- 553k
$ /usr/bin/unzip /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/file.zip -d /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su
Archive: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/file.zip
creating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/
inflating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/AMapLocationKit
creating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/Headers/
inflating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/Headers/AMapLocationKit.h
inflating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/Headers/AMapGeoFenceError.h
inflating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/Headers/AMapGeoFenceManager.h
inflating: /var/folders/6t/z1p2xgls4wz_pw36r58hhspr0000gn/T/d20200919-24852-1mkf6su/AMapLocationKit.framework/Headers/AMapLocationRegionObj.h