Skip to content

Commit

Permalink
Add code for mouse support in NeXT keyboard
Browse files Browse the repository at this point in the history
  • Loading branch information
jepler committed Dec 21, 2022
1 parent f6a77d2 commit b580773
Showing 1 changed file with 63 additions and 4 deletions.
67 changes: 63 additions & 4 deletions CircuitPython_NeXT_Keyboard_RP2040/code.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
from adafruit_hid.consumer_control import ConsumerControl
from adafruit_hid.keyboard import Keyboard
from adafruit_hid.keyboard import Keycode
from adafruit_hid.mouse import Mouse
from adafruit_pioasm import Program
from adafruit_ticks import ticks_add, ticks_less, ticks_ms
from next_keycode import (
Expand All @@ -21,6 +22,14 @@
shift_modifiers,
)


# Compared to a modern mouse, the DPI of the NeXT mouse is low. Increasing this
# number makes the pointer move further faster, but it also makes moves chunky.
# Customize this number according to the trade-off you want, but also check
# whether your operating system can assign a higher "sensitivity" or
# "acceleration" for the mouse.
MOUSE_SCALE = 8

# Customize the power key's keycode. You can change it to `Keycode.POWER` if
# you really want to accidentally power off your computer!
POWER_KEY_SENDS = Keycode.F1
Expand Down Expand Up @@ -79,7 +88,8 @@ def set_leds(i):
return pack_message_str(f"0000000001110{i:02b}0000000")


QUERY = pack_message_str("000001000", 1)
QUERY = pack_message_str("000001000", True)
MOUSEQUERY = pack_message_str("010001000", True)
RESET = pack_message_str("0111101111110000000000")

BIT_BREAK = 1 << 11
Expand All @@ -94,12 +104,19 @@ def is_mod_report(report):
return not bool(report & BIT_MOD)


def extract_bits(report, *positions):
result = 0
for p in positions:
result = (result << 1)
if report & (1 << p):
result |= 1
#result = (result << 1) | bool(report & (1<<p))
return result

# keycode bits are backwards compared to other information sources
# (bit 0 is first)
def keycode(report):
b = f"{report >> 12:07b}"
b = "".join(reversed(b))
return int(b, 2)
return extract_bits(report, 12, 13, 14, 15, 16, 17, 18)


def modifiers(report):
Expand All @@ -122,11 +139,18 @@ def modifiers(report):
)


def signfix(num, sign_pos):
"""Fix a signed number if the bit with weight `sign_pos` is actually the sign bit"""
if num & sign_pos:
return num - 2*sign_pos
return num

class KeyboardHandler:
def __init__(self):
self.old_modifiers = 0
self.cc = ConsumerControl(usb_hid.devices)
self.kbd = Keyboard(usb_hid.devices)
self.mouse = Mouse(usb_hid.devices)

def set_key_state(self, key, state):
if state:
Expand All @@ -144,6 +168,27 @@ def set_key_state(self, key, state):
else:
self.kbd.release(key)

def handle_mouse_report(self, report):
if report == 1536: # the "nothing happened" report
return

dx = extract_bits(report, 11,12,13,14,15,16,17)
dx = -signfix(dx, 64)
dy = extract_bits(report, 0,1,2,3,4,5,6)
dy = -signfix(dy, 64)
b1 = not extract_bits(report, 18)
b2 = not extract_bits(report, 7)

self.mouse.report[0] = (
Mouse.MIDDLE_BUTTON if (b1 and b2) else
Mouse.LEFT_BUTTON if b1 else
Mouse.RIGHT_BUTTON if b2
else 0)
if dx or dy:
self.mouse.move(dx * MOUSE_SCALE, dy * MOUSE_SCALE)
else:
self.mouse._send_no_move() # pylint: disable=protected-access

def handle_report(self, report_value):
if report_value == 1536: # the "nothing happened" report
return
Expand Down Expand Up @@ -205,5 +250,19 @@ def handle_report(self, report_value):
sm.restart()
sm.write(RESET)
time.sleep(0.1)

sm.write(MOUSEQUERY)
deadline = ticks_add(ticks_ms(), 100)
while ticks_less(ticks_ms(), deadline):
if sm.in_waiting:
sm.readinto(recv_buf)
value = recv_buf[0]
handler.handle_mouse_report(value)
break
else:
print("keyboard did not respond - resetting")
sm.restart()
sm.write(RESET)
time.sleep(0.1)
finally: # Release all keys before e.g., code is reloaded
handler.kbd.release_all()

0 comments on commit b580773

Please sign in to comment.