问题描述及影响
在iOS17.4/17.4.1上,当新安装App的用户首次启动并获取IDFA权限时,此时会弹出IDFA授权弹窗让用户进行选择,但是用户此时并没有操作授权或者拒绝,系统就会收到用户拒绝的回调。
基于此,在iOS17.4/17.4.1上,如果有业务依赖此回调进行相关逻辑,都可能会受到影响。经过测试,在iOS17.5上,苹果已经修复此问题。
修复方案
如问题描述一样,在iOS17.4/17.4.1上,当新安装App的用户首次启动并获取IDFA权限时,此时会弹出IDFA授权弹窗让用户进行选择,但是用户此时并没有操作授权或者拒绝,系统就会收到用户拒绝的回调。
但是此时再获取IDFA授权状态,发现依然是ATTrackingManagerAuthorizationStatusNotDetermined
未授权,基于修复iOS15时的授权弹窗问题,在监听UIApplicationDidBecomeActiveNotification
通知时,用户授权操作完成后会触发此通知,所以有了修复此问题的方案。
IDFA授权回调方法中,当首次回调中状态返回ATTrackingManagerAuthorizationStatusDenied
时,对iOS17.4进行特殊判断
1、OC实现
swift
- (void)requestTrackingAuthorization {
if (@available(iOS 14, *)) {
[ATTrackingManager requestTrackingAuthorizationWithCompletionHandler:^(ATTrackingManagerAuthorizationStatus status) {
BOOL granted = YES;
switch (status) {
case ATTrackingManagerAuthorizationStatusDenied:
// NSLog(@"用户拒绝");
if (@available(iOS 17.4, *)) {
if (ATTrackingManager.trackingAuthorizationStatus == ATTrackingManagerAuthorizationStatusNotDetermined) {
[[NSNotificationCenter defaultCenter] addObserverForName:UIApplicationDidBecomeActiveNotification object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {
[self requestTrackingAuthorization];
}];
return;
}
}
granted = NO;
break;
case ATTrackingManagerAuthorizationStatusAuthorized:
// NSLog(@"用户允许");
break;
case ATTrackingManagerAuthorizationStatusNotDetermined:
// NSLog(@"用户未做选择或未弹窗");
granted = NO;
break;
default:
break;
}
//回调
~~~~
}];
} else {
// Fallback on earlier versions
}
}
2、swift实现
javascript
final class BugFixingATTrackingRequestManager {
class func requestTrackingAuthorization() async -> ATTrackingManager.AuthorizationStatus {
let status = await ATTrackingManager.requestTrackingAuthorization()
if status == .denied, ATTrackingManager.trackingAuthorizationStatus == .notDetermined {
debugPrint("iOS 17.4 ATT bug detected")
for await _ in await NotificationCenter.default.notifications(named: UIApplication.didBecomeActiveNotification) {
return await requestTrackingAuthorization()
}
}
return status
}
}