Как программно выполнить сопряжение и подключить устройство HID bluetooth (Bluetooth-клавиатура) на Android

Я могу соединить клавиатуру bluetooth, но не могу подключиться, чтобы сделать ее устройством ввода. Я просмотрел документацию, представленную на сайте разработчика - http://developer.android.com/guide/topics/connectivity/bluetooth.html#Profiles

Он говорит, что Android Bluetooth API предоставляет реализации для следующих профилей Bluetooth, но вы можете реализовать интерфейс BluetoothProfile, чтобы написать свои собственные классы для поддержки конкретного Bluetooth профиль.

  • гарнитура
  • профиль A2DP
  • Работоспособности Устройства

нет документации, как реализовать BluetoothProfile для HID bluetooth-устройства (клавиатуры)

Android сам реализовал bluetooth-соединение для HID-устройств, но эти API скрыты. Я тоже попробовал их использовать. Я не получаю никаких ошибок, но клавиатура не подключена в качестве устройства ввода. Вот что я сделал ... --4-->

private void connect(final BluetoothDevice bluetoothDevice) {
    if(bluetoothDevice.getBluetoothClass().getDeviceClass() == 1344){
        final BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
            BluetoothProfile.ServiceListener mProfileListener = new BluetoothProfile.ServiceListener() {
                @Override
                public void onServiceConnected(int profile, BluetoothProfile proxy) {
                    Log.i("btclass", profile + "");

                    if (profile == getInputDeviceHiddenConstant()) {
                        Class instance = null;
                        try {
                            //instance = Class.forName("android.bluetooth.IBluetoothInputDevice");
                            instance = Class.forName("android.bluetooth.BluetoothInputDevice");
                            Method connect = instance.getDeclaredMethod("connect", BluetoothDevice.class);
                            Object value = connect.invoke(proxy, bluetoothDevice);
                            Log.e("btclass", value.toString());
                        } catch (ClassNotFoundException e) {
                            e.printStackTrace();
                        } catch (InvocationTargetException e) {
                            e.printStackTrace();
                        } catch (NoSuchMethodException e) {
                            e.printStackTrace();
                        } catch (IllegalAccessException e) {
                            e.printStackTrace();
                        }



                    }
                }

                @Override
                public void onServiceDisconnected(int profile) {

                }
            };

            mBluetoothAdapter.getProfileProxy(this, mProfileListener,getInputDeviceHiddenConstant());


    }

}

public static int getInputDeviceHiddenConstant() {
    Class<BluetoothProfile> clazz = BluetoothProfile.class;
    for (Field f : clazz.getFields()) {
        int mod = f.getModifiers();
        if (Modifier.isStatic(mod) && Modifier.isPublic(mod) && Modifier.isFinal(mod)) {
            try {
                if (f.getName().equals("INPUT_DEVICE")) {
                    return f.getInt(null);
                }
            } catch (Exception e) {
                Log.e("", e.toString(), e);
            }
        }
    }
    return -1;
}

2 ответов


из-за соображений безопасности сторонние приложения не могут подключаться к клавиатуре bluetooth, поскольку приложение может быть кейлоггером. Таким образом, это может быть сделано только вручную пользователем.


вот код, который я использовал на Android Marshmallow (6.0).. Чтобы запустить соединение L2CAP (необходимо для HID)

public static BluetoothSocket createL2CAPBluetoothSocket(String address, int psm){
    return createBluetoothSocket(BluetoothSocket.TYPE_L2CAP, -1, false,false, address, psm);
}

// method for creating a bluetooth client socket
private static BluetoothSocket createBluetoothSocket(int type, int fd, boolean auth, boolean encrypt, String address, int port){
    Log.e(TAG, "Creating socket with " + address + ":" + port);

    try {
        Constructor<BluetoothSocket> constructor = BluetoothSocket.class.getDeclaredConstructor(
                int.class, int.class,boolean.class,boolean.class,String.class, int.class);
        constructor.setAccessible(true);
        BluetoothSocket clientSocket = (BluetoothSocket) constructor.newInstance(type,fd,auth,encrypt,address,port);
        return clientSocket;
    }catch (Exception e) {
        e.printStackTrace();
    }

    return null;
}

public Boolean connect(View v) {

    try {
        // TODO: Check bluetooth enabled
        mDevice = getController();

        if (mDevice != null) {
            Log.e(TAG, "Controller is paired");

            // Create socket
            mSocket = createL2CAPBluetoothSocket(mDevice.getAddress(), 0x1124);

            if (mSocket != null) {

                if (!mSocket.isConnected()) {
                    mSocket.connect();
                }

                Log.e(TAG, "Socket successfully created");

                ConnectedThread mConnectedThread = new ConnectedThread(mSocket);
                mConnectedThread.run();
            }

        } else {
            showToast("Controller is not connected");
        }

        return true;

    } catch (Exception e) {
        e.printStackTrace();

        if (e instanceof IOException){
            // handle this exception type
        } else {
            // We didn't expect this one. What could it be? Let's log it, and let it bubble up the hierarchy.

        }

        return false;
    }
}

private BluetoothDevice getController() {
    Set<BluetoothDevice> pairedDevices = mBluetoothAdapter.getBondedDevices();

    if (pairedDevices.size() > 0) {
        for (BluetoothDevice device : pairedDevices) {
            if (device.getName().equals("Wireless Controller"))    // Change to match DS4 - node name
            {
                Log.d(TAG, "Found device named: " + device.getName());

                return device;
            }
        }
    }

    return null;
}

у него все еще могут быть проблемы с созданием службы, и вам нужно установить правильный L2CAP PSAM для устройства, но надеюсь, что это может помочь..