Android 4.3: как подключиться к нескольким устройствам Bluetooth с низким энергопотреблением?
Задача:
Я занимаюсь разработкой приложения для Android, которое должно одновременно подключаться к нескольким (идентичным, которые можно различить по их идентификатору) чип-устройствам BLE для отправки и получения обновлений. Я использовал учебники на официальной веб-странице Google:
http://developer.android.com/guide/topics/connectivity/bluetooth-le.html
Это помогло мне создать Java-класс DeviceScanActivity, который позволяет мне сканировать и перечислять все доступные устройства BLE в непосредственной близости, подобно тому, что в настоящее время делают большинство приложений этого типа. Я также узнал, что до 7 внешних ведомых устройств могут быть подключены к одному и тому же ведущему устройству одновременно. Однако реализация этого сообщения для меня все еще неясна. Полезная ссылка:
Android 4.3: как подключиться к нескольким Bluetooth-устройствам с низким энергопотреблением
Тем не менее, он не дает достаточно подробностей для меня, чтобы понять, как работает такая реализация.
Я изучал эту тему некоторое время и не смог найти ни одного примера реализации. Я знаю, что этот вопрос задавался много раз, но, кажется, нет какого-либо рабочего решения / демоверсии, доступного в Интернете, чтобы сделать вещи более понятными для меня. Я буду очень признателен, если кто-нибудь подскажет мне ресурсное / рабочее решение, которое подробно объяснит шаги, необходимые для изменения моего существующего класса Java для реализации этой функциональности.
DeviceScanActivity.java:
import android.app.Activity;
import android.app.ListActivity;
import android.app.ProgressDialog;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.bluetooth.BluetoothManager;
import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.content.pm.PackageManager;
import android.os.Bundle;
import android.os.Handler;
import android.view.LayoutInflater;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;
import java.util.ArrayList;
/**
* Activity for scanning and displaying available Bluetooth LE devices.
*/
public class DeviceScanActivity extends ListActivity {
private LeDeviceListAdapter mLeDeviceListAdapter;
private BluetoothAdapter mBluetoothAdapter;
private boolean mScanning;
private Handler mHandler;
private static final int REQUEST_ENABLE_BT = 1;
// Stops scanning after 10 seconds.
private static final long SCAN_PERIOD = 10000;
private static final String CONFIG_FILE = "file";
private int _countClick = 0;
private boolean experimenterMode = false;
private String pairedDeviceAddress = "";
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getActionBar().setTitle(R.string.title_devices);
mHandler = new Handler();
// Use this check to determine whether BLE is supported on the device. Then you can
// selectively disable BLE-related features.
if (!getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE)) {
Toast.makeText(this, R.string.ble_not_supported, Toast.LENGTH_SHORT).show();
finish();
}
// Initializes a Bluetooth adapter. For API level 18 and above, get a reference to
// BluetoothAdapter through BluetoothManager.
final BluetoothManager bluetoothManager =
(BluetoothManager) getSystemService(Context.BLUETOOTH_SERVICE);
mBluetoothAdapter = bluetoothManager.getAdapter();
// Checks if Bluetooth is supported on the device.
if (mBluetoothAdapter == null) {
Toast.makeText(this, R.string.error_bluetooth_not_supported, Toast.LENGTH_SHORT).show();
finish();
return;
}
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.main, menu);
if (!mScanning) {
menu.findItem(R.id.menu_stop).setVisible(false);
menu.findItem(R.id.menu_scan).setVisible(true);
menu.findItem(R.id.menu_refresh).setActionView(null);
} else {
menu.findItem(R.id.menu_stop).setVisible(true);
menu.findItem(R.id.menu_scan).setVisible(false);
menu.findItem(R.id.menu_refresh).setActionView(
R.layout.actionbar_indeterminate_progress);
}
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.menu_scan:
mLeDeviceListAdapter.clear();
scanLeDevice(true);
break;
case R.id.menu_stop:
scanLeDevice(false);
break;
case android.R.id.home:
//onBackPressed();
_countClick++;
if(_countClick>8)
{
toggleExperimenterMode();
}
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
protected void onResume() {
super.onResume();
// Ensures Bluetooth is enabled on the device. If Bluetooth is not currently enabled,
// fire an intent to display a dialog asking the user to grant permission to enable it.
if (!mBluetoothAdapter.isEnabled()) {
if (!mBluetoothAdapter.isEnabled()) {
Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT);
}
}
// Restore preferences
SharedPreferences settings = getSharedPreferences(CONFIG_FILE, 0);
experimenterMode = settings.getBoolean("experimenterMode", false);
pairedDeviceAddress = settings.getString("pairedDeviceAddress", "");
// Initializes list view adapter.
mLeDeviceListAdapter = new LeDeviceListAdapter();
setListAdapter(mLeDeviceListAdapter);
scanLeDevice(true);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
// User chose not to enable Bluetooth.
if (requestCode == REQUEST_ENABLE_BT && resultCode == Activity.RESULT_CANCELED) {
finish();
return;
}
super.onActivityResult(requestCode, resultCode, data);
}
@Override
protected void onPause() {
super.onPause();
scanLeDevice(false);
mLeDeviceListAdapter.clear();
}
@Override
protected void onListItemClick(ListView l, View v, int position, long id) {
final BluetoothDevice device = mLeDeviceListAdapter.getDevice(position);
if (device == null) return;
connectToDevice(device);
}
private void connectToDevice(BluetoothDevice device){
final Intent intent = new Intent(this, DeviceControlActivity.class);
intent.putExtra(DeviceControlActivity.EXTRAS_DEVICE_NAME, device.getName());
intent.putExtra(DeviceControlActivity.EXTRAS_DEVICE_ADDRESS, device.getAddress());
if (mScanning) {
mBluetoothAdapter.stopLeScan(mLeScanCallback);
mScanning = false;
}
SharedPreferences settings = getSharedPreferences(CONFIG_FILE, 0);
SharedPreferences.Editor editor = settings.edit();
editor.putString("pairedDeviceAddress", device.getAddress());
editor.commit();
startActivity(intent);
}
//---------------------------------------------------
private void scanLeDevice(final boolean enable) {
ProgressDialog progress;
if (enable) {
// Stops scanning after a pre-defined scan period.
mHandler.postDelayed(new Runnable() {
@Override
public void run() {
mScanning = false;
mBluetoothAdapter.stopLeScan(mLeScanCallback);
invalidateOptionsMenu();
}
}, SCAN_PERIOD);
mScanning = true;
mBluetoothAdapter.startLeScan(mLeScanCallback);
} else {
mScanning = false;
mBluetoothAdapter.stopLeScan(mLeScanCallback);
}
invalidateOptionsMenu();
}
private void toggleExperimenterMode(){
_countClick=0;
experimenterMode = !experimenterMode;
SharedPreferences settings = getSharedPreferences(CONFIG_FILE, 0);
SharedPreferences.Editor editor = settings.edit();
editor.putBoolean("experimenterMode", experimenterMode);
editor.commit();
String t = experimenterMode ? "You are now in Experimenter mode":"You are now in User mode";
Toast.makeText(this, t, Toast.LENGTH_SHORT).show();
}
// Adapter for holding devices found through scanning.
private class LeDeviceListAdapter extends BaseAdapter {
private ArrayList<BluetoothDevice> mLeDevices;
private LayoutInflater mInflator;
public LeDeviceListAdapter() {
super();
mLeDevices = new ArrayList<BluetoothDevice>();
mInflator = DeviceScanActivity.this.getLayoutInflater();
}
public void addDevice(BluetoothDevice device) {
if(!mLeDevices.contains(device)) {
mLeDevices.add(device);
}
if (!experimenterMode && ! pairedDeviceAddress.isEmpty()) {
if (device.getAddress().equals(pairedDeviceAddress)) {
connectToDevice(device);
}
}
}
public BluetoothDevice getDevice(int position) {
return mLeDevices.get(position);
}
public void clear() {
mLeDevices.clear();
}
@Override
public int getCount() {
return mLeDevices.size();
}
@Override
public Object getItem(int i) {
return mLeDevices.get(i);
}
@Override
public long getItemId(int i) {
return i;
}
@Override
public View getView(int i, View view, ViewGroup viewGroup) {
ViewHolder viewHolder;
// General ListView optimization code.
if (view == null) {
view = mInflator.inflate(R.layout.listitem_device, null);
viewHolder = new ViewHolder();
viewHolder.deviceAddress = (TextView) view.findViewById(R.id.device_address);
viewHolder.deviceName = (TextView) view.findViewById(R.id.device_name);
view.setTag(viewHolder);
} else {
viewHolder = (ViewHolder) view.getTag();
}
BluetoothDevice device = mLeDevices.get(i);
final String deviceName = device.getName();
if (deviceName != null && deviceName.length() > 0)
viewHolder.deviceName.setText(deviceName);
else
viewHolder.deviceName.setText(R.string.unknown_device);
viewHolder.deviceAddress.setText(device.getAddress());
return view;
}
}
// Device scan callback.
private BluetoothAdapter.LeScanCallback mLeScanCallback =
new BluetoothAdapter.LeScanCallback() {
@Override
public void onLeScan(final BluetoothDevice device, int rssi, byte[] scanRecord) {
runOnUiThread(new Runnable() {
@Override
public void run() {
mLeDeviceListAdapter.addDevice(device);
mLeDeviceListAdapter.notifyDataSetChanged();
}
});
}
};
static class ViewHolder {
TextView deviceName;
TextView deviceAddress;
}
}
Текущая информация:
- Android 4.3: как подключиться к нескольким Bluetooth-устройствам с низким энергопотреблением
- https://developer.android.com/guide/topics/connectivity/bluetooth-le.html
- Поддерживает ли Android 4.3 несколько подключений устройств BLE?
- Ble несколько соединений
- https://labs.hybris.com/2014/10/06/connecting-to-multiple-ble-devices/
- https://e2e.ti.com/support/wireless_connectivity/f/538/t/225935
Все эти страницы содержат некоторые указания, но они не предоставляют достаточно подробных сведений о том, как можно реализовать несколько соединений.
Целевое представление пользовательского интерфейса:
Идея заключается в том, что мое устройство Android Nexus 7 5.1.1 сможет подключаться максимум к 5 ведомым устройствам BLE и отправлять / получать обновления.
1 ответ
Я сделал это в Android с двумя устройствами. Во-первых, по моему опыту, гораздо лучше разделить ваш код. В моем случае я использовал сервис для процесса сканирования. Посредством широковещательного приемника я отправил найденное устройство Object на основной вид деятельности. В этом упражнении в обычном режиме выполняется только пользовательский интерфейс и мое обслуживание. Когда моя основная деятельность привела к появлению объекта устройства Bluetooth, я передаю этот объект отдельной службе, где выполняется только связь Bluetooth, и с другим приемником широковещательной передачи я отправляю полученные данные обратно в свою основную деятельность для отображения значений. Если вы хотите подключить несколько устройств, вы можете использовать этот шаблон так же, как и с одним устройством. В службе сканера в вашем LeScan Callback вы проверяете найденные устройства на их Macadress. Я сошел с ума по String Array со всеми адресами Mac, к которым я хочу подключиться. Когда я нашел одно, я удалил его из массива, и когда вы нашли все нужные вам устройства, вы можете остановить весь сервис. Кроме того: я получил опыт, что вы не должны связывать себя служением. Лучше запускать и останавливать ваш сервис вручную. Когда я хотел уничтожить мою связанную службу, довольно часто соединение оставалось в живых, и при повторном подключении мой Bluetooth ведет себя странно, или я не могу найти свое устройство сервера Ble, потому что внутренне оно все еще связано.
У меня была такая же проблема для моего приложения, документации по BLE на Android очень мало, но после долгих исследований я создал хорошо работающую библиотеку Bluetooth для соединения между несколькими устройствами: https://github.com/niedev/RTranslator/tree/master/app/src/main/java/nie/translator/rtranslatordevedition/voice_translation/_conversation_mode/communication/communicator
ps В моих тестах библиотеки максимальное количество стабильно подключенных устройств - 4 (даже с переподключением)
Изменить: модераторы продолжают отменять мой ответ, потому что я использую ссылку на свой код вместо того, чтобы вставлять его прямо здесь, но я не могу, потому что код составляет почти 4600 строк кода, разделенных на 10 классов (и количество символов в ответе ограничено)