在开发应用的过程中,遇到这样一个问题,就是在国产的iphone设备中,第一次安装新的应用时,启动后都会询问用户是否需要打开蜂窝支持,而有一些用户选择了拒绝通过蜂窝访问应用时,会导致再往后使用该应用时可能误以为是服务器出现问题,或者应用出现问题,而忘记了安装应用时已经拒绝通过蜂窝访问,导致无法正常访问服务器。
而为了方便用户使用,我们则需要判断当前应用是否打开了蜂窝支持。
首先,我们需要判断的是当前用户的设备是访问当前的网络的状态是:无网络,2G,3G,4G还是WiFi网络。
而在获取当前网络状态时,我们可以通过key(_statusBar)来获取当前系统的状态栏中的网络状态,而传统的获取方式就是这样:
NSArray *children;
UIApplication *app = [UIApplication sharedApplication];
NSString *state = [[NSString alloc] init];
children = [[[app valueForKeyPath:@"_statusBar"] valueForKeyPath:@"foregroundView"] subviews];
for (id child in children)
{
if ([child isKindOfClass:NSClassFromString(@"UIStatusBarDataNetworkItemView")])
{
//获取到状态栏
switch([[child valueForKeyPath:@"dataNetworkType"] intValue])
{
case 0: state = @"无网络";
//无网模式
break;
case 1: state = @"2G";
break;
case 2: state = @"3G";
break;
case 3: state = @"4G";
break;
case 5: state = @"wifi";
break;
default: break;
}
}
如果用户的设备是iPhone X之前的设备,这个方法是可以直接获取到当前设备的网络状态,但是在iPhone X的设备中,获取网络状态属性却不一样了。
在iPhone X手机 wifi情况下,启动时候闪退,在4G网络时候不闪退。这是项目中使用状态栏中图标判断当前网络的具体状态,而 iPhone X手机状态栏和其他版本手机存在差异,状态栏是多嵌套了一层,所以在读取时候需要注意。可以在代码里面搜索 foregroundView 这个关键字,找到位置进行必要修改。
Error:
[<UIStatusBar_Modern 0x7fb8eee05a00> valueForUndefinedKey:]: this class is not key value coding-compliant for the key foregroundView.
所以此时,我们就需要针对iphone X设备进行相关的网络状态的获取,如下实现:
NSArray *children;
UIApplication *app = [UIApplication sharedApplication];
NSString *state = [[NSString alloc] init];
//iPhone X
if ([[app valueForKeyPath:@"_statusBar"] isKindOfClass:NSClassFromString(@"UIStatusBar_Modern")])
{
children = [[[[app valueForKeyPath:@"_statusBar"] valueForKeyPath:@"_statusBar"] valueForKeyPath:@"foregroundView"] subviews]; for (UIView *view in children)
{
for (id child in view.subviews)
{
//wifi
if ([child isKindOfClass:NSClassFromString(@"_UIStatusBarWifiSignalView")])
{
state = @"wifi";
}
//2G 3G 4G
if ([child isKindOfClass:NSClassFromString(@"_UIStatusBarStringView")])
{
if ([[child valueForKey:@"_originalText"] containsString:@"G"])
{
state = [child valueForKey:@"_originalText"];
}
}
}
}
}
那么结合上面的两个方法,封装成一个方法,就可以变成如下编写:
/**
* 获取网络状态
*
* @return 返回网络状态
*/
+ (NSString *)getNetWorkStates
{
NSArray *children;
UIApplication *app = [UIApplication sharedApplication];
NSString *state = [[NSString alloc] init];
//iPhone X
if ([[app valueForKeyPath:@"_statusBar"] isKindOfClass:NSClassFromString(@"UIStatusBar_Modern")])
{
children = [[[[app valueForKeyPath:@"_statusBar"] valueForKeyPath:@"_statusBar"] valueForKeyPath:@"foregroundView"] subviews]; for (UIView *view in children)
{
for (id child in view.subviews)
{
//wifi
if ([child isKindOfClass:NSClassFromString(@"_UIStatusBarWifiSignalView")])
{
state = @"wifi";
}
//2G 3G 4G
if ([child isKindOfClass:NSClassFromString(@"_UIStatusBarStringView")])
{
if ([[child valueForKey:@"_originalText"] containsString:@"G"])
{
state = [child valueForKey:@"_originalText"];
}
}
}
}
}
else
{
children = [[[app valueForKeyPath:@"_statusBar"] valueForKeyPath:@"foregroundView"] subviews];
for (id child in children)
{
if ([child isKindOfClass:NSClassFromString(@"UIStatusBarDataNetworkItemView")])
{
//获取到状态栏
switch([[child valueForKeyPath:@"dataNetworkType"] intValue])
{
case 0: state = @"无网络";
//无网模式
break;
case 1: state = @"2G";
break;
case 2: state = @"3G";
break;
case 3: state = @"4G";
break;
case 5: state = @"wifi";
break;
default: break;
}
}
}
}
return state;
}
当获取当前设备的网络状态下,那现在就要进行相关当前设备是否拥有蜂窝访问的权限,此时在iOS9.0+的系统中,就新增了可以判断蜂窝权限判断的SDK,所以此时我们的设备如果还是iOS9.0之前的系统的,我们就要限制不能通过该SDK进行判断网络权限,否则程序就会直接崩溃,具体的实现方法如下:
/**
判断网络是否有蜂窝权限
*/
+ (void)judgeNetWork
{
//该检查权限的SDK 仅限于iOS 9.0+的系统
if (SystemVersion >= 9.0)
{
CTCellularData *cellularData = [[CTCellularData alloc]init];
cellularData.cellularDataRestrictionDidUpdateNotifier = ^(CTCellularDataRestrictedState state){
//获取联网状态
switch (state) {
case kCTCellularDataRestricted:
DLog(@"Restricrted / 受限制的");
break;
case kCTCellularDataNotRestricted:
DLog(@"Not Restricted /不受限制的");
break;
case kCTCellularDataRestrictedStateUnknown:
DLog(@"Unknown/不明网路");
break;
default:
break;
};
};
}
}
当我们判断网络的蜂窝权限之后,则需要相关的提示用户是否跳转到手机的设置中打开本应用的网络蜂窝支持的权限,而我们就可以通过这样的一句代码可以直接跳转到当前应用的设置位置:
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:UIApplicationOpenSettingsURLString]];
如果仅仅只是为了提示用户跳转到设置中设置的前提下,我们可以通过弹窗的方式提示用户自行选择跳转到设置中,设置蜂窝权限:
/**
提示弹窗网络权限设置
*/
+ (void)showTipFromNetwork
{
UIAlertAction *alertSure = [UIAlertAction actionWithTitle:@"知道了" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
}];
UIAlertAction *alertCancel = [UIAlertAction actionWithTitle:@"去设置" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
[[UIApplication sharedApplication] openURL:[NSURL URLWithString:UIApplicationOpenSettingsURLString]];
}];
UIAlertController *alerVC = [UIAlertController alertControllerWithTitle:@"'XXXX应用' 已关闭蜂窝移动数据" message:@"您可以在通过在手机的 '设置'-->'蜂窝移动网络'中打开蜂窝支持" preferredStyle:UIAlertControllerStyleAlert];
[alerVC addAction:alertCancel];
[alerVC addAction:alertSure];
[[UIApplication sharedApplication].keyWindow.rootViewController presentViewController:alerVC animated:YES completion:nil];
}
可能这样的提示方式会与系统自带的相关网络提示第一次弹窗时有重叠,或者也不会强制应用去开启蜂窝权限,所以具体情况,具体实现是否需要弹窗提示。
大千世界,求同存异;相遇是缘,相识是份,相知便是“猿粪”(缘分)
From MZou