Documente DeviceScanActivity
This commit is contained in:
parent
7c281c8fc6
commit
9bdd404fc6
3 changed files with 106 additions and 18 deletions
|
@ -37,7 +37,7 @@
|
||||||
<ConfirmationsSetting value="0" id="Add" />
|
<ConfirmationsSetting value="0" id="Add" />
|
||||||
<ConfirmationsSetting value="0" id="Remove" />
|
<ConfirmationsSetting value="0" id="Remove" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_7" default="true" assert-keyword="true" jdk-15="true" project-jdk-name="1.8 (1)" project-jdk-type="JavaSDK">
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_1_8" default="true" assert-keyword="true" jdk-15="true" project-jdk-name="1.8 (1)" project-jdk-type="JavaSDK">
|
||||||
<output url="file://$PROJECT_DIR$/build/classes" />
|
<output url="file://$PROJECT_DIR$/build/classes" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectType">
|
<component name="ProjectType">
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
<?xml version="1.0" encoding="utf-8"?><!--
|
<?xml version="1.0" encoding="utf-8"?><!--
|
||||||
Copyright 2013 The Android Open Source Project
|
Copyright 2013 The Android Open Source Project
|
||||||
Copyright 2017 Louis-Guillaume Dubois
|
Copyright 2017 CentraleSupélec
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
you may not use this file except in compliance with the License.
|
you may not use this file except in compliance with the License.
|
||||||
|
@ -21,28 +21,35 @@
|
||||||
|
|
||||||
<!-- Min/target SDK versions (<uses-sdk>) managed by build.gradle -->
|
<!-- Min/target SDK versions (<uses-sdk>) managed by build.gradle -->
|
||||||
|
|
||||||
|
|
||||||
<!--
|
<!--
|
||||||
Declare this required feature if you want to make the app available to BLE-capable
|
Declare this required feature if you want to make the app available to BLE-capable
|
||||||
devices only. If you want to make your app available to devices that don't support BLE,
|
devices only. If you want to make your app available to devices that don't support BLE,
|
||||||
you should omit this in the manifest. Instead, determine BLE capability by using
|
you should omit this in the manifest. Instead, determine BLE capability by using
|
||||||
PackageManager.hasSystemFeature(FEATURE_BLUETOOTH_LE)
|
PackageManager.hasSystemFeature(FEATURE_BLUETOOTH_LE)
|
||||||
-->
|
-->
|
||||||
|
<!-- Bluetooth LE obligatoire -->
|
||||||
<uses-feature
|
<uses-feature
|
||||||
android:name="android.hardware.bluetooth_le"
|
android:name="android.hardware.bluetooth_le"
|
||||||
android:required="true" />
|
android:required="true" />
|
||||||
|
|
||||||
|
<!-- Service "location" obligatoire pour pouvoir démarrer un scan BLE -->
|
||||||
<uses-feature
|
<uses-feature
|
||||||
android:name=" android.hardware.location"
|
android:name="android.hardware.location"
|
||||||
android:required="true" />
|
android:required="true" />
|
||||||
|
|
||||||
|
<!-- permissions nécessaires pour scanner les appareils BLE et se connecter à eux. -->
|
||||||
<uses-permission android:name="android.permission.BLUETOOTH" />
|
<uses-permission android:name="android.permission.BLUETOOTH" />
|
||||||
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
|
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />
|
||||||
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
||||||
|
|
||||||
|
<!-- Structure de l’application -->
|
||||||
<application
|
<application
|
||||||
android:icon="@drawable/ic_launcher"
|
android:icon="@drawable/ic_launcher"
|
||||||
android:label="@string/app_name"
|
android:label="@string/app_name"
|
||||||
android:theme="@android:style/Theme.Holo.Light">
|
android:theme="@android:style/Theme.Holo.Light">
|
||||||
|
|
||||||
|
<!-- Activité principale : qui s’affiche lors d’un clic sur l’icône
|
||||||
|
de lancement. -->
|
||||||
<activity
|
<activity
|
||||||
android:name=".DeviceScanActivity"
|
android:name=".DeviceScanActivity"
|
||||||
android:label="@string/app_name">
|
android:label="@string/app_name">
|
||||||
|
@ -52,13 +59,19 @@
|
||||||
<category android:name="android.intent.category.LAUNCHER" />
|
<category android:name="android.intent.category.LAUNCHER" />
|
||||||
</intent-filter>
|
</intent-filter>
|
||||||
</activity>
|
</activity>
|
||||||
<activity android:name=".DeviceControlActivity" />
|
|
||||||
|
|
||||||
|
<!-- Activité secondaire, appelé lors du clic sur le nom d’un appareil scanné. -->
|
||||||
|
<activity android:name=".SimpleDetailActivity"></activity>
|
||||||
|
|
||||||
|
<!-- Service utilisé par l’activité secondaire. -->
|
||||||
<service
|
<service
|
||||||
android:name=".BluetoothLeService"
|
android:name=".BluetoothLeService"
|
||||||
android:enabled="true" />
|
android:enabled="true" />
|
||||||
|
|
||||||
<activity android:name=".SimpleDetailActivity"></activity>
|
<!-- Activité d’origine de l’application de démonstration, plus utilisé.
|
||||||
|
<activity android:name=".DeviceControlActivity" />
|
||||||
|
-->
|
||||||
|
|
||||||
</application>
|
</application>
|
||||||
|
|
||||||
</manifest>
|
</manifest>
|
|
@ -45,20 +45,33 @@ import java.util.ArrayList;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Activity for scanning and displaying available Bluetooth LE devices.
|
* Activité qui scanne et affiche les appareils Bluetooth Low Energy
|
||||||
|
* offrant un service BLE particulier.
|
||||||
|
*
|
||||||
|
* Le scan ne cherche que les appareils qui proposent notre service privé
|
||||||
|
* de visualisation de la valeur du potentiomètre de la carte Curiosity
|
||||||
|
* et d’écriture d’une valeur.
|
||||||
*/
|
*/
|
||||||
public class DeviceScanActivity extends ListActivity {
|
public class DeviceScanActivity extends ListActivity {
|
||||||
private LeDeviceListAdapter mLeDeviceListAdapter;
|
|
||||||
private BluetoothAdapter mBluetoothAdapter;
|
|
||||||
private boolean mScanning;
|
|
||||||
private Handler mHandler;
|
|
||||||
|
|
||||||
|
/* Attributs. */
|
||||||
|
private LeDeviceListAdapter mLeDeviceListAdapter; // adapte la liste des appareils scanné pour les afficher sur l’interface
|
||||||
|
private BluetoothAdapter mBluetoothAdapter; // représente l’interface BLE, permet le scan
|
||||||
|
private boolean mScanning; // si l’activité est en train de scanner
|
||||||
|
private Handler mHandler; // gestionnaire de tâches
|
||||||
|
|
||||||
|
/* Constantes : valeurs de retour pour onActivityResult */
|
||||||
private static final int REQUEST_ENABLE_COARSE_LOCATION = 2;
|
private static final int REQUEST_ENABLE_COARSE_LOCATION = 2;
|
||||||
private static final int REQUEST_ENABLE_BT = 1;
|
private static final int REQUEST_ENABLE_BT = 1;
|
||||||
|
|
||||||
// Stops scanning after 10 seconds.
|
/* Constante : durée maximale du scan. */
|
||||||
private static final long SCAN_PERIOD = 10000;
|
private static final long SCAN_PERIOD = 10000;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode appelé lors de la création de l’activité, ou lors d’un changement
|
||||||
|
* de l’orientation de l’écran (portrait/paysage.)
|
||||||
|
* @param savedInstanceState
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public void onCreate(Bundle savedInstanceState) {
|
public void onCreate(Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
|
@ -85,7 +98,7 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// demande des permissions au cas où
|
// Demande de la permission d’accès à la localisation, pour permettre le scan BLE.
|
||||||
if (ContextCompat.checkSelfPermission(this,
|
if (ContextCompat.checkSelfPermission(this,
|
||||||
Manifest.permission.ACCESS_COARSE_LOCATION)
|
Manifest.permission.ACCESS_COARSE_LOCATION)
|
||||||
!= PackageManager.PERMISSION_GRANTED) {
|
!= PackageManager.PERMISSION_GRANTED) {
|
||||||
|
@ -113,14 +126,24 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode de construction du menu (les boutons) de la barre supérieure.
|
||||||
|
* @param menu
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean onCreateOptionsMenu(Menu menu) {
|
public boolean onCreateOptionsMenu(Menu menu) {
|
||||||
|
// Affiche les boutons définis dans res/menu/main.xml
|
||||||
getMenuInflater().inflate(R.menu.main, menu);
|
getMenuInflater().inflate(R.menu.main, menu);
|
||||||
|
|
||||||
|
// Sans scan en cours, n’affiche que le bouton "scan"
|
||||||
if (!mScanning) {
|
if (!mScanning) {
|
||||||
menu.findItem(R.id.menu_stop).setVisible(false);
|
menu.findItem(R.id.menu_stop).setVisible(false);
|
||||||
menu.findItem(R.id.menu_scan).setVisible(true);
|
menu.findItem(R.id.menu_scan).setVisible(true);
|
||||||
menu.findItem(R.id.menu_refresh).setActionView(null);
|
menu.findItem(R.id.menu_refresh).setActionView(null);
|
||||||
} else {
|
} else {
|
||||||
|
// Si un scan est en cours, affiche un bouton de progression et
|
||||||
|
// le bouton "stop", mais pas le bouton "scan".
|
||||||
menu.findItem(R.id.menu_stop).setVisible(true);
|
menu.findItem(R.id.menu_stop).setVisible(true);
|
||||||
menu.findItem(R.id.menu_scan).setVisible(false);
|
menu.findItem(R.id.menu_scan).setVisible(false);
|
||||||
menu.findItem(R.id.menu_refresh).setActionView(
|
menu.findItem(R.id.menu_refresh).setActionView(
|
||||||
|
@ -129,25 +152,36 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode appelée lors de la pression sur un bouton de
|
||||||
|
* la barre supérieure.
|
||||||
|
* @param item
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean onOptionsItemSelected(MenuItem item) {
|
public boolean onOptionsItemSelected(MenuItem item) {
|
||||||
switch (item.getItemId()) {
|
switch (item.getItemId()) {
|
||||||
|
// apppui sur "scan"
|
||||||
case R.id.menu_scan:
|
case R.id.menu_scan:
|
||||||
mLeDeviceListAdapter.clear();
|
mLeDeviceListAdapter.clear(); // vider la liste des appareils affichés
|
||||||
scanLeDevice(true);
|
scanLeDevice(true); // démarrage du scan
|
||||||
break;
|
break;
|
||||||
|
// appui sur "stop"
|
||||||
case R.id.menu_stop:
|
case R.id.menu_stop:
|
||||||
scanLeDevice(false);
|
scanLeDevice(false); // arrêt du scan
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Appelée après onCreate() ou lors du retour de l’activité au premier plan.
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
protected void onResume() {
|
protected void onResume() {
|
||||||
super.onResume();
|
super.onResume();
|
||||||
|
|
||||||
// Vérifie si le service de localisation est actif sur l’appareil
|
// Vérifie si le service de localisation est actif sur l’appareil.
|
||||||
boolean location_enabled;
|
boolean location_enabled;
|
||||||
try {
|
try {
|
||||||
location_enabled = Settings.Secure.LOCATION_MODE_OFF !=
|
location_enabled = Settings.Secure.LOCATION_MODE_OFF !=
|
||||||
|
@ -181,16 +215,27 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
scanLeDevice(true);
|
scanLeDevice(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode appelé lors du retour depuis une activité lancée avec startActivityForResult.
|
||||||
|
* Traitement de la valeur reçue (résultat) envoyée par cette activité.
|
||||||
|
* @param requestCode
|
||||||
|
* @param resultCode
|
||||||
|
* @param data
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
|
||||||
// User chose not to enable Bluetooth.
|
// User chose not to enable Bluetooth.
|
||||||
if (requestCode == REQUEST_ENABLE_BT && resultCode == Activity.RESULT_CANCELED) {
|
if (requestCode == REQUEST_ENABLE_BT && resultCode == Activity.RESULT_CANCELED) {
|
||||||
|
// Application inutile sans BLE => arrêt.
|
||||||
finish();
|
finish();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
super.onActivityResult(requestCode, resultCode, data);
|
super.onActivityResult(requestCode, resultCode, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode appelée lors du passage de l’activité à l’arrière-plan.
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
protected void onPause() {
|
protected void onPause() {
|
||||||
super.onPause();
|
super.onPause();
|
||||||
|
@ -198,10 +243,23 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
mLeDeviceListAdapter.clear();
|
mLeDeviceListAdapter.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode appelée lorsque l’utilisateur clique sur un élément de la liste des appareils
|
||||||
|
* détectés par le scan.
|
||||||
|
* @param l
|
||||||
|
* @param v
|
||||||
|
* @param position
|
||||||
|
* @param id
|
||||||
|
*/
|
||||||
@Override
|
@Override
|
||||||
protected void onListItemClick(ListView l, View v, int position, long id) {
|
protected void onListItemClick(ListView l, View v, int position, long id) {
|
||||||
|
// Récupération de l’objet représentant l’appareil BLE.
|
||||||
final BluetoothDevice device = mLeDeviceListAdapter.getDevice(position);
|
final BluetoothDevice device = mLeDeviceListAdapter.getDevice(position);
|
||||||
|
|
||||||
if (device == null) return;
|
if (device == null) return;
|
||||||
|
|
||||||
|
// Lancement de l’activité "SimpleDetailActivity" pour se connecter à l’appareil
|
||||||
|
// et utiliser ses services BLE.
|
||||||
final Intent intent = new Intent(this, SimpleDetailActivity.class);
|
final Intent intent = new Intent(this, SimpleDetailActivity.class);
|
||||||
intent.putExtra(SimpleDetailActivity.EXTRAS_DEVICE_NAME, device.getName());
|
intent.putExtra(SimpleDetailActivity.EXTRAS_DEVICE_NAME, device.getName());
|
||||||
intent.putExtra(SimpleDetailActivity.EXTRAS_DEVICE_ADDRESS, device.getAddress());
|
intent.putExtra(SimpleDetailActivity.EXTRAS_DEVICE_ADDRESS, device.getAddress());
|
||||||
|
@ -212,8 +270,16 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
startActivity(intent);
|
startActivity(intent);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Méthode qui lance ou arrête le scan des appareils BLE à notre portée qui proposent notre service
|
||||||
|
* privé.
|
||||||
|
* @param enable
|
||||||
|
*/
|
||||||
private void scanLeDevice(final boolean enable) {
|
private void scanLeDevice(final boolean enable) {
|
||||||
|
// Liste (de taille 1) qui contient l’UUID de notre service privé.
|
||||||
UUID [] uuids = {GattConstants.PRIVATE_SERVICE_UUID};
|
UUID [] uuids = {GattConstants.PRIVATE_SERVICE_UUID};
|
||||||
|
|
||||||
|
// Démarrage du scan.
|
||||||
if (enable) {
|
if (enable) {
|
||||||
// Stops scanning after a pre-defined scan period.
|
// Stops scanning after a pre-defined scan period.
|
||||||
mHandler.postDelayed(new Runnable() {
|
mHandler.postDelayed(new Runnable() {
|
||||||
|
@ -228,6 +294,7 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
mScanning = true;
|
mScanning = true;
|
||||||
mBluetoothAdapter.startLeScan(uuids, mLeScanCallback);
|
mBluetoothAdapter.startLeScan(uuids, mLeScanCallback);
|
||||||
} else {
|
} else {
|
||||||
|
// Arrêt du scan.
|
||||||
mScanning = false;
|
mScanning = false;
|
||||||
mBluetoothAdapter.stopLeScan(mLeScanCallback);
|
mBluetoothAdapter.stopLeScan(mLeScanCallback);
|
||||||
}
|
}
|
||||||
|
@ -235,6 +302,8 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Adapter for holding devices found through scanning.
|
// Adapter for holding devices found through scanning.
|
||||||
|
// Contient les appareils détectés lors du scan et permet de les afficher sur
|
||||||
|
// l’interface.
|
||||||
private class LeDeviceListAdapter extends BaseAdapter {
|
private class LeDeviceListAdapter extends BaseAdapter {
|
||||||
private ArrayList<BluetoothDevice> mLeDevices;
|
private ArrayList<BluetoothDevice> mLeDevices;
|
||||||
private LayoutInflater mInflator;
|
private LayoutInflater mInflator;
|
||||||
|
@ -300,7 +369,10 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Device scan callback.
|
/**
|
||||||
|
* Classe définissant la fonction appelée lorsqu’un nouvel appareil est détecté
|
||||||
|
* lors d’un scan BLE.
|
||||||
|
*/
|
||||||
private BluetoothAdapter.LeScanCallback mLeScanCallback =
|
private BluetoothAdapter.LeScanCallback mLeScanCallback =
|
||||||
new BluetoothAdapter.LeScanCallback() {
|
new BluetoothAdapter.LeScanCallback() {
|
||||||
|
|
||||||
|
@ -316,6 +388,9 @@ public class DeviceScanActivity extends ListActivity {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Classe définissant les données à afficher sur chaque élément de la liste de l’interface.
|
||||||
|
*/
|
||||||
static class ViewHolder {
|
static class ViewHolder {
|
||||||
TextView deviceName;
|
TextView deviceName;
|
||||||
TextView deviceAddress;
|
TextView deviceAddress;
|
||||||
|
|
Loading…
Reference in a new issue