web-dev-qa-db-ja.com

WiFi接続のSamsung「自動ネットワークスイッチ」を無効にする

このアプリでは、直接通信のために独自のワイヤレスアクセスポイント(インターネット接続なし)をブロードキャストするデバイスへのWiFi接続の開始を処理します。

すべてのテストデバイスで非常にうまく機能します。ただし、特定のSamsungデバイス(Galaxy S4、Galaxy Note 3)では、Samsungが追加した「不安定な」ネットワークを探す「自動ネットワークスイッチ」と呼ばれる設定がWi-Fi設定の下にあるというユーザーからの報告を受け取っています。自動的に切断され、モバイルデータに戻ります。残念ながら、デバイスにはインターネット接続がないため、Samsungは不安定なネットワークとして報告し、すぐに切断します。

これらのデバイスのいずれもテストに使用できないため、他の誰かがこの問題を認識しているか、プログラムでこの設定を無効にするか、回避する方法を知っているのか興味がありますか?

接続に使用するコードは次のとおりです。

/**
 * Attempt to connect to an open wifi network with the given SSID
 * @param ssid the SSID of the unsecured wireless network to connect to
 */
public static void connectToOpenNetwork (String ssid) {
    WifiManager mgr = getManager();
    WifiConfiguration configuration = getOpenWifiConfiguration(ssid);
    mgr.addNetwork(configuration);
    mgr.saveConfiguration();

    int res = findOpenNetworkId(ssid);
    if (res != INVALID_NETWORK_ID) {
        mgr.enableNetwork(res, true);
        mgr.reconnect();
    } else {
        Log.e(TAG, "Received request to connect to network " + ssid + " but the network was not found in the configurations.");
    }
}

/**
 * Get a WifiConfiguration object configured for an unsecured wireless network with the
 * given SSID.
 * @param ssid the SSID of the network to configure
 * @return a WifiConfiguration object that can be passed to
 * {@link WifiManager#addNetwork(Android.net.wifi.WifiConfiguration)}
 */
private static WifiConfiguration getOpenWifiConfiguration (String ssid) {
    WifiConfiguration config = new WifiConfiguration();

    config.SSID = "\"" + ssid + "\"";
    config.allowedKeyManagement.set(WifiConfiguration.KeyMgmt.NONE);

    return config;
}
13
kcoppock

編集:だから、興味のある人のためのさらなる研究、それは4.3に基づいてサムスンTouchwizバージョンで追加された機能のようです。内部的には、設定の名前は「wifi_watchdog_connectivity_check」です。設定が有効になっているかどうかをsureで確認できるかどうかを確認するには、以下のコードを使用しますが、それ以外の場合はオンになっていると仮定する必要があります。

だから、私が発見したのは、あなたが接続を試み、OSがネットワークから離れると、Wi-Fi設定は「無効」状態にあるということです。そのため、問題が発生した後、WifiManagerから構成の状態をチェックすることで、問題が発生したことを確実に検出できます。

WifiManager m = (WifiManger) getSystemService(Context.WIFI_SERVICE);
List<WifiConfiguration> networks = m.getConfiguredNetworks();
String mySsid = "My Network";
mySsid = "\"" + mySsid + "\"";

boolean isDisabled = false;
for (WifiConfiguration config : networks) {
    if (mySsid.equals(config.SSID)) {
        if (config.status = WifiConfiguration.Status.DISABLED) {
            isDisabled = true;
            break;
        }
    }
}

//If isDisabled is true, the network was disabled by the OS

その後、システム設定アプリケーションから設定の名前の解決を試みることができます。

/** Gets the resources of another installed application */
private static Resources getExternalResources(Context ctx, String namespace) {
    PackageManager pm = ctx.getPackageManager();
    try {
        return (pm == null) ? null : pm.getResourcesForApplication(namespace);
    } catch (PackageManager.NameNotFoundException ex) {
        return null;
    }
}

/** Gets a resource ID from another installed application */
private static int getExternalIdentifier(Context ctx, String namespace, 
        String key, String type) {
    Resources res = getExternalResources(ctx, namespace);
    return (res == null) ? 0 : res.getIdentifier(key, type, namespace);
}

/** Gets a String resource from another installed application */
public static String getExternalString(Context ctx, String namespace, 
        String key, String defVal) {
    int resId = getExternalIdentifier(ctx, namespace, key, "string");
    if (resId != 0) {
        Resources res = getExternalResources(ctx, namespace);
        return res.getString(resId);
    } else {
        return defVal;
    }
}

次に、それを使用して文字列を取得します。

String autoNetworkSwitch = getExternalString(this, "com.Android.settings",
        "wifi_watchdog_connectivity_check", "Unknown");

文字列が存在する場合、現在のユーザーの言語のローカライズされた文字列を返します。


この結果に興味のある人にとっては、このオプションは実際には在庫Android設定ですが、これらのSamsungデバイスではより積極的であるようです。設定は隠された設定ですAndroid.provider.Settings.Java

/**
 * Setting to turn off poor network avoidance on Wi-Fi. Feature is enabled by default and
 * the setting needs to be set to 0 to disable it.
 * @hide
 */
public static final String WIFI_WATCHDOG_POOR_NETWORK_TEST_ENABLED =
       "wifi_watchdog_poor_network_test_enabled";

Settings$Secure にとって API == 15 || API == 16、またはSettings$Global にとって API >= 17。これは、サードパーティのアプリケーションで有効または無効にできる設定ではありません。ただし、canが検出され、警告されます。私の解決策はこれです:

import static Android.os.Build.VERSION.SDK_INT;
import static Android.os.Build.VERSION_CODES.ICE_CREAM_SANDWICH_MR1;
import static Android.os.Build.VERSION_CODES.JELLY_BEAN_MR1;

/**
 * Checks whether the "Avoid poor networks" setting (named "Auto network switch" on 
 * some Samsung devices) is enabled, which can in some instances interfere with Wi-Fi.
 *
 * @return true if the "Avoid poor networks" or "Auto network switch" setting is enabled
 */
public static boolean isPoorNetworkAvoidanceEnabled (Context ctx) {
    final int SETTING_UNKNOWN = -1;
    final int SETTING_ENABLED = 1;
    final String AVOID_POOR = "wifi_watchdog_poor_network_test_enabled";
    final String WATCHDOG_CLASS = "Android.net.wifi.WifiWatchdogStateMachine";
    final String DEFAULT_ENABLED = "DEFAULT_POOR_NETWORK_AVOIDANCE_ENABLED";
    final ContentResolver cr = ctx.getContentResolver();

    int result;

    if (SDK_INT >= JELLY_BEAN_MR1) {
        //Setting was moved from Secure to Global as of JB MR1
        result = Settings.Global.getInt(cr, AVOID_POOR, SETTING_UNKNOWN);
    } else if (SDK_INT >= ICE_CREAM_SANDWICH_MR1) {
        result = Settings.Secure.getInt(cr, AVOID_POOR, SETTING_UNKNOWN);
    } else {
        //Poor network avoidance not introduced until ICS MR1
        //See Android.provider.Settings.Java
        return false;
    }

    //Exit here if the setting value is known
    if (result != SETTING_UNKNOWN) {
        return (result == SETTING_ENABLED);
    }

    //Setting does not exist in database, so it has never been changed.
    //It will be initialized to the default value.
    if (SDK_INT >= JELLY_BEAN_MR1) {
        //As of JB MR1, a constant was added to WifiWatchdogStateMachine to determine 
        //the default behavior of the Avoid Poor Networks setting.
        try {
            //In the case of any failures here, take the safe route and assume the 
            //setting is disabled to avoid disrupting the user with false information
            Class wifiWatchdog = Class.forName(WATCHDOG_CLASS);
            Field defValue = wifiWatchdog.getField(DEFAULT_ENABLED);
            if (!defValue.isAccessible()) defValue.setAccessible(true);
            return defValue.getBoolean(null);
        } catch (IllegalAccessException ex) {
            return false;
        } catch (NoSuchFieldException ex) {
            return false;
        } catch (ClassNotFoundException ex) {
            return false;
        } catch (IllegalArgumentException ex) {
            return false;
        }
    } else {
        //Prior to JB MR1, the default for the Avoid Poor Networks setting was
        //to enable it unless explicitly disabled
        return true;
    }
}

適切に測定するために、Intentを使用してユーザーを高度なWi-Fi設定に誘導できます。

/**
 *  Ensure that an Activity is available to receive the given Intent
 */
public static boolean activityExists (Context ctx, Intent intent) {
    final PackageManager mgr = ctx.getPackageManager();
    final ResolveInfo info = mgr.resolveActivity(i, PackageManager.MATCH_DEFAULT_ONLY);
    return (info != null);
}

public static void showAdvancedWifiIfAvailable (Context ctx) {
    final Intent i = new Intent(Settings.ACTION_WIFI_IP_SETTINGS);
    if (activityExists(ctx, i)) {
        ctx.startActivity(i);
    }
}

興味深いトリビア:このIntentは、[設定]> [Wi-Fi]> [詳細]と同じActivityを表示しますが、異なるタイトルで表示されます(IP設定とAdvanced Wi-Fi)。

21
kcoppock