蓝牙HID——将android设备变成蓝牙键盘(BluetoothHidDevice)

前言

前段时间发现自己的老笔记本键盘失灵了,又没有多的键盘,于是苦恼了好久。于是萌生了自己做一个键盘的想法。这段时间一直在研究蓝牙HID,通过蓝牙HID将android手机变成一个蓝牙键盘,这样就不用担心无键盘的问题了。
蓝牙HID系列篇章:
蓝牙HID——将android设备变成蓝牙键盘(BluetoothHidDevice)
蓝牙HID——android利用手机来解锁电脑/平板/iPhone
蓝牙HID——Android手机注册HID时出现 Could not bind to Bluetooth (HID Device) Service with Intent * 的问题分析
蓝牙HID——将android设备变成蓝牙鼠标/触控板(BluetoothHidDevice)

HID开发

通过研究发现android9.0之后开放了BluetoothHidDevice等HID相关的API,从此入手开始HID开发。(一定要看到最后!)
首先通过获取HID设备代理得到BluetoothHidDevice

mBtAdapter.getProfileProxy(this, new BluetoothProfile.ServiceListener() {
            @Override
            public void onServiceConnected(int i, BluetoothProfile bluetoothProfile) {
                Log.d(TAG, "onServiceConnected:" + i);
                if (i == BluetoothProfile.HID_DEVICE) {
                    if (!(bluetoothProfile instanceof BluetoothHidDevice)) {
                        Log.e(TAG, "Proxy received but it's not BluetoothHidDevice");
                        return;
                    }
                    mHidDevice = (BluetoothHidDevice) bluetoothProfile;
                    registerBluetoothHid();
                    //启动设备发现
                    startActivityForResult(new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE), 1);
                }

            }

            @Override
            public void onServiceDisconnected(int i) {
                Log.d(TAG, "onServiceDisconnected:" + i);

            }
        }, BluetoothProfile.HID_DEVICE);

需要开启设备发现:

startActivityForResult(new Intent(BluetoothAdapter.ACTION_REQUEST_DISCOVERABLE), 1);

这样才能被主机端发现并进行配对,上面的方式会弹框手动点击允许才可以启动设备发现。当然也可以通过反射调用隐藏的接口setScanMode开启设备发现,这样就没有弹框:

    public static void setScanMode(BluetoothAdapter btAdapter) {
        //启动设备发现,让HID能被其他设备发现
        try {
            for(Method m: BluetoothAdapter.class.getMethods()){
                if("setScanMode".equals(m.getName())&& m.getParameterCount()>1){
                    m.invoke(btAdapter,BluetoothAdapter.SCAN_MODE_CONNECTABLE_DISCOVERABLE, 100000);
                    break;
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

通过得到BluetoothHidDevice注册为HID应用

  mHidDevice.registerApp(sdpSettings, null, qosSettings, Executors.newCachedThreadPool(), new BluetoothHidDevice.Callback() {
            @Override
            public void onAppStatusChanged(BluetoothDevice pluggedDevice, boolean registered) {
                Log.d(TAG, "onAppStatusChanged:" + (pluggedDevice!= null ? pluggedDevice.getName() : "null") + " registered:" + registered);
                if (registered) {

                    List<BluetoothDevice> matchingDevices = mHidDevice.getDevicesMatchingConnectionStates(mMatchingStates);

                    Log.d(TAG, "paired devices: " + matchingDevices + "  " + mHidDevice.getConnectionState(pluggedDevice));
                    if (pluggedDevice != null && mHidDevice.getConnectionState(pluggedDevice) != BluetoothProfile.STATE_CONNECTED) {
                        boolean result = mHidDevice.connect(pluggedDevice);
                        Log.d(TAG, "hidDevice connect:"  + result);
                    } else if (matchingDevices != null && matchingDevices.size() > 0){
                        //TODO 选择设备进行连接
                    } else {
                        //TODO 蓝牙HID注册成功,但未进行配对。
                    }
                }
            }

            @Override
            public void onConnectionStateChanged(BluetoothDevice device, int state) {
                Log.d(TAG, "onConnectionStateChanged:" + device + "  state:" + state);
                 if (state == BluetoothProfile.STATE_CONNECTED) {
                     mHostDevice = device;
                } if (state == BluetoothProfile.STATE_DISCONNECTED) {
                    mHostDevice = null;
                } else if (state == BluetoothProfile.STATE_CONNECTING) {
                
                }
            }
        });

当收到onAppStatusChanged回调registered为true代表设备支持蓝牙HID,收不到回调代表设备不支持。
主要通过SDP协议(Service Discovery Protocol)配置成HID键盘:

    private final BluetoothHidDeviceAppSdpSettings sdpSettings = new BluetoothHidDeviceAppSdpSettings(
            HidConfig.NAME, HidConfig.DESCRIPTION, HidConfig.PROVIDER,
            BluetoothHidDevice.SUBCLASS1_COMBO, HidConfig.KEYBOARD_COMBO);

BluetoothHidDeviceAppQosSettings可以为空。

与主机端连接过程中会收到onConnectionStateChanged连接状态的回掉,连接成功后得到主机端抽象设备BluetoothDevice,之后可以向此设备发送报告。

        mHidDevice.sendReport(mHostDevice, 8, new byte[]{mModifyKeyByte, 0, (byte) code, 0, 0, 0, 0, 0});

注意report id与报告描述符定义的id对应,否则发送失败。定义的字节长度也与报告描述符定义的字节对应,前两个字节为功能键,后6个字节为字符键,这里定义了可同时有6个字符键按下,释放时为0x00。
详细键盘设备配置如下:

public class HidConfig {
 
    public final static String NAME = "Evin Keyboard";
 
    public final static String DESCRIPTION = "Evin for you";
 
    public final static String PROVIDER = "Evin";

    public static final byte[] KEYBOARD_COMBO =
            {
                    (byte) 0x05, (byte) 0x01, //USAGE_PAGE (Generic Desktop)
                    (byte) 0x09, (byte) 0x06, //USAGE (Keyboard)
                    (byte) 0xA1, (byte) 0x01, //COLLECTION (Application)
                    (byte) 0x85, (byte) 0x08, //REPORT_ID (8)
                    (byte) 0x05, (byte) 0x07, //USAGE_PAGE (Keyboard)
                    (byte) 0x19, (byte) 0xE0, //USAGE_MINIMUM (Keyboard LeftControl)
                    (byte) 0x29, (byte) 0xE7, //USAGE_MAXIMUM (Keyboard Right GUI)
                    (byte) 0x15, (byte) 0x00, //LOGICAL_MINIMUM (0)
                    (byte) 0x25, (byte) 0x01, //LOGICAL_MAXIMUM (1)
                    //第一个字节
                    (byte) 0x75, (byte) 0x01, //REPORT_SIZE (1)
                    (byte) 0x95, (byte) 0x08, //REPORT_COUNT (8)
                    (byte) 0x81, (byte) 0x02, //INPUT (Data,Var,Abs)
                    //第二个字节
                    (byte) 0x95, (byte) 0x01, //REPORT_COUNT (1)
                    (byte) 0x75, (byte) 0x08, //REPORT_SIZE (8)
                    (byte) 0x81, (byte) 0x03, //INPUT (Cnst,Var,Abs)
                    //后六个字节
                    (byte) 0x95, (byte) 0x06, //REPORT_COUNT (6)
                    (byte) 0x75, (byte) 0x08, //REPORT_SIZE (8)
                    (byte) 0x15, (byte) 0x00, //LOGICAL_MINIMUM (0)
                    (byte) 0x25, (byte) 0x65, //LOGICAL_MAXIMUM (101)
                    (byte) 0x05, (byte) 0x07, //USAGE_PAGE (Keyboard)
                    (byte) 0x19, (byte) 0x00, //USAGE_MINIMUM (Reserved (no event indicated))
                    (byte) 0x29, (byte) 0x65, //USAGE_MAXIMUM (Keyboard Application)
                    (byte) 0x81, (byte) 0x00, //INPUT (Data,Ary,Abs)
                    (byte) 0xC0  //END_COLLECTION
            };
 }

一个外设主要通过HID报告描述符来供主机端识别,正确定义描述符,并发送正确的报告给主机端才能正常控制主机端。上面的报告描述符定义了2个字节功能键(ctrl、shift、alt等),使用了8个连续id的键,故只占用一个字节,另一个字节保留。

 data0 --功能键分为左右两边,按下为1,释放为0
  |--bit0: Left Control
  |--bit1: Left Shift
  |--bit2: Left Alt
  |--bit3: Left GUI
  |--bit4: Right Control
  |--bit5: Right Shift
  |--bit6: Right Alt
  |--bit7: Right GUI
  data1 -- 预留

蓝牙HID的和USB HID的键盘按键值一致,具体可自行搜索或者参考HID表:USB HID Table

效果

自定义键盘可参考另一篇博客:android实现仿真键盘(KeyboardView适配)
在这里插入图片描述

可以实现复制粘贴、上下左右切换,同时增加声音反馈,模拟真实键盘感受。
在这里插入图片描述

完整效果看下面视频:

将android手机变成蓝牙键盘

视频中用的是华为P30pro手机,搭载android10系统(之前升级成鸿蒙系统,但鸿蒙系统HID注册后无响应,于是刷回android系统)。

同理,利用蓝牙HID,可以将支持蓝牙HID的设备模拟成各种蓝牙外设,比如鼠标、触控板、按键面板、游戏手柄(摇杆)。

最后,希望大家点个赞,鼓励一下!

  • 36
    点赞
  • 67
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 13
    评论
【软件功能】 蓝牙遥控器软件,可以将你的手机变成一部电脑遥控器,使用手机可以控制电脑的鼠标操作、键盘的输入,如果你的手机具有手写功能,还可以将你的手机变成手写板。 这个软件分为两部分,一个是安装到手机上,另一个安装在电脑上。 手机端: 软件文件名为BtRemoteControl.jar,它要求安装的手机具有java功能,并支持MIDP2.0,具有蓝牙功能。 电脑端: 要想在电脑上运行遥控器,要求安装蓝牙驱动程序widcomm驱动或IVT BlueSoleil驱动。 同时,还要求电脑上安装.net运行库2.0,如果你的电脑没有安装,必须先安装Microsoft .NET Framework 2.0 。 【使用说明】 电脑端的安装使用: 蓝牙遥控器电脑端在电脑上安装后,双击图标桌面上产生的图标。 首先在“请选择接受遥控的蓝牙虚拟串口”后面,选择蓝牙的虚拟串口名字,软件里已经包括了从com1到com20的串口名字,请根据自己机器上的蓝牙虚拟串口名字选择,如果你无法确定,可以用手机上的蓝牙遥控器软件测试1下。 选择好串口名字后,点下“打开串口”按钮,将显示“蓝牙串口状态:"已打开,这时就可以接受手机方的控制了。 手机端的安装使用: 和其它的JAVA程序在手机上的安装过程一样,先用蓝牙、红外线或数据线将手机与电脑连接,将BtRemoteControl.jar安装到手机,在手机的功能表里就会出现“蓝牙遥控器”标志,运行软件,会出现“启动蓝牙”选择,运行该选择,软件将搜索周围准备接受遥控的电脑上的蓝牙设备,搜索片刻后,将显示搜到蓝牙设备,并接着显示蓝牙设备上的蓝牙虚拟串口服务是否存在。 如果搜索后显示未找到蓝牙设备,请检查你的手机是否启动了蓝牙,你的电脑上的蓝牙驱动程序是否启动,蓝牙设备是否插好。 在找到蓝牙设备蓝牙上的虚拟串口后,手机上将会有如下3个菜单选择: 发送测试:用来测试电脑上的蓝牙虚拟串口是否可以接受数据,同时可以测试出蓝牙虚拟串口的名字,供电脑方的遥控软件设置使用,按下这个按钮,就可以向电脑的虚拟串口发送1个数据,测试串口。 控制鼠标和键盘:选择了这个菜单,将显示遥控鼠标、键盘界面.这个选择有3个功能,用户可以按下#号键,在这三个功能之间切换,这三个功能: 1、遥控鼠标。手机方向键的上下左右四个键,控制鼠标上下左右移动,按下0键,遥控单击鼠标右键,按下9键,遥控单击鼠标左键,按下ok键,遥控双击鼠标左键。按下1,3,7个键,调整鼠标移动的距离,1键是短距离,3键是中距离,7键是长距离。 2、遥控鼠标自动移动。按下上下左右四个方向键,鼠标将按照操作自动上下左右移动,一直到屏幕的边缘才停止,按下*号键,停止自动运行。 3、遥控键盘。按下手机上的0到9号键,将向电脑发送命令0到9,在电脑上,可以进行设置,当接到命令时如何控制键盘。电脑端的操作如下图,选择接到命令时,要按下的按键,点“添加”按钮,就可以添加成功,选择后,殿下“删除”按钮,就可以删除: 文字输入:按下这个菜单,将可以输入文字到电脑。在手机中输入文字,然后点下“发送文字”按钮,就可以将文字发送到电脑了。
评论 13
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

言并肃

感谢大哥支持!您的鼓励是我动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值