1.常规操作经典蓝牙 最近有一个功能需要使用蓝牙BLE,如果在使用之前蓝牙是关闭的状态,需要先打开蓝牙,并且在使用完之后,需要把蓝牙关闭。 我们的功能是在后台服务执行,最开始使用常规的方式判断并打开蓝牙:
boolean isEnableBefor = bluetoothAdapter.isEnabled();
if (!isEnableBefor) {
bluetoothAdapter.enable();
}
这个时候会打开整个经典蓝牙,然后用户界面也会感知到蓝牙的打开;如果用户正在使用,然后这个时候功能执行完毕关闭蓝牙,会让用户觉得体验很不友好,甚至认为这是个程序设计上的问题。
2.BLE和经典蓝牙的关系 于是决定本功能模块单独使用蓝牙BLE,而不去操作经典蓝牙,这样就不会影响到设置模块对蓝牙开关的操作逻辑了。 实际上我们在之前分析过,在打开蓝牙的过程中,首先就是打开BLE,在Android系统中对应的就是GattService;再打开BLE之后,再去打开其他的Profile服务,所有Profile服务开启完毕之后,再通过广播形式通知蓝牙开关打开成功。 我们可以把蓝牙BLE看成是整个经典蓝牙树的一个子集。我们现在要做的,就是只启动GattService 整个Profile,不去启动后面其他的Profile。
3.源码修改 实际上在源码中存在有单独操作BLE的方法,但在谷歌官方的SDK中,并没有开放这些方法。但源码位于:framework\base\core\java\android\bluetooth\BluetoothAdapter.java
@SystemApi
public boolean isLeEnabled() {
final int state = getLeState();
if (DBG) {
Log.d(TAG, "isLeEnabled(): " + BluetoothAdapter.nameForState(state));
}
return (state == BluetoothAdapter.STATE_ON || state == BluetoothAdapter.STATE_BLE_ON);
}
* @return true to indicate Bluetooth LE will be available, or false on immediate error
* @hide
*/
@SystemApi
public boolean enableBLE() {
if (!isBleScanAlwaysAvailable()) {
return false;
}
......
return false;
}
我们看到,这些方法都是隐藏的,而且必须是系统级应用才能调用。 我们要调用,需要去掉@SystemApi,然后再编译一个framework.jar,具体的操作可以看这里:编译自己的framework 或者用我已经编译好的jar包:framwork.jar
4.前提条件 在打开BLE之前我们还要做一件事,就是确保 isBleScanAlwaysAvailable()返回的是true,不然执行enableBLE()是没有作用的。我在一步步查看了Bluetooth源码之后发现,其实就是操作的一个全局的字段,可执行如下操作:
Settings.Global.putInt(context.getContentResolver(),Settings.Global.BLE_SCAN_ALWAYS_AVAILABLE,1);
这样就会保证对BLE的操作可以顺利执行下去了。
5.状态改变的广播接收 打开BLE的时候,是一个异步的过程,状态的改变蓝牙服务会通过广播的形式通知出来:
IntentFilter filter = new IntentFilter();
filter.addAction(BluetoothAdapter.ACTION_BLE_STATE_CHANGED);
filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED);
context.registerReceiver(btReceiver, filter);
BroadcastReceiver btReceiver = new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
if(intent.getAction().equals(BluetoothAdapter.ACTION_BLE_STATE_CHANGED)){
int state = intent.getIntExtra(BluetoothAdapter.EXTRA_STATE, BluetoothAdapter.STATE_OFF);
Log.d(TAG,"ACTION_LE_STATE_CHANGED: " + state);
if (BluetoothAdapter.STATE_BLE_ON == state || BluetoothAdapter.STATE_ON == state) {
Log.d(TAG, "BluetoothBLE opened!");
}
}
}
};
|