檢測 iOS 系統網絡權限被關閉


背景

一直都有用戶反饋無法正常聯網的問題,經過定位,發現很大一部分用戶是因為網絡權限被系統關閉,經過資料搜集和排除發現根本原因是:

  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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM