A bit stuck here, might need your help. I want to read several BLE characteristics at once, some people suggest using PriorityQueue for that. I already know all the uuids, etc. just need a way to read several at once. Could anyone explain how exactly should it look like? Or maybe there is yet another easier solution?
Thanks in advance, here is my code:
public static final BluetoothGattCallback mGattCallback = new BluetoothGattCallback() {
PriorityQueue<BluetoothGattCharacteristic> queue = new PriorityQueue<BluetoothGattCharacteristic>();
// When connection state changes
@Override
public void onConnectionStateChange(BluetoothGatt gatt, int status, int newState) {
if (newState == BluetoothProfile.STATE_CONNECTED) {
Log.v(TAG, "Connected!");
gatt.discoverServices();
}
if (newState == BluetoothProfile.STATE_DISCONNECTED) {
Log.v(TAG, "Disconnected...");
}
}
@Override
public void onServicesDiscovered(BluetoothGatt gatt, int status) {
List<BluetoothGattService> services = gatt.getServices();
BluetoothGattService rightService = null;
for (int i = 0; i < services.size(); i++) {
if (services.get(i).getCharacteristics().size() > 8) {
rightService = services.get(i);
}
}
List<UUID> uuidsList;
UUID TRANSMISSION_POWER = rightService.getCharacteristics().get(4).getUuid();
UUID BROADCASTING_INTERVAL = rightService.getCharacteristics().get(6).getUuid();
UUID BEACON_NAME = rightService.getCharacteristics().get(8).getUuid();
UUID CONNECTION_MODE = rightService.getCharacteristics().get(9).getUuid();
//UUID SOFT_REBOOT = rightService.getCharacteristics().get(10).getUuid();
uuidsList = new ArrayList<UUID>();
uuidsList.add(TRANSMISSION_POWER);
uuidsList.add(BROADCASTING_INTERVAL);
uuidsList.add(BEACON_NAME);
uuidsList.add(CONNECTION_MODE);
//uuidsList.add(SOFT_REBOOT);
queue.add(rightService.getCharacteristic(uuidsList.get(0)));
queue.add(rightService.getCharacteristic(uuidsList.get(1)));
queue.add(rightService.getCharacteristic(uuidsList.get(2)));
}
@Override
public void onCharacteristicRead(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
Log.v(TAG, "CHARACTERISTIC VALUE___: " + characteristic.getIntValue(BluetoothGattCharacteristic.FORMAT_UINT8, 0));
onServicesDiscovered(gatt, 0);
}
};
UPDATE:
even after putting them on different threads it still only reacts to one gatt.readCharacteristic(...). like following:
// Gatt Callback
public static final BluetoothGattCallback mGattCallback = new BluetoothGattCallback() {
// When connection state changes
@Override
public void onConnectionStateChange(BluetoothGatt gatt, int status, int newState) {
if (newState == BluetoothProfile.STATE_CONNECTED) {
Log.v(TAG, "Connected!");
gatt.discoverServices();
}
if (newState == BluetoothProfile.STATE_DISCONNECTED) {
Log.v(TAG, "Disconnected...");
}
}
@Override
public void onServicesDiscovered(BluetoothGatt gatt, int status) {
List<BluetoothGattService> services = gatt.getServices();
/*
DISPLAY ALL SERVICES AND CHARACTERISTICS
for (int i = 0; i < services.size(); i++) {
Log.v(TAG, "SERVICE____: " + services.get(i).getUuid());
for (int k = 0; k < services.get(i).getCharacteristics().size(); k++) {
Log.v(TAG, "CHARACTERISTIC____: " + services.get(i).getCharacteristics().get(k).getUuid());
}
}
*/
BluetoothGattService rightService = null;
for (int i = 0; i < services.size(); i++) {
if (services.get(i).getCharacteristics().size() > 8) {
rightService = services.get(i);
}
}
List<UUID> uuidsList;
UUID TRANSMISSION_POWER = rightService.getCharacteristics().get(4).getUuid();
UUID BROADCASTING_INTERVAL = rightService.getCharacteristics().get(6).getUuid();
UUID BEACON_NAME = rightService.getCharacteristics().get(8).getUuid();
UUID CONNECTION_MODE = rightService.getCharacteristics().get(9).getUuid();
//UUID SOFT_REBOOT = rightService.getCharacteristics().get(10).getUuid();
uuidsList = new ArrayList<UUID>();
uuidsList.add(TRANSMISSION_POWER);
uuidsList.add(BROADCASTING_INTERVAL);
uuidsList.add(BEACON_NAME);
uuidsList.add(CONNECTION_MODE);
//uuidsList.add(SOFT_REBOOT);
class powerThread extends Thread{
UUID uuid;
BluetoothGatt gatt;
BluetoothGattService service;
public powerThread(UUID uuid, BluetoothGatt gatt, BluetoothGattService service) {
this.gatt = gatt;
this.service = service;
this.uuid = uuid;
}
@Override
public void run() {
gatt.readCharacteristic(service.getCharacteristic(uuid));
}
}
powerThread pt = new powerThread(TRANSMISSION_POWER, gatt, rightService);
pt.run();
class intervalThread extends Thread{
UUID uuid;
BluetoothGatt gatt;
BluetoothGattService service;
public intervalThread(UUID uuid, BluetoothGatt gatt, BluetoothGattService service) {
this.gatt = gatt;
this.service = service;
this.uuid = uuid;
}
@Override
public void run() {
gatt.readCharacteristic(service.getCharacteristic(uuid));
}
}
intervalThread it = new intervalThread(BROADCASTING_INTERVAL, gatt, rightService);
it.run();
}
@Override
public void onCharacteristicRead(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
Log.v(TAG, "CHARACTERISTIC VALUE___: " + characteristic.getIntValue(BluetoothGattCharacteristic.FORMAT_UINT8, 0));
}
};
Thanks a lot for your post, Robert K. I wanted my app to read all the characteristic values by itself and show the characteristic value in the ExpandableList instead of the UUID. Robert K's code worked for me but I have experienced an offset of the values that were being displayed. So I made some slight changes and want to share it: 1.
This is how I modified
onServicesDiscovered
andOnCharacteristicRead
:the introduced List of characteristics is emptied after each disconnection.
I introduced a list to save the characteristic values and an integer showing which index I am at (these need to be emptied / set to zero after each disconnection). I modified the
displayData
method, which is called after eachcharacteristicRead
:public
Instead of filling the UUID, I fill the corresponding element of mCharValues
ExpandableListView.OnChildClickListener
. Apart from what I introduced here I left the rest of the code pretty much the same.To anyone who might encounter the same problem, here is an easy solution using a List<> of characteristics.