38

检测 iOS 系统网络权限被关闭

 5 years ago
source link: http://www.cocoachina.com/ios/20180723/24274.html?amp%3Butm_medium=referral
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

背景

一直都有用户反馈无法正常联网的问题,经过定位,发现很大一部分用户是因为网络权限被系统关闭,经过资料搜集和排除发现根本原因是:

  1. 第一次打开 app 不能访问网络,无任何提示

  2. 第一次打开 app 直接提示「已为“XXX”关闭网络」

  3. 第一次打开 app ,用户点错了选择了「不允许」或「WLAN」

对于第 1 种情况,出现在 iOS 10 比较多,一旦出现后系统设置里也找不到「无线数据」这一配置选项,随着 iOS 的更新,貌似被 Apple 修复了,GitHub 上面有 ZIKCellularAuthorization 其进行分析和提出一种解决方案,强制让系统弹出那个询问框。

但是第 2、3种情况现在 iOS 12 还经常有发生,对于这种情况,我们只要检测出来,并提示引导用户去打开网络权限即可,本文提出一新的方法来检测这种情况。

CTCellularData 的局限性

关于网络权限问题,网络上搜集的资料大多数提到了用 CTCellularData 的 cellularDataRestrictionDidUpdateNotifier 方法去判断网络权限关闭,但这样判断会有不完善的情况(后面提到)

CoreTelephony 里的 CTCellularData 可以用来监测 app 的蜂窝网络权限,其定义如下:

typedef NS_ENUM(NSUInteger, CTCellularDataRestrictedState) {
    kCTCellularDataRestrictedStateUnknown, 
    kCTCellularDataRestricted,            
    kCTCellularDataNotRestricted          
};

通过注册 cellularDataRestrictionDidUpdateNotifier 回调可以并判断其 state 可以判断蜂窝数据的权限

CTCellularData *cellularData = [[CTCellularData alloc] init];
    cellularData.cellularDataRestrictionDidUpdateNotifier = ^(CTCellularDataRestrictedState restrictedState) {
           ...
        }
    };

系统设置里 有三种选项分别对应:

  • 系统选项     CTCellularDataRestrictedState

  • 关闭     kCTCellularDataRestricted

  • WLAN     kCTCellularDataRestricted

  • WALN 与蜂窝移动网     kCTCellularDataNotRestricted

实测发现:

1、若用户此时用蜂窝数据上网,但在「允许“XXX”使用的数据」,选择了「WLAN」 或 「关闭」,回调拿到的值是

kCTCellularDataRestricted ,此时我们可以确定是因为权限问题导致用户不能访问,应该去提示用户打开网络权限。

2、若用户此时用 Wi-Fi 上网,但在「允许“XXX”使用的数据」设置中选择了 「关闭」,我们拿到的值是 kCTCellularDataRestricted ,这种情况下同样需要提示用户打开网络权限。

3、若用户此时用 Wi-Fi 上网,但在「允许“XXX”使用的数据」设置中选择了 「WLAN」,我们拿到的值是 kCTCellularDataRestricted ,但是此时用户是有网络访问权限的,此时不应该去提示用户。

判断思路

结合 SCNetworkReachabilityRef 的回调,以及对网络状态的区分来判断:

通过判断 SCNetworkReachabilityRef 回调的 flag 发现 kSCNetworkReachabilityFlagsReachable 为 0,则说明网络不通,此时可能有两种情况:

  1. 未打开任何数据连接(Wi-Fi 蜂窝数据)或者开启了飞行模式

  2. 网络权限被关闭

所以我们的判断思路就是要判断出用户是否 开启了 Wi-Fi 或者 蜂窝数据,如果都不是那必定是网络权限被关闭。

实现细节

判断当前网络类型

思路:

1、先通过 CaptiveNetwork 去判断有没有开启 Wi-Fi,这个判断无论在网络权限是否打开下的判断都是准确的。

2、由于在没有网络权限的情况下,没有办法直接去判断是否开启了蜂窝数据,这里只能通过一种比较 trick 的方式,通过状态栏去判断用户是否开启了蜂窝数据,但是在一些极端的情况下,不一定准确,比如用户同时开启 Wi-Fi 和蜂窝数据,此时先关闭 Wi-Fi 然后迅速关闭蜂窝数据,此时手机处于无网络状态,我们在第 1 步判断出了 Wi-Fi 不可用,但是通过状态栏的方式拿到却还是 Wi-Fi,在这种比较边界的情况下,只能延时一会儿再次检查。

- (void)getCurrentNetworkType:(void(^)(ZYNetworkType))block {
    if ([self isWiFiEnable]) {
        return block(ZYNetworkTypeWiFi);
    }
   ZYNetworkType type = [self getNetworkTypeFromStatusBar];
    if (type == ZYNetworkTypeOffline) {
        block(ZYNetworkTypeOffline);
    } else if (type == ZYNetworkTypeWiFi) { // 这时候从状态栏拿到的是 Wi-Fi 说明状态栏没有刷新,延迟一会再获取
        dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.1 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
            [self getCurrentNetworkType:block];
        });
    } else {
        block(ZYNetworkTypeCellularData);
    }
}

判断是否连接到 Wi-Fi

判断 Wi-Fi 的方法比较简单,导入 SystemConfiguration/CaptiveNetwork.h 并使用下面方法判断即可

- (BOOL)isWiFiEnable {
    NSArray *interfaces = (__bridge_transfer NSArray *)CNCopySupportedInterfaces();
    if (!interfaces) {
        return NO;
    }
    NSDictionary *info = nil;
    for (NSString *ifnam in interfaces) {
        info = (__bridge_transfer NSDictionary *)CNCopyCurrentNetworkInfo((__bridge CFStringRef)ifnam);
        if (info && [info count]) { break; }
    }
    return (info != nil);
}

从状态栏判断网络类型

上面提到,由于在网络权限拒绝的情况下,我们唯一比较有效的方法是通过状态栏去判断,这个判断方法在网上可以找到,但是 在 iPhone X 会出现 crash 的情况,我针对 iPhone X 做了补充和适配。

- (ZYNetworkType)getNetworkTypeFromStatusBar {
    NSInteger type = 0;
    @try {
        UIApplication *app = [UIApplication sharedApplication];
        UIView *statusBar = [app valueForKeyPath:@"statusBar"];

        if (statusBar == nil ){
            return ZYNetworkTypeUnknown;
        }
        BOOL isModernStatusBar = [statusBar isKindOfClass:NSClassFromString(@"UIStatusBar_Modern")];
        if (isModernStatusBar) { // 在 iPhone X 上 statusBar 属于 UIStatusBar_Modern ,需要特殊处理
            id currentData = [statusBar valueForKeyPath:@"statusBar.currentData"];
            BOOL wifiEnable = [[currentData valueForKeyPath:@"_wifiEntry.isEnabled"] boolValue];
            // 这里不能用 _cellularEntry.isEnabled 来判断,该值即使关闭仍然有是 YES
            BOOL cellularEnable = [[currentData valueForKeyPath:@"_cellularEntry.type"] boolValue];
            return  wifiEnable     ? ZYNetworkTypeWiFi :
                    cellularEnable ? ZYNetworkTypeCellularData : ZYNetworkTypeOffline;

        } else { // 传统的 statusBar
            NSArray *children = [[statusBar valueForKeyPath:@"foregroundView"] subviews];
            for (id child in children) {
                if ([child isKindOfClass:[NSClassFromString(@"UIStatusBarDataNetworkItemView") class]]) {
                    type = [[child valueForKeyPath:@"dataNetworkType"] intValue];
                    // type == 1  => 2G
                    // type == 2  => 3G
                    // type == 3  => 4G
                    // type == 4  => LTE
                    // type == 5  => Wi-Fi
                }
            }
            return type == 0 ? ZYNetworkTypeOffline :
                   type == 5 ? ZYNetworkTypeWiFi    : ZYNetworkTypeCellularData;
        }
    } @catch (NSException *exception) {
    }
    return 0;
}

整体判断代码

- (void)startCheck {

    /* iOS 10 以下默认通过 **/

    /* 先用 currentReachable 判断,若返回的为 YES 则说明:
     1. 用户选择了 「WALN 与蜂窝移动网」并处于其中一种网络环境下。
     2. 用户选择了 「WALN」并处于 WALN 网络环境下。

     此时是有网络访问权限的,直接返回 ZYNetworkAccessible
    **/
    if ([UIDevice currentDevice].systemVersion.floatValue < 10.0 || [self currentReachable]) {
        [self notiWithAccessibleState:ZYNetworkAccessible];
        return;
    }

    CTCellularDataRestrictedState state = _cellularData.restrictedState;

    switch (state) {
        case kCTCellularDataRestricted: {// 系统 API 返回 无蜂窝数据访问权限

            [self getCurrentNetworkType:^(ZYNetworkType type) {
                /*  若用户是通过蜂窝数据 或 WLAN 上网,走到这里来 说明权限被关闭**/

                if (type == ZYNetworkTypeCellularData || type == ZYNetworkTypeWiFi) {
                    [self notiWithAccessibleState:ZYNetworkRestricted];
                } else {  // 可能开了飞行模式,无法判断
                    [self notiWithAccessibleState:ZYNetworkUnknown];
                }
            }];

            break;
        }
        case kCTCellularDataNotRestricted: // 系统 API 访问有有蜂窝数据访问权限,那就必定有 Wi-Fi 数据访问权限
            [self notiWithAccessibleState:ZYNetworkAccessible];
            break;
        case kCTCellularDataRestrictedStateUnknown:
            [self notiWithAccessibleState:ZYNetworkUnknown];
            break;
        default:
            break;
    };
}

ZYNetworkAccessibity

GitHub : ZYNetworkAccessibity

我已经把上面的方法做了封装,将 ZYNetworkAccessibity.h 和 ZYNetworkAccessibity.m 拖项目中,监听 ZYNetworkAccessibityChangedNotification 通知即可

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(networkChanged:) name:ZYNetworkAccessibityChangedNotification object:nil];

然后处理通知

- (void)networkChanged:(NSNotification *)notification {

    ZYNetworkAccessibleState state = ZYNetworkAccessibity.currentState;

    if (state == ZYNetworkRestricted) {
        NSLog(@"网络权限被关闭");
    }
}

另外还实现了自动提醒用户打开权限,如果你需要,请打开

[ZYNetworkAccessibity setAlertEnable:YES];

作者:ziecho

链接:https://www.jianshu.com/p/81d0b7f06eba


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK