Search code examples
androidandroid-broadcastreceiverandroid-7.0-nougat

SCAN_RESULTS_AVAILABLE_ACTION: BroadcastReceiver#onReceive not called on Android 7.0


I'm developing an app for Android in which I have to scan the available WiFi LANs.

I'm testing the app and it works on devices with Android from 4.4 to 6.x, but on Nougat (in fact I've tested on 2 different devices, a Huawei and a HTC, with Android 7) the onReceive of my BroadcastReceiver is never called.

If I try with WIFI_STATE_CHANGED_ACTION I'm able to receive this action even on Android 7.

I've checked the changes' documentation in Nougat, but I found nothing about this. Why doesn't it work on Nougat? What do I have to add/change?

Here is a stripped down version of my app.

In AndroidManifest.xml I've added:

    <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
    <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
    <uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />

MainActivity.java

public class MainActivity extends AppCompatActivity {
private ListView _listSSID;
private ArrayAdapter<String> _adapter;
private WiFiConnection _wifiConnection = null;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
}

@Override
protected void onStart() {
    super.onStart();
    _wifiConnection = new WiFiConnection(this);
    _adapter = new ArrayAdapter<>(this, android.R.layout.simple_list_item_1,
            android.R.id.text1, _wifiConnection.getBestSSIDs());
    _listSSID = (ListView) findViewById(R.id.listSSID);
    _listSSID.setAdapter(_adapter);
    _listSSID.setOnItemClickListener(new AdapterView.OnItemClickListener() {
        @Override
        public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
            String ssid = _adapter.getItem(position);
            Log.d("OnClick", ssid);
        }
    });
    startScan();
}

@Override
protected void onStop() {
    super.onStop();

    _wifiConnection.stopScan();
    unregisterReceiver(_wifiScanReceiver);
}

void startScan() {
    checkPermission(this);

    registerReceiver(_wifiScanReceiver,
            new IntentFilter(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION));
    Thread t = new Thread(_wifiConnection.getWifiScanner());
    t.start();
}

private final BroadcastReceiver _wifiScanReceiver = new BroadcastReceiver() {
    @Override
    public void onReceive(Context c, Intent intent) {
        if (intent.getAction().equals(WifiManager.SCAN_RESULTS_AVAILABLE_ACTION)) {
            if (_wifiConnection != null && _wifiConnection.isWifiEnabled()) {
                _wifiConnection.updateScanData("No WLANs found");
            }
            _adapter.notifyDataSetChanged();
        }
    }
};

static final int MY_PERMISSIONS_REQUEST = 1042;

private static final String PERMISSIONS_TAG = "PERMISSION";

@TargetApi(Build.VERSION_CODES.M)
public static boolean checkPermission(Activity activity) {
    boolean permission = true;
    if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
        permission = activity.checkSelfPermission(Manifest.permission.ACCESS_WIFI_STATE) == PackageManager.PERMISSION_GRANTED;
        Log.d(PERMISSIONS_TAG, "ACCESS_WIFI_STATE: " + permission);
        permission = activity.checkSelfPermission(Manifest.permission.CHANGE_WIFI_STATE) == PackageManager.PERMISSION_GRANTED;
        Log.d(PERMISSIONS_TAG, "CHANGE_WIFI_STATE: " + permission);
        permission = activity.checkSelfPermission(Manifest.permission.CHANGE_WIFI_MULTICAST_STATE) == PackageManager.PERMISSION_GRANTED;
        Log.d(PERMISSIONS_TAG, "CHANGE_WIFI_MULTICAST_STATE: " + permission);

        List<String> requiringList = new ArrayList<>();
        permission = activity.checkSelfPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_GRANTED;
        Log.d(PERMISSIONS_TAG, "WRITE_EXTERNAL_STORAGE: " + permission);
        if (!permission) {
            requiringList.add(Manifest.permission.WRITE_EXTERNAL_STORAGE);
        }

        permission = activity.checkSelfPermission(Manifest.permission.ACCESS_COARSE_LOCATION) == PackageManager.PERMISSION_GRANTED;
        Log.d(PERMISSIONS_TAG, "ACCESS_COARSE_LOCATION: " + permission);
        if (!permission) {
            requiringList.add(Manifest.permission.ACCESS_COARSE_LOCATION);
        }

        if (requiringList.size() > 0) {
            String[] stringArray = requiringList.toArray(new String[0]);
            activity.requestPermissions(stringArray, MY_PERMISSIONS_REQUEST);
        }
    }
    return permission;
}
}

WiFiConnection.java

class WiFiConnection
{
    private static final int SCAN_INTERVAL = 1200;

    final private List<String> _bestSSIDs = new ArrayList<>();

    private WifiManager _wifiManager;

    private final WiFiScanner _startScan = new WiFiScanner();

    WiFiConnection(Activity activity) {
        _wifiManager = (WifiManager) activity.getApplicationContext().getSystemService(Context.WIFI_SERVICE);
    }

    List<String> getBestSSIDs() {return _bestSSIDs; }
    WiFiScanner getWifiScanner() { return _startScan; }
    void stopScan() { _startScan.stop(); }

    boolean isWifiEnabled() { return _wifiManager.isWifiEnabled(); }

    void updateScanData(final String noLansMsg) {
        if ((_wifiManager != null && _wifiManager.isWifiEnabled())) {
            List<ScanResult> scanResults = _wifiManager.getScanResults();
            Log.d("WiFi", "found nets: " + scanResults.size());

            // my sorting code
        }
    }

    private class WiFiScanner implements Runnable
    {
        private boolean _stop = false;
        public void stop() {_stop = true;}
        @Override
        public void run() {
            while (!_stop) {
                _wifiManager.startScan();
                try {
                    Thread.sleep(SCAN_INTERVAL);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

Solution

  • It seems it is necessary to turn on GPS to receive SCAN_RESULTS_AVAILABLE_ACTION on Android 7.0.