| /**************************************************************************** |
| ** |
| ** Copyright (C) 2016 The Qt Company Ltd. |
| ** Contact: https://www.qt.io/licensing/ |
| ** |
| ** This file is part of the QtBluetooth module of the Qt Toolkit. |
| ** |
| ** $QT_BEGIN_LICENSE:LGPL$ |
| ** Commercial License Usage |
| ** Licensees holding valid commercial Qt licenses may use this file in |
| ** accordance with the commercial license agreement provided with the |
| ** Software or, alternatively, in accordance with the terms contained in |
| ** a written agreement between you and The Qt Company. For licensing terms |
| ** and conditions see https://www.qt.io/terms-conditions. For further |
| ** information use the contact form at https://www.qt.io/contact-us. |
| ** |
| ** GNU Lesser General Public License Usage |
| ** Alternatively, this file may be used under the terms of the GNU Lesser |
| ** General Public License version 3 as published by the Free Software |
| ** Foundation and appearing in the file LICENSE.LGPL3 included in the |
| ** packaging of this file. Please review the following information to |
| ** ensure the GNU Lesser General Public License version 3 requirements |
| ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. |
| ** |
| ** GNU General Public License Usage |
| ** Alternatively, this file may be used under the terms of the GNU |
| ** General Public License version 2.0 or (at your option) the GNU General |
| ** Public license version 3 or any later version approved by the KDE Free |
| ** Qt Foundation. The licenses are as published by the Free Software |
| ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 |
| ** included in the packaging of this file. Please review the following |
| ** information to ensure the GNU General Public License requirements will |
| ** be met: https://www.gnu.org/licenses/gpl-2.0.html and |
| ** https://www.gnu.org/licenses/gpl-3.0.html. |
| ** |
| ** $QT_END_LICENSE$ |
| ** |
| ****************************************************************************/ |
| |
| #include <QtCore/QLoggingCategory> |
| #include <QtCore/private/qjnihelpers_p.h> |
| #include <QtCore/qrandom.h> |
| #include "localdevicebroadcastreceiver_p.h" |
| #include "android/jni_android_p.h" |
| |
| QT_BEGIN_NAMESPACE |
| |
| Q_DECLARE_LOGGING_CATEGORY(QT_BT_ANDROID) |
| |
| const char *scanModes[] = {"SCAN_MODE_NONE", "SCAN_MODE_CONNECTABLE", "SCAN_MODE_CONNECTABLE_DISCOVERABLE"}; |
| const char *bondModes[] = {"BOND_NONE", "BOND_BONDING", "BOND_BONDED"}; |
| |
| LocalDeviceBroadcastReceiver::LocalDeviceBroadcastReceiver(QObject *parent) : |
| AndroidBroadcastReceiver(parent), previousScanMode(0) |
| { |
| addAction(valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ActionBondStateChanged)); |
| addAction(valueForStaticField(JavaNames::BluetoothAdapter, JavaNames::ActionScanModeChanged)); |
| addAction(valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ActionAclConnected)); |
| addAction(valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ActionAclDisconnected)); |
| if (QtAndroidPrivate::androidSdkVersion() >= 15) |
| addAction(valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ActionPairingRequest)); //API 15 |
| |
| //cache integer values for host & bonding mode |
| //don't use the java fields directly but refer to them by name |
| QAndroidJniEnvironment env; |
| for (uint i = 0; i < (sizeof(hostModePreset)/sizeof(hostModePreset[0])); i++) { |
| hostModePreset[i] = QAndroidJniObject::getStaticField<jint>( |
| "android/bluetooth/BluetoothAdapter", |
| scanModes[i]); |
| if (env->ExceptionCheck()) { |
| env->ExceptionDescribe(); |
| env->ExceptionClear(); |
| hostModePreset[i] = 0; |
| } |
| } |
| |
| for (uint i = 0; i < (sizeof(bondingModePreset)/sizeof(bondingModePreset[0])); i++) { |
| bondingModePreset[i] = QAndroidJniObject::getStaticField<jint>( |
| "android/bluetooth/BluetoothDevice", |
| bondModes[i]); |
| if (env->ExceptionCheck()) { |
| env->ExceptionDescribe(); |
| env->ExceptionClear(); |
| bondingModePreset[i] = 0; |
| } |
| } |
| } |
| |
| void LocalDeviceBroadcastReceiver::onReceive(JNIEnv *env, jobject context, jobject intent) |
| { |
| Q_UNUSED(context); |
| Q_UNUSED(env); |
| |
| QAndroidJniObject intentObject(intent); |
| const QString action = intentObject.callObjectMethod("getAction", "()Ljava/lang/String;").toString(); |
| qCDebug(QT_BT_ANDROID) << QStringLiteral("LocalDeviceBroadcastReceiver::onReceive() - event: %1").arg(action); |
| |
| if (action == valueForStaticField(JavaNames::BluetoothAdapter, |
| JavaNames::ActionScanModeChanged).toString()) { |
| |
| const QAndroidJniObject extrasBundle = |
| intentObject.callObjectMethod("getExtras","()Landroid/os/Bundle;"); |
| const QAndroidJniObject keyExtra = valueForStaticField(JavaNames::BluetoothAdapter, |
| JavaNames::ExtraScanMode); |
| |
| int extra = extrasBundle.callMethod<jint>("getInt", |
| "(Ljava/lang/String;)I", |
| keyExtra.object<jstring>()); |
| |
| if (previousScanMode != extra) { |
| previousScanMode = extra; |
| |
| if (extra == hostModePreset[0]) |
| emit hostModeStateChanged(QBluetoothLocalDevice::HostPoweredOff); |
| else if (extra == hostModePreset[1]) |
| emit hostModeStateChanged(QBluetoothLocalDevice::HostConnectable); |
| else if (extra == hostModePreset[2]) |
| emit hostModeStateChanged(QBluetoothLocalDevice::HostDiscoverable); |
| else |
| qCWarning(QT_BT_ANDROID) << "Unknown Host State"; |
| } |
| } else if (action == valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ActionBondStateChanged).toString()) { |
| //get BluetoothDevice |
| QAndroidJniObject keyExtra = valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ExtraDevice); |
| const QAndroidJniObject bluetoothDevice = |
| intentObject.callObjectMethod("getParcelableExtra", |
| "(Ljava/lang/String;)Landroid/os/Parcelable;", |
| keyExtra.object<jstring>()); |
| |
| //get new bond state |
| keyExtra = valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ExtraBondState); |
| const QAndroidJniObject extrasBundle = |
| intentObject.callObjectMethod("getExtras","()Landroid/os/Bundle;"); |
| int bondState = extrasBundle.callMethod<jint>("getInt", |
| "(Ljava/lang/String;)I", |
| keyExtra.object<jstring>()); |
| |
| QBluetoothAddress address(bluetoothDevice.callObjectMethod<jstring>("getAddress").toString()); |
| if (address.isNull()) |
| return; |
| |
| if (bondState == bondingModePreset[0]) |
| emit pairingStateChanged(address, QBluetoothLocalDevice::Unpaired); |
| else if (bondState == bondingModePreset[1]) |
| ; //we ignore this as Qt doesn't have equivalent API value |
| else if (bondState == bondingModePreset[2]) |
| emit pairingStateChanged(address, QBluetoothLocalDevice::Paired); |
| else |
| qCWarning(QT_BT_ANDROID) << "Unknown BOND_STATE_CHANGED value:" << bondState; |
| |
| } else if (action == valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ActionAclConnected).toString() || |
| action == valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ActionAclDisconnected).toString()) { |
| |
| const QString connectEvent = valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ActionAclConnected).toString(); |
| const bool isConnectEvent = |
| action == connectEvent ? true : false; |
| |
| //get BluetoothDevice |
| const QAndroidJniObject keyExtra = valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ExtraDevice); |
| QAndroidJniObject bluetoothDevice = |
| intentObject.callObjectMethod("getParcelableExtra", |
| "(Ljava/lang/String;)Landroid/os/Parcelable;", |
| keyExtra.object<jstring>()); |
| |
| QBluetoothAddress address(bluetoothDevice.callObjectMethod<jstring>("getAddress").toString()); |
| if (address.isNull()) |
| return; |
| |
| emit connectDeviceChanges(address, isConnectEvent); |
| } else if (action == valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ActionPairingRequest).toString()) { |
| |
| QAndroidJniObject keyExtra = valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ExtraPairingVariant); |
| int variant = intentObject.callMethod<jint>("getIntExtra", |
| "(Ljava/lang/String;I)I", |
| keyExtra.object<jstring>(), |
| -1); |
| |
| int key = -1; |
| switch (variant) { |
| case -1: //ignore -> no pairing variant set |
| return; |
| case 0: //BluetoothDevice.PAIRING_VARIANT_PIN |
| { |
| qCDebug(QT_BT_ANDROID) << "Pairing : PAIRING_VARIANT_PIN -> use Android default handling"; |
| |
| // The section below is disabled because this Android pairing variant |
| // requires the user to enter a pin. Since QBluetoothLocalDevice does |
| // not have a setPin() equivalent which might be used to return the user's value. |
| // For now we ignore this request. If an app ignores such requests, |
| // Android shows a "fall-back" pin code entry form. |
| /* |
| //generate a random key |
| const QString pin = QStringLiteral("%1").arg(QRandomGenerator::global()->bounded(1000000), |
| 6, 10, QLatin1Char('0')); |
| const QAndroidJniObject javaPin = QAndroidJniObject::fromString(pin); |
| |
| //get BluetoothDevice |
| keyExtra = valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ExtraDevice); |
| QAndroidJniObject bluetoothDevice = |
| intentObject.callObjectMethod("getParcelableExtra", |
| "(Ljava/lang/String;)Landroid/os/Parcelable;", |
| keyExtra.object<jstring>()); |
| if (!bluetoothDevice.isValid()) |
| return; |
| |
| QAndroidJniObject bytePin = QAndroidJniObject::callStaticObjectMethod("android/bluetooth/BluetoothDevice", |
| "convertPinToBytes", |
| "(Ljava/lang/String;)[B", |
| javaPin.object<jstring>()); |
| if (!bytePin.isValid()) { |
| if (env->ExceptionCheck()) { |
| env->ExceptionDescribe(); |
| env->ExceptionClear(); |
| } |
| return; |
| } |
| |
| jboolean result = bluetoothDevice.callMethod<jboolean>("setPin", "([B)Z", bytePin.object<jbyteArray>()); |
| if (!result) { |
| if (env->ExceptionCheck()) { |
| env->ExceptionDescribe(); |
| env->ExceptionClear(); |
| } |
| return; |
| } |
| |
| const QBluetoothAddress address(bluetoothDevice.callObjectMethod<jstring>("getAddress").toString()); |
| emit pairingDisplayPinCode(address, pin);*/ |
| break; |
| } |
| case 2: //BluetoothDevice.PAIRING_VARIANT_PASSKEY_CONFIRMATION |
| { |
| qCDebug(QT_BT_ANDROID) << "Pairing : PAIRING_VARIANT_PASSKEY_CONFIRMATION"; |
| keyExtra = valueForStaticField(JavaNames::BluetoothDevice, |
| JavaNames::ExtraPairingKey); |
| key = intentObject.callMethod<jint>("getIntExtra", |
| "(Ljava/lang/String;I)I", |
| keyExtra.object<jstring>(), |
| -1); |
| if (key == -1) |
| return; |
| |
| keyExtra = valueForStaticField(JavaNames::BluetoothDevice, JavaNames::ExtraDevice); |
| QAndroidJniObject bluetoothDevice = |
| intentObject.callObjectMethod("getParcelableExtra", |
| "(Ljava/lang/String;)Landroid/os/Parcelable;", |
| keyExtra.object<jstring>()); |
| |
| if (!bluetoothDevice.isValid()) |
| return; |
| |
| //we need to keep a reference around in case the user confirms later on |
| pairingDevice = bluetoothDevice; |
| |
| QBluetoothAddress address(bluetoothDevice.callObjectMethod<jstring>("getAddress").toString()); |
| |
| //User has choice to confirm or not. If no confirmation is happening |
| //the OS default pairing dialog can be used or timeout occurs. |
| emit pairingDisplayConfirmation(address, QString::number(key)); |
| break; |
| } |
| default: |
| qCWarning(QT_BT_ANDROID) << "Unknown pairing variant: " << variant; |
| return; |
| } |
| } |
| } |
| |
| bool LocalDeviceBroadcastReceiver::pairingConfirmation(bool accept) |
| { |
| if (!pairingDevice.isValid()) |
| return false; |
| |
| // setPairingConfirmation() is likely to throw SecurityException for BLUETOOTH_PRIVILEGED |
| // as this permission is not obtainable for 3rdparties |
| // Note: Normally it would not have been added to Qt API but it used to be BLUETOOTH_ADMIN |
| QAndroidJniEnvironment env; |
| bool success = pairingDevice.callMethod<jboolean>("setPairingConfirmation", |
| "(Z)Z", accept); |
| if (success) { |
| if (env->ExceptionCheck()) { |
| env->ExceptionDescribe(); |
| env->ExceptionClear(); |
| } |
| } |
| pairingDevice = QAndroidJniObject(); |
| return success; |
| } |
| |
| QT_END_NAMESPACE |