ACTION_PICK_WIFI_NETWORK on a dialog with avaliable networks

前端 未结 2 1317
故里飘歌
故里飘歌 2021-01-14 15:41

I\'m trying to create a Dialog which shows something like ACTION_PICK_WIFI_NETWORK but instead of open Android Settings / WiFi open it

相关标签:
2条回答
  • 2021-01-14 16:26

    From the ui perspective you need a Custom adapter:

    private class WifiAdapter extends ArrayAdapter<ScanResult> {
    
        public WifiAdapter(Context context, int resource, List<ScanResult> objects) {
            super(context, resource, objects);
        }
    
        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            if (convertView == null) {
                convertView = getLayoutInflater().inflate(R.layout.wifi_item, parent, false);
            }
            ScanResult result = getItem(position);
            ((TextView) convertView.findViewById(R.id.wifi_name)).setText(formatSSDI(result));
            ((ImageView) convertView.findViewById(R.id.wifi_img)).setImageLevel(getNormalizedLevel(result));
            return convertView;
        }
    
        private int getNormalizedLevel(ScanResult r) {
            int level = WifiManager.calculateSignalLevel(r.level,
                    5);
            Log.e(getClass().getSimpleName(), "level " + level);
            return level;
        }
    
        private String formatSSDI(ScanResult r) {
            if (r == null || r.SSID == null || "".equalsIgnoreCase(r.SSID.trim())) {
                return "no data";
            }
            return r.SSID.replace("\"", "");
        }
    

    I slightly changed your showWifiListDialog:

    private void showWifiListDialog(List<ScanResult> results) {
        Collections.sort(results, new Comparator<ScanResult>() {
            @Override
            public int compare(ScanResult lhs, ScanResult rhs) {
                return rhs.level > lhs.level ? 1 : rhs.level < lhs.level ? -1 : 0;
            }
        });
        AlertDialog.Builder builderSingle = new AlertDialog.Builder(
                this);
        final WifiAdapter arrayAdapter = new WifiAdapter(
                this,
                android.R.layout.select_dialog_item, results);
    
        builderSingle.setNegativeButton(getString(android.R.string.cancel),
                new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                    }
                });
        builderSingle.setAdapter(arrayAdapter,
                new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        String strName = arrayAdapter.getItem(which).SSID;
                        Toast.makeText(getApplicationContext(), "Selected " + strName, Toast.LENGTH_SHORT).show();
                    }
                });
        AlertDialog dialog = builderSingle.create();
        dialog.show();
    }
    

    the Wifi Item is

    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="horizontal">
    
        <TextView
            android:textSize="16sp"
            android:padding="5dp"
            android:layout_gravity="center_vertical"
            android:id="@+id/wifi_name"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1" />
    
        <ImageView
            android:id="@+id/wifi_img"
            android:layout_gravity="center_vertical"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:src="@drawable/wifi_level" />
    </LinearLayout>
    

    and the drawable wifi_level is

    <?xml version="1.0" encoding="utf-8"?>
    <level-list xmlns:android="http://schemas.android.com/apk/res/android">
        <item
            android:drawable="@drawable/ic_signal_wifi_0_bar_black_24dp"
            android:maxLevel="0" />
        <item
            android:drawable="@drawable/ic_signal_wifi_1_bar_black_24dp"
            android:maxLevel="1" />
        <item
            android:drawable="@drawable/ic_signal_wifi_2_bar_black_24dp"
            android:maxLevel="2" />
        <item
            android:drawable="@drawable/ic_signal_wifi_3_bar_black_24dp"
            android:maxLevel="3" />
    
        <item
            android:drawable="@drawable/ic_signal_wifi_4_bar_black_24dp"
            android:maxLevel="4" />
    </level-list>
    

    I took the five png from here

    For the Connection, the answer is yes it should be possible. Accordingly to the documentation at least. You can instantiate an object of WifiConfiguration, and feed it with info of the network you want to connect to (SSID and password). It is not a straightforward thing to do. If you have to take in consideration the different kind of keys encryption, (WPA, WEP, free wifi). Onceyou filled up the object, you have to call

    mWifiManager.disconect();
    int resId = mWifiManager.addNetwork(config);
    mWifiManager.enableNetwork(resId, true);
    

    Edit:

    If you want to show the wifi-signal-strength icon with and without padlock, you could use custom attribute

    <?xml version="1.0" encoding="utf-8"?>
    <resources>
        <declare-styleable name="wifi">
            <attr name="state_locked" format="boolean" />
        </declare-styleable>
    </resources>
    

    and update its state in a subclass of ImageView:

     public class WifiImageView extends ImageView {
    
    private static final int[] STATE_LOCKED = {R.attr.state_locked};
    private boolean mWifiLocked;
    
    public WifiImageView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }
    
    @Override
    public int[] onCreateDrawableState(int extraSpace) {
        final int[] drawableState = super.onCreateDrawableState(extraSpace + 1);
        if (mWifiLocked) {
            mergeDrawableStates(drawableState, STATE_LOCKED);
        }
        return drawableState;
    }
    
    public void setStateLocked(boolean locked) {
        mWifiLocked = locked;
        refreshDrawableState();
    }
    

    }

    Now assuming that the android:src of your WifeImageView is a selector

    <?xml version="1.0" encoding="utf-8"?>
    <selector xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:custom="http://schemas.android.com/apk/res-auto">
        <item custom:state_locked="false" android:drawable="@drawable/wifi_level" />
        <item custom:state_locked="true" android:drawable="@drawable/wifi_level_lock" />
    </selector>
    

    In your Adapter you can easily switch between the two level-list, adding the following two lines of code

     boolean protectedWifi = result.capabilities.contains ("WEP") || result.capabilities.contains("WPA");
     ((WifiImageView) convertView.findViewById(R.id.wifi_img)).setStateLocked(protectedWifi);
    

    protectedWifi is evaluated true if result.capabilities contains WEP or WPA, and setStateLocked(protectedWifi); will switch between the two level-lists accordingly to its value. Of course, in the wifi_item.xml, you have two change from ImageView, to the custom WifiImageView.

    0 讨论(0)
  • 2021-01-14 16:26

    I found the library WifiUtils to be extremely useful. To connect to a selected wifi network, create connectToWifi(ssid, pass)

    private fun connectToWifi(ssid: String, password: String) {
        WifiUtils.withContext(applicationContext)
          .connectWith(ssid, password)
          .onConnectionResult(::connectWifiResultListener)
          .start()
    }
    

    Then, replace Toast.makeText(getApplicationContext(), "Selected " + strName, Toast.LENGTH_SHORT).show(); in showWifiListDialog setAdapter onClick with connectToWifi(strName, password)

    The password can be empty "" if the wifi network doesn't require authentication

    The result will return to connectWifiResultListener

     private fun connectWifiResultListener(isSuccess: Boolean) {
        if (isSuccess)
          // do something
        else}
          // show error
    }
    

    In addition, I adapted scanWifi from the WifiUtils library with @Skizo-ozᴉʞS wifi scanning solution and it worked like a charm. So, instead of the startWifiScans method and the wifiReceiver which calls showWifiListDialog(results) I used

    WifiUtils.withContext(applicationContext).scanWifi(::getScanResults).start()
    

    And call showWifiListDialog in getScanResults

    private fun getScanResults(results: List<ScanResult>) {
        if (results.isEmpty()) { return }
        showWifiListDialog(results)
      }
    
    0 讨论(0)
提交回复
热议问题