推投屏盒子有两张无线网卡,一个负责对外上网,一个负责对内提供AP局域网。这样的设计是为了在没有外网的环境下也能够实现投屏功能(避免了很多市面上的投屏盒子的缺点。比如必须连接WiFi,并且不支持8021x/eap的WiFi连接),通过连接轻推投屏盒子的AP,实现某些公司需要的公司内网认证。
好了,接下来进入正题,上网WiFi我们使用Android系统的提供WiFi模块管理——WiFiManager。
WiFiManager:是系统提供给开发者使用的系统服务管理类,WifiManager会调用service和framework层, 驱动层进行函数调用,
然后驱动层会回调至上层, 以广播的形式实现通知。简单来说就是,只需要使用WifiManager进行函数操作完成UI, 监听对应的广播消息, 就可完成功能了。
笔者所用到方法:
WiFiManager.startScan():开始扫描WiFi,扫描结果通过广播通知
WiFiManager.disconnect():断开网络WiFi
WiFiManager.addNetwork(WifiConfiguration config):添加WiFi配置,返回networkId
WiFiManager.enableNetwork(networkId, true):允许与以前配置的网络相关联,并尝试连接WiFi
WiFiManager.getScanResults():获取设备缓存的扫描结果,这里部分手机的缓存会很少,跟实际扫描结果不一致,这个时候需要调用一次扫描更新缓存
第一步:获取WiFiManager
WifiManager wifiManager = (WifiManager) context.getSystemService(Context.WIFI_SERVICE);
第二步:获取扫描结果缓存
ScanResult.SSID 表示WiFi名,这里需要注意有"",如果界面需要显示,那么得把这个去掉,连接的时候这个得保留才行
ScanResult.BSSID mac,表示WiFi的唯一标识,
ScanResult.frequency 表示WiFi的频率
ScanResult.level 表示WiFi的信号强度,这里为负数,越大表示信号越好
List<ScanResult> scanResults = wifiManager.getScanResults();
第三步:连接WiFi
/**
* wif连接
*
* @param scanResult ScanResult
* @param userName String eap连接需要的用户名
* @param pass String 密码
* @return 操作结果
*/
public boolean connectionWifi(ScanResult scanResult,String userName,String pass) {
WifiConfiguration config = getWifiConfig(ScanResult scanResult,String userName,String pass);
networkId = wifiManager.addNetwork(config);
wifiManager.saveConfiguration();
return wifiManager.enableNetwork(networkId, true);
}
工具方法
判断WiFi类型
/**
* 获取wifi安全类型.
*
* @param result ScanResult
* @return 0
*/
public static int getSecurity(ScanResult result) {
if (null != result && null != result.capabilities) {
if (result.capabilities.contains("WEP")) {
return SECURITY_WEP;
} else if (result.capabilities.contains("PSK")) {
return SECURITY_PSK;
} else if (result.capabilities.contains("EAP")) {
return SECURITY_EAP;
}
}
return SECURITY_NONE;
}
获取wif连接配置
/**
* 获取wif连接配置
*
* @param scanResult ScanResult
* @return
*/
public WifiConfiguration getConfig(ScanResult scanResult,String userName,String pass){
WifiConfiguration config = new WifiConfiguration();
switch(getSecurity(scanResult)){
case SECURITY_NONE:
initOpenConfig(config);
break;
case SECURITY_WEP:
initWepConfig(scanResult,config,pass);
break;
case SECURITY_PSK:
initPskConfig(scanResult,config,pass);
break;
case SECURITY_EAP:
initEapConfig(scanResult,config,userName,pass);
break;
}
}
/**
* 初始化公开类型WiFi
*
* @param config WifiConfiguration
*/
private void initOpenConfig(WifiConfiguration config){
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
}
private void initWepConfig(ScanResult scanResult, WifiConfiguration config,String pass) {
config.hiddenSSID = true;
//安全认证协议
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);
//身份验证算法
config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.OPEN);
config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.SHARED);
config.wepTxKeyIndex = 0;
if (isHexWepKey(pass))
config.wepKeys[0] = wifiConnDO.getPassword();//密码
else
config.wepKeys[0] = convertToQuotedString(pass);//密码
}
private void initPskConfig(ScanResult scanResult, WifiConfiguration config,String pass) {
config.hiddenSSID = true;
//公认安全协议
config.allowedProtocols.set(WifiConfiguration.Protocol.RSN);
config.allowedProtocols.set(WifiConfiguration.Protocol.WPA);
//安全认证协议
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.WPA_PSK);
//密码为WPA
config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.CCMP);
config.allowedPairwiseCiphers.set(WifiConfiguration.PairwiseCipher.TKIP);
//公认的的公共组密码
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.CCMP);
config.allowedGroupCiphers.set(WifiConfiguration.GroupCipher.TKIP);
//身份验证算法
config.allowedAuthAlgorithms.set(WifiConfiguration.AuthAlgorithm.OPEN);
config.preSharedKey = convertToQuotedString(pass);//密码
config.status = WifiConfiguration.Status.ENABLED;
}
private void initEapConfig(ScanResult scanResult, WifiConfiguration config,String userName,String pass) {
config.hiddenSSID = false;
//安全认证协议
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.WPA_EAP);
config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.IEEE8021X);
//企业级WiFi的相关信息输入配置
WifiEnterpriseConfig enterpriseConfig = new WifiEnterpriseConfig();
enterpriseConfig.setIdentity(wifiConnDO.getUsername());//用户身份
enterpriseConfig.setPassword(wifiConnDO.getPassword());//用户密码
enterpriseConfig.setAnonymousIdentity("");//匿名身份
enterpriseConfig.setEapMethod(WifiEnterpriseConfig.Eap.PEAP);//EAP类型
enterpriseConfig.setPhase2Method(WifiEnterpriseConfig.Phase2.NONE);
config.enterpriseConfig = enterpriseConfig;
}
static boolean isHexWepKey(@Nullable String wepKey) {
final int passwordLen = wepKey == null ? 0 : wepKey.length();
return passwordLen != 0 && (passwordLen == 10 || passwordLen == 26 || passwordLen == 58) && wepKey.matches("[0-9A-Fa-f]*");
}
static String convertToQuotedString(@NonNull String string) {
if (TextUtils.isEmpty(string))
return "";
final int lastPos = string.length() - 1;
if (lastPos < 0 || (string.charAt(0) == '"' && string.charAt(lastPos) == '"'))
return string;
return "\"" + string + "\"";
}
在我们的实际使用中发现,公司的WiFi有些问题,已经连接过的再次连接的时候会出现连不上的问题,通过对WifiStateMachine的各种日志分析之后发现是我们公司的认证服务有些异常,但这不是我们负责的,所以我们这边采取的方案是,移除配置,然后添加新的配置,问题成功解决。
获取已经系统保存的WiFi配置信息和新的连接方法
protected boolean connectionWifi(WifiConnDO wifiConnDO) {
//获取已保存的WiFi配置
WifiConfiguration config = getSaved(wifiConnDO);
int networkId;
if (config != null) {
networkId = config.networkId;
//移除以保存的配置并保存操作
wifiManager.removeNetwork(networkId);
wifiManager.saveConfiguration();
}
//获取WiFi配置
config = getWifiConfig(wifiConnDO);
// config.BSSID = wifiConnDO.getBssid();
//添加并保存配置
networkId = wifiManager.addNetwork(config);
wifiManager.saveConfiguration();
//使用WiFi
return wifiManager.enableNetwork(networkId, true);;
}
private WifiConfiguration getSaved(ScanResult result) {
List<WifiConfiguration> existingConfigs = wifiManager.getConfiguredNetworks();
if (existingConfigs == null) {
return null;
}
for (WifiConfiguration existingConfig : existingConfigs) {
if (existingConfig.SSID.equals(result.SSID)) {
return existingConfig;
}
}
return null;
}
关于连接WiFi的广播部分这里就不做过多说明了,与其他技术博客的一样即可。
整个过程中我们遇到很多问题,诸如上网芯片支持的频道过少发现不了公司5GWiFi的信道,公司WiFi上网定期会断之后重新连接会切到其他BSSID的同名WiFi导致上网速度慢…
有疑问和指教的地方请私联:[email protected]