檢測(cè) iOS 系統(tǒng)網(wǎng)絡(luò)權(quán)限被關(guān)閉

背景

一直都有用戶反饋無(wú)法正常聯(lián)網(wǎng)的問題橄霉,經(jīng)過(guò)定位晌姚,發(fā)現(xiàn)很大一部分用戶是因?yàn)榫W(wǎng)絡(luò)權(quán)限被系統(tǒng)關(guān)閉,經(jīng)過(guò)資料搜集和排除發(fā)現(xiàn)根本原因是:

1痘括、第一次打開 app 不能訪問網(wǎng)絡(luò)长窄,無(wú)任何提示
2滔吠、第一次打開 app 直接提示「已為“XXX”關(guān)閉網(wǎng)絡(luò)」
3、第一次打開 app 挠日,用戶點(diǎn)錯(cuò)了選擇了「不允許」或「WLAN」

對(duì)于第 1 種情況疮绷,出現(xiàn)在 iOS 10 比較多,一旦出現(xiàn)后系統(tǒng)設(shè)置里也找不到「無(wú)線數(shù)據(jù)」這一配置選項(xiàng)嚣潜,隨著 iOS 的更新冬骚,貌似被 Apple 修復(fù)了,GitHub 上面有 ZIKCellularAuthorization 其進(jìn)行分析和提出一種解決方案懂算,強(qiáng)制讓系統(tǒng)彈出那個(gè)詢問框只冻。

但是第 2、3種情況現(xiàn)在 iOS 12 還經(jīng)常有發(fā)生计技,對(duì)于這種情況喜德,我們只要檢測(cè)出來(lái),并提示引導(dǎo)用戶去打開網(wǎng)絡(luò)權(quán)限即可垮媒,本文提出一新的方法來(lái)檢測(cè)這種情況舍悯。

CTCellularData 的局限性

關(guān)于網(wǎng)絡(luò)權(quán)限問題,網(wǎng)絡(luò)上搜集的資料大多數(shù)提到了用 CTCellularData 的 cellularDataRestrictionDidUpdateNotifier 方法去判斷網(wǎng)絡(luò)權(quán)限關(guān)閉睡雇,但這樣判斷會(huì)有不完善的情況(后面提到)

CoreTelephony 里的 CTCellularData 可以用來(lái)監(jiān)測(cè) app 的蜂窩網(wǎng)絡(luò)權(quán)限萌衬,其定義如下:

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

通過(guò)注冊(cè) cellularDataRestrictionDidUpdateNotifier 回調(diào)可以并判斷其 state 可以判斷蜂窩數(shù)據(jù)的權(quán)限

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

系統(tǒng)設(shè)置里 有三種選項(xiàng)分別對(duì)應(yīng):

系統(tǒng)選項(xiàng) CTCellularDataRestrictedState
關(guān)閉 kCTCellularDataRestricted
WLAN kCTCellularDataRestricted
WALN 與蜂窩移動(dòng)網(wǎng) kCTCellularDataNotRestricted

實(shí)測(cè)發(fā)現(xiàn):
1、若用戶此時(shí)用蜂窩數(shù)據(jù)上網(wǎng)入桂,但在「允許“XXX”使用的數(shù)據(jù)」奄薇,選擇了「WLAN」 或 「關(guān)閉」,回調(diào)拿到的值是
kCTCellularDataRestricted 抗愁,此時(shí)我們可以確定是因?yàn)闄?quán)限問題導(dǎo)致用戶不能訪問馁蒂,應(yīng)該去提示用戶打開網(wǎng)絡(luò)權(quán)限。

2蜘腌、若用戶此時(shí)用 Wi-Fi 上網(wǎng)沫屡,但在「允許“XXX”使用的數(shù)據(jù)」設(shè)置中選擇了 「關(guān)閉」,我們拿到的值是 kCTCellularDataRestricted 撮珠,這種情況下同樣需要提示用戶打開網(wǎng)絡(luò)權(quán)限沮脖。

3、若用戶此時(shí)用 Wi-Fi 上網(wǎng)芯急,但在「允許“XXX”使用的數(shù)據(jù)」設(shè)置中選擇了 「WLAN」勺届,我們拿到的值是 kCTCellularDataRestricted ,但是此時(shí)用戶是有網(wǎng)絡(luò)訪問權(quán)限的娶耍,此時(shí)不應(yīng)該去提示用戶免姿。

判斷思路

結(jié)合 SCNetworkReachabilityRef 的回調(diào),以及對(duì)網(wǎng)絡(luò)狀態(tài)的區(qū)分來(lái)判斷:
通過(guò)判斷 SCNetworkReachabilityRef 回調(diào)的 flag 發(fā)現(xiàn) kSCNetworkReachabilityFlagsReachable 為 0榕酒,則說(shuō)明網(wǎng)絡(luò)不通胚膊,此時(shí)可能有兩種情況:
1故俐、未打開任何數(shù)據(jù)連接(Wi-Fi 蜂窩數(shù)據(jù))或者開啟了飛行模式
2、網(wǎng)絡(luò)權(quán)限被關(guān)閉

所以我們的判斷思路就是要判斷出用戶是否 開啟了 Wi-Fi 或者 蜂窩數(shù)據(jù)紊婉,如果都不是那必定是網(wǎng)絡(luò)權(quán)限被關(guān)閉药版。

實(shí)現(xiàn)細(xì)節(jié)

判斷當(dāng)前網(wǎng)絡(luò)類型

思路:
1、先通過(guò) CaptiveNetwork 去判斷有沒有開啟 Wi-Fi喻犁,這個(gè)判斷無(wú)論在網(wǎng)絡(luò)權(quán)限是否打開下的判斷都是準(zhǔn)確的槽片。
2、由于在沒有網(wǎng)絡(luò)權(quán)限的情況下株汉,沒有辦法直接去判斷是否開啟了蜂窩數(shù)據(jù)筐乳,這里只能通過(guò)一種比較 trick 的方式,通過(guò)狀態(tài)欄去判斷用戶是否開啟了蜂窩數(shù)據(jù)乔妈,但是在一些極端的情況下蝙云,不一定準(zhǔn)確,比如用戶同時(shí)開啟 Wi-Fi 和蜂窩數(shù)據(jù)路召,此時(shí)先關(guān)閉 Wi-Fi 然后迅速關(guān)閉蜂窩數(shù)據(jù)勃刨,此時(shí)手機(jī)處于無(wú)網(wǎng)絡(luò)狀態(tài),我們?cè)诘?1 步判斷出了 Wi-Fi 不可用股淡,但是通過(guò)狀態(tài)欄的方式拿到卻還是 Wi-Fi身隐,在這種比較邊界的情況下,只能延時(shí)一會(huì)兒再次檢查唯灵。

- (void)getCurrentNetworkType:(void(^)(ZYNetworkType))block {
    if ([self isWiFiEnable]) {
        return block(ZYNetworkTypeWiFi);
    }
   ZYNetworkType type = [self getNetworkTypeFromStatusBar];
    if (type == ZYNetworkTypeOffline) {
        block(ZYNetworkTypeOffline);
    } else if (type == ZYNetworkTypeWiFi) { // 這時(shí)候從狀態(tài)欄拿到的是 Wi-Fi 說(shuō)明狀態(tài)欄沒有刷新贾铝,延遲一會(huì)再獲取
        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 的方法比較簡(jiǎn)單,導(dǎo)入 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);
}

從狀態(tài)欄判斷網(wǎng)絡(luò)類型

上面提到埠帕,由于在網(wǎng)絡(luò)權(quán)限拒絕的情況下垢揩,我們唯一比較有效的方法是通過(guò)狀態(tài)欄去判斷,這個(gè)判斷方法在網(wǎng)上可以找到敛瓷,但是 在 iPhone X 會(huì)出現(xiàn) crash 的情況叁巨,我針對(duì) iPhone X 做了補(bǔ)充和適配。

- (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 來(lái)判斷锋勺,該值即使關(guān)閉仍然有是 YES
            BOOL cellularEnable = [[currentData valueForKeyPath:@"_cellularEntry.type"] boolValue];
            return  wifiEnable     ? ZYNetworkTypeWiFi :
                    cellularEnable ? ZYNetworkTypeCellularData : ZYNetworkTypeOffline;
    
        } else { // 傳統(tǒng)的 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 以下默認(rèn)通過(guò) **/
    
    /* 先用 currentReachable 判斷,若返回的為 YES 則說(shuō)明:
     1. 用戶選擇了 「WALN 與蜂窩移動(dòng)網(wǎng)」并處于其中一種網(wǎng)絡(luò)環(huán)境下狡蝶。
     2. 用戶選擇了 「WALN」并處于 WALN 網(wǎng)絡(luò)環(huán)境下庶橱。
     
     此時(shí)是有網(wǎng)絡(luò)訪問權(quán)限的,直接返回 ZYNetworkAccessible
    **/
    if ([UIDevice currentDevice].systemVersion.floatValue < 10.0 || [self currentReachable]) {
        [self notiWithAccessibleState:ZYNetworkAccessible];
        return;
    }
    
    CTCellularDataRestrictedState state = _cellularData.restrictedState;
    
    switch (state) {
        case kCTCellularDataRestricted: {// 系統(tǒng) API 返回 無(wú)蜂窩數(shù)據(jù)訪問權(quán)限
            
            [self getCurrentNetworkType:^(ZYNetworkType type) {
                /*  若用戶是通過(guò)蜂窩數(shù)據(jù) 或 WLAN 上網(wǎng)贪惹,走到這里來(lái) 說(shuō)明權(quán)限被關(guān)閉**/
                
                if (type == ZYNetworkTypeCellularData || type == ZYNetworkTypeWiFi) {
                    [self notiWithAccessibleState:ZYNetworkRestricted];
                } else {  // 可能開了飛行模式苏章,無(wú)法判斷
                    [self notiWithAccessibleState:ZYNetworkUnknown];
                }
            }];
            
            break;
        }
        case kCTCellularDataNotRestricted: // 系統(tǒng) API 訪問有有蜂窩數(shù)據(jù)訪問權(quán)限,那就必定有 Wi-Fi 數(shù)據(jù)訪問權(quán)限
            [self notiWithAccessibleState:ZYNetworkAccessible];
            break;
        case kCTCellularDataRestrictedStateUnknown:
            [self notiWithAccessibleState:ZYNetworkUnknown];
            break;
        default:
            break;
    };
}

ZYNetworkAccessibity

GitHub : ZYNetworkAccessibity

我已經(jīng)把上面的方法做了封裝馍乙,將 ZYNetworkAccessibity.h 和 ZYNetworkAccessibity.m 拖項(xiàng)目中布近,監(jiān)聽 ZYNetworkAccessibityChangedNotification 通知即可

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

然后處理通知

- (void)networkChanged:(NSNotification *)notification {
    
    ZYNetworkAccessibleState state = ZYNetworkAccessibity.currentState;

    if (state == ZYNetworkRestricted) {
        NSLog(@"網(wǎng)絡(luò)權(quán)限被關(guān)閉");
    }
}

另外還實(shí)現(xiàn)了自動(dòng)提醒用戶打開權(quán)限,如果你需要丝格,請(qǐng)打開

[ZYNetworkAccessibity setAlertEnable:YES];
image.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末撑瞧,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子显蝌,更是在濱河造成了極大的恐慌预伺,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,576評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件曼尊,死亡現(xiàn)場(chǎng)離奇詭異酬诀,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)骆撇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,515評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門瞒御,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人神郊,你說(shuō)我怎么就攤上這事肴裙。” “怎么了涌乳?”我有些...
    開封第一講書人閱讀 168,017評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵蜻懦,是天一觀的道長(zhǎng)。 經(jīng)常有香客問我夕晓,道長(zhǎng)宛乃,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,626評(píng)論 1 296
  • 正文 為了忘掉前任蒸辆,我火速辦了婚禮征炼,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘吁朦。我一直安慰自己柒室,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,625評(píng)論 6 397
  • 文/花漫 我一把揭開白布逗宜。 她就那樣靜靜地躺著雄右,像睡著了一般。 火紅的嫁衣襯著肌膚如雪纺讲。 梳的紋絲不亂的頭發(fā)上擂仍,一...
    開封第一講書人閱讀 52,255評(píng)論 1 308
  • 那天,我揣著相機(jī)與錄音熬甚,去河邊找鬼逢渔。 笑死,一個(gè)胖子當(dāng)著我的面吹牛乡括,可吹牛的內(nèi)容都是我干的肃廓。 我是一名探鬼主播智厌,決...
    沈念sama閱讀 40,825評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼盲赊!你這毒婦竟也來(lái)了铣鹏?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,729評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤哀蘑,失蹤者是張志新(化名)和其女友劉穎诚卸,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體绘迁,經(jīng)...
    沈念sama閱讀 46,271評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡合溺,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,363評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了缀台。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片棠赛。...
    茶點(diǎn)故事閱讀 40,498評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖膛腐,靈堂內(nèi)的尸體忽然破棺而出恭朗,到底是詐尸還是另有隱情,我是刑警寧澤依疼,帶...
    沈念sama閱讀 36,183評(píng)論 5 350
  • 正文 年R本政府宣布痰腮,位于F島的核電站,受9級(jí)特大地震影響律罢,放射性物質(zhì)發(fā)生泄漏膀值。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,867評(píng)論 3 333
  • 文/蒙蒙 一误辑、第九天 我趴在偏房一處隱蔽的房頂上張望沧踏。 院中可真熱鬧,春花似錦巾钉、人聲如沸翘狱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,338評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)潦匈。三九已至,卻和暖如春赚导,著一層夾襖步出監(jiān)牢的瞬間茬缩,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,458評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工吼旧, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留凰锡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,906評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像掂为,于是被迫代替她去往敵國(guó)和親裕膀。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,507評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容

  • 一勇哗、Python簡(jiǎn)介和環(huán)境搭建以及pip的安裝 4課時(shí)實(shí)驗(yàn)課主要內(nèi)容 【Python簡(jiǎn)介】: Python 是一個(gè)...
    _小老虎_閱讀 5,748評(píng)論 0 10
  • 應(yīng)用間通信 應(yīng)用程式只能間接與設(shè)備上的其他應(yīng)用進(jìn)行通信魂角。您可以使用AirDrop與其他應(yīng)用程序共享文件和數(shù)據(jù)。您還...
    nicedayCoco閱讀 746評(píng)論 0 1
  • 文/霞客 今晨讀了高適的《燕歌行》智绸,最豪氣雄壯的一句,自是“男兒本自重橫行”访忿。男兒啊瞧栗,何謂真正的男兒?...
    大觀學(xué)園閱讀 313評(píng)論 1 0
  • 汽車電商行業(yè)和傳統(tǒng)電商行業(yè)略有不同海铆,傳統(tǒng)電商行業(yè)自亞馬遜B2C和易buy的C2C開始迹恐,到阿里生態(tài)下的蘑菇街等等,走...
    唐靖淳閱讀 233評(píng)論 0 1
  • 8.13 星期日 小雨(121) 快樂的周末快樂出發(fā)卧斟,今天兒子和同學(xué)約好單飛到比如世界游玩殴边。無(wú)論做什么事...
    王界程閱讀 188評(píng)論 0 0