Mouse not detected.
- @cghague
I am using a Bluetooth device that converts a wired USB keyboard and mouse into a Bluetooth-enabled combo. This allows me to plug in a USB receiver for a keyboard and mouse combo into the Bluetooth device, making it accessible via Bluetooth.
I am trying to use this setup to make my TinyPilot device Bluetooth-enabled so that I can connect my laptop to the TinyPilot via this Bluetooth device and use my laptop's keyboard and trackpad wirelessly. However, during testing with the TinyPilot, the bluetooth device detects only the keyboard, but not the mouse. Additionally, when attempting to use the mouse through TinyPilot, its movement is erratic, while the keyboard works perfectly.
I suspect this issue might be a bug in TinyPilot's configuration, particularly in the init-usb-gadget script. For reference, I tested the Bluetooth device with a USB combo receiver, and it successfully detected both the keyboard and mouse simultaneously.
- CCharles Hague @cghague2025-01-09 02:44:28.572Z
Hi Ather, thanks for your question about using TinyPilot with your Bluetooth device.
The configuration you've described isn't one we've seen before, so we'll need to investigate this issue. Can you please provide the adapter's make and model as a starting point? Can you please also send me a link to your debug logs (while the adapter is connected) by going to System, then Logs, then Get Shareable URL?
- AAther @ather
I am using Bluetooth Adapter for Keyboard and Mouse (BT-600) from Handheld Scientific.
Here is the logs data WITH the device connected : https://logs.tinypilotkvm.com/ciST8suL
Here is the logs data WITHOUT the device connected: https://logs.tinypilotkvm.com/CfPoky6QP.S I am using the TinyPilot Power Connector.
- CCharles Hague @cghague2025-01-10 02:00:41.905Z
Thanks for sharing those details; that's a fascinating device!
I've reviewed your logs, and there is no obvious explanation for this issue. However, I did note that the adapter's features page has some caveats about which keyboard, mouse, and hub combinations the adapter supports. I also noted that the manual hints that the adapter uses relative mouse mode.
Therefore, I suspect the issue is that the adapter might be incompatible with either absolute positioning mode or the type of composite USB device that TinyPilot presents. I appreciate this isn't the answer you were hoping for, so it might be worth checking with the adapter's manufacturer to see if they have any suggestions. Please let me know if you have any questions.
- AAther @ather
Can you help me understand why TinyPilot uses absolute positioning instead of relative?
If I can modify the descriptor of the mouse what are my limitations of switching to relative for TinyPilot?
- CCharles Hague @cghague2025-03-21 16:39:55.911Z
Hi Ather, thanks for your great question about mouse positioning modes.
Absolute mouse positioning is highly compatible with many target machines, works well with a browser-based interface, and is ideal for use with the video stream's fixed dimensions. As a result, it is popular with graphics tablets, touchscreens, trackpads, and remote access software.
However, some older machines and limited systems only support relative mouse positioning. We're always looking to improve TinyPilot's compatibility, so we're actively interested in adding support for relative mouse positioning. Unfortunately, I can't provide a timeframe for when or if we will add it.
Unfortunately, adding support for relative mouse positioning involves more than changing the USB descriptors, so that approach won't work. We also advise against manually changing those descriptors, as software updates will overwrite your modifications.
I hope this is helpful. Please let me know if you have any questions.
- AIn reply toather⬆:Ather @ather
@cghague I managed to get it partially working by using relative positioning with HID and making some updates to mouse.py. My understanding of this is still somewhat limited, though. It doesn’t work as smoothly as absolute positioning, but it’s a good starting point.
HID Descriptor for mouse:
/opt/tinypilot-privileged/init-usb-gadgetecho 5 > "${USB_MOUSE_FUNCTIONS_DIR}/report_length" # Write the report descriptor D=$(mktemp) { echo -ne \\x05\\x01 # USAGE_PAGE (Generic Desktop) echo -ne \\x09\\x02 # USAGE (Mouse) echo -ne \\xA1\\x01 # COLLECTION (Application) # 8-buttons echo -ne \\x05\\x09 # USAGE_PAGE (Button) echo -ne \\x19\\x01 # USAGE_MINIMUM (Button 1) echo -ne \\x29\\x08 # USAGE_MAXIMUM (Button 8) echo -ne \\x15\\x00 # LOGICAL_MINIMUM (0) echo -ne \\x25\\x01 # LOGICAL_MAXIMUM (1) echo -ne \\x95\\x08 # REPORT_COUNT (8) echo -ne \\x75\\x01 # REPORT_SIZE (1) echo -ne \\x81\\x02 # INPUT (Data,Var,Abs) # x,y absolute coordinates echo -ne \\x05\\x01 # USAGE_PAGE (Generic Desktop) echo -ne \\x09\\x30 # USAGE (X) echo -ne \\x09\\x31 # USAGE (Y) echo -ne \\x15\\x81 # LOGICAL_MINIMUM (-127) echo -ne \\x25\\x7F # LOGICAL_MAXIMUM (127) echo -ne \\x75\\x08 # REPORT_SIZE (8) echo -ne \\x95\\x02 # REPORT_COUNT (2) echo -ne \\x81\\x06 # INPUT (Data,Var,Rel) # vertical wheel echo -ne \\x09\\x38 # USAGE (wheel) echo -ne \\x15\\x81 # LOGICAL_MINIMUM (-127) echo -ne \\x25\\x7F # LOGICAL_MAXIMUM (127) echo -ne \\x75\\x08 # REPORT_SIZE (8) echo -ne \\x95\\x01 # REPORT_COUNT (1) echo -ne \\x81\\x06 # INPUT (Data,Var,Rel) # horizontal wheel echo -ne \\x05\\x0C # USAGE_PAGE (Consumer Devices) echo -ne \\x0A\\x38\\x02 # USAGE (AC Pan) echo -ne \\x15\\x81 # LOGICAL_MINIMUM (-127) echo -ne \\x25\\x7F # LOGICAL_MAXIMUM (127) echo -ne \\x75\\x08 # REPORT_SIZE (8) echo -ne \\x95\\x01 # REPORT_COUNT (1) echo -ne \\x81\\x06 # INPUT (Data,Var,Rel) echo -ne \\xC0 # END_COLLECTION } >> "$D"
/opt/tinypilot/app/hid/mouse.py
from hid import write as hid_write # Global variables to store the previous mouse coordinates _last_x = None _last_y = None # Sensitivity multiplier: adjust this value to increase/decrease movement SENSITIVITY_MULTIPLIER = 1250 # Experiment with this value def send_mouse_event(mouse_path, buttons, current_x, current_y, vertical_wheel_delta, horizontal_wheel_delta): x, y = _scale_mouse_coordinates(current_x, current_y) buf = [0] * 5 buf[0] = buttons buf[1] = x & 0xff buf[2] = y & 0xff buf[3] = vertical_wheel_delta & 0xff buf[4] = horizontal_wheel_delta & 0xff hid_write.write_to_hid_interface(mouse_path, buf) def _scale_mouse_coordinates(current_x, current_y): global _last_x, _last_y if _last_x is None or _last_y is None: # First event, assume no movement. delta_x = 0 delta_y = 0 else: delta_x = current_x - _last_x delta_y = current_y - _last_y _last_x, _last_y = current_x, current_y # Multiply the difference by the sensitivity multiplier. x = int(delta_x * SENSITIVITY_MULTIPLIER) y = int(delta_y * SENSITIVITY_MULTIPLIER) # Clip values to ensure they're within the expected range (-127 to 127). x = max(-127, min(127, x)) y = max(-127, min(127, y)) return x, y def _translate_vertical_wheel_delta(vertical_wheel_delta): # In JavaScript, a negative wheel delta number indicates upward scrolling, # but in HID, negative means downward scrolling, so we negate the value to # translate from JavaScript semantics to HID semantics. return vertical_wheel_delta * -1
- CCharles Hague @cghague2025-03-25 04:17:05.321Z
Hi Ather, thanks for sharing your incredible modifications! We have some internal tickets open for investigating relative mouse mode, and I'd like to share these changes with our development team there, as they might find the approach interesting. Would that be okay with you?
- AAther @ather
Sure, I am happy to be of help in this.