android BTLE扫描回调似乎未调用



我想在ViewModel中运行BTLE扫描。我不确定这是否是最好的方法,但这对我来说是一个很好的学习实验。当我以更简单的设置运行扫描时,我确实成功地列出了我的BT设备。

在CCD_ 4中设置权限BLUETOOTHBLUETOOTH_ADMINACCESS_COARSE_LOCATION

我在主设置活动中有一个checkPermissions方法,用于检查权限,并在需要时请求权限。我还有一个onRequestPermissionsResult的覆盖。位置权限显示在我的应用程序权限中。这些方法或多或少是开源项目的复制/粘贴(见下文(。

好吧,我的扫描不起作用。回调中的日志从未显示在日志中。没有错误,我在日志中看到的唯一可疑的东西是:D/BluetoothLeScanner: onScannerRegistered() - status=0 scannerId=12 mScannerId=0

我的ViewModel如下所示:

interface ScanningStatus {
void setCurrentlyScanning(boolean updatedStatus);
boolean getCurrentlyScanning();
}

public class SettingsScanningViewModel extends AndroidViewModel {
private MutableLiveData<ArrayList<BluetoothDevice>> mDeviceList;
private ArrayList<BluetoothDevice> mInternalDeviceList;
private MutableLiveData<Boolean> mIsScanning;
private static final String TAG = "viewmodel";
private BluetoothManager mBtManager;
private BluetoothAdapter mBtAdapter;
private BluetoothLeScanner mBleScanner;
private static final long SCAN_PERIOD = 5000;

public SettingsScanningViewModel(Application mApplication) {
super(mApplication);
mDeviceList = new MutableLiveData<>();
mInternalDeviceList = new ArrayList<>();
mIsScanning = new MutableLiveData<>();
mBtManager = (BluetoothManager) mApplication.getSystemService(Context.BLUETOOTH_SERVICE);
mBtAdapter = mBtManager.getAdapter();
mBleScanner = mBtAdapter.getBluetoothLeScanner();
}

MutableLiveData<ArrayList<BluetoothDevice>> getBtDevices() {
return mDeviceList;
}

MutableLiveData<Boolean> getScanningStatus() {
return mIsScanning;
}

private void flushList() {
mInternalDeviceList.clear();
mDeviceList.setValue(mInternalDeviceList);
}

private void injectIntoList(BluetoothDevice btDevice) {
mInternalDeviceList.add(btDevice);
mDeviceList.setValue(mInternalDeviceList);
}

private ScanCallback scanCallback = new ScanCallback() {
@Override
public void onScanResult(int callbackType, ScanResult result) {
Log.d(TAG, "in callback");
BluetoothDevice mDevice = result.getDevice();
if (!mInternalDeviceList.contains(mDevice) && mDevice.getName() != null) {
injectIntoList(mDevice);
}
}
@Override
public void onScanFailed(int errorCode) {
Log.d(TAG, "onScanFailed: " + errorCode);
}
};

private ScanningStatus scanningStatus = new ScanningStatus() {
@Override
public void setCurrentlyScanning(boolean status) {
mIsScanning.setValue(status);
}
public boolean getCurrentlyScanning() {
return mIsScanning.getValue();
}
};

void doBtScan() {
Log.d(TAG, "flush list first");
flushList();
Log.d(TAG, "starting scan with scanCallback " + scanCallback);
scanningStatus.setCurrentlyScanning(true);
mBleScanner.startScan(scanCallback);
Log.d(TAG, "scan should be running for " + SCAN_PERIOD);
Handler handler = new Handler();
handler.postDelayed(() -> {
mBleScanner.stopScan(scanCallback);
scanningStatus.setCurrentlyScanning(false);
}, SCAN_PERIOD);
}
}

我真的不确定出了什么问题。是因为与我发起和请求权限的主要活动相比,我在ViewModel中创建了新的BluetoothManagerBluetoothAdapterBluetoothLeScanner吗?如果是这样的话,我该如何重用这些相同的对象?

谢谢。

OKKKKKKKKKK又是一个试图找出问题的例子,持续了几个小时,发布了帖子,然后200万后我自己发现了。

看起来ACCESS_COARSE_LOCATION还不够,需要ACCESS_FINE_LOCATION。所以我想我一定使用了一个与以前不同的SDK!

谢谢。

这三个步骤对我很有效。

  1. GPS应该打开(如果还没有打开,写一些代码来打开(。

  2. ACCESS_COARSE_LOCATION和ACCESS_FINE_LOCATION权限都应该在清单中声明,并在运行时询问。

  3. 最重要的是下面关于扫描设置的答案

https://stackoverflow.com/a/53831870/10432212

最新更新