Adafruit-Retrogame/joyBonnetAsMouse.py

178 lines
5.9 KiB
Python

#!/usr/bin/python
# Somewhat minimal Adafruit Joy Bonnet handler. Runs in background,
# translates inputs from buttons and ADC joystick to virtual USB keyboard
# events.
# Prerequisites:
# sudo apt-get install python-pip python-smbus python-dev
# sudo pip install evdev
# Be sure to enable I2C via raspi-config. Also, udev rules will need to
# be set up per retrogame directions.
# Credit to Pimoroni for Picade HAT scripts as starting point.
import time
import signal
import os
import sys
from datetime import datetime
try:
from evdev import uinput, UInput, ecodes as e
except ImportError:
exit("This library requires the evdev module\nInstall with: sudo pip install evdev")
try:
import RPi.GPIO as gpio
except ImportError:
exit("This library requires the RPi.GPIO module\nInstall with: sudo pip install RPi.GPIO")
try:
from smbus import SMBus
except ImportError:
exit("This library requires the smbus module\nInstall with: sudo pip install smbus")
DEBUG = False
BOUNCE_TIME = 0.01 # Debounce time in seconds
BUTTON_A = 12
BUTTON_B = 6
BUTTON_X = 16
BUTTON_Y = 13
SELECT = 20
START = 26
PLAYER1 = 23
PLAYER2 = 22
BUTTONS = [BUTTON_A, BUTTON_B, BUTTON_X, BUTTON_Y, SELECT, START, PLAYER1, PLAYER2]
MOUSE_SENSITIVITY = 4 # 0-10
MOUSE_DEADZONE = 40 # Values under this are zeroed
KEYS= { # EDIT KEYCODES IN THIS TABLE TO YOUR PREFERENCES:
# See /usr/include/linux/input.h for keycode names
# Keyboard Bonnet EmulationStation
BUTTON_A: e.KEY_RIGHT, # 'A' button as mouse click
BUTTON_B: e.KEY_DOWN, # 'B' button as right click
BUTTON_X: e.KEY_UP, # 'X' button
BUTTON_Y: e.KEY_LEFT, # 'Y' button
SELECT: e.BTN_RIGHT, # 'Select' button
START: e.BTN_MOUSE, # 'Start' button
PLAYER1: e.KEY_PAGEUP, # '#1' button
PLAYER2: e.KEY_PAGEDOWN, # '#2' button
}
###################################### ADS1015 microdriver #################################
# Register and other configuration values:
ADS1x15_DEFAULT_ADDRESS = 0x48
ADS1x15_POINTER_CONVERSION = 0x00
ADS1x15_POINTER_CONFIG = 0x01
ADS1015_REG_CONFIG_CQUE_NONE = 0x0003 # Disable the comparator and put ALERT/RDY in high state (default)
ADS1015_REG_CONFIG_CLAT_NONLAT = 0x0000 # Non-latching comparator (default)
ADS1015_REG_CONFIG_CPOL_ACTVLOW = 0x0000 # ALERT/RDY pin is low when active (default)
ADS1015_REG_CONFIG_CMODE_TRAD = 0x0000 # Traditional comparator with hysteresis (default)
ADS1015_REG_CONFIG_DR_1600SPS = 0x0080 # 1600 samples per second (default)
ADS1015_REG_CONFIG_MODE_SINGLE = 0x0100 # Power-down single-shot mode (default)
ADS1015_REG_CONFIG_GAIN_ONE = 0x0200 # gain of 1
ADS1015_REG_CONFIG_MUX_SINGLE_0 = 0x4000 # channel 0
ADS1015_REG_CONFIG_MUX_SINGLE_1 = 0x5000 # channel 1
ADS1015_REG_CONFIG_MUX_SINGLE_2 = 0x6000 # channel 2
ADS1015_REG_CONFIG_MUX_SINGLE_3 = 0x7000 # channel 3
ADS1015_REG_CONFIG_OS_SINGLE = 0x8000 # start a single conversion
ADS1015_REG_CONFIG_CHANNELS = (ADS1015_REG_CONFIG_MUX_SINGLE_0, ADS1015_REG_CONFIG_MUX_SINGLE_1,
ADS1015_REG_CONFIG_MUX_SINGLE_2, ADS1015_REG_CONFIG_MUX_SINGLE_3)
def ads_read(channel):
#configdata = bus.read_i2c_block_data(ADS1x15_DEFAULT_ADDRESS, ADS1x15_POINTER_CONFIG, 2)
#print("Getting config byte = 0x%02X%02X" % (configdata[0], configdata[1]))
configword = ADS1015_REG_CONFIG_CQUE_NONE | ADS1015_REG_CONFIG_CLAT_NONLAT | ADS1015_REG_CONFIG_CPOL_ACTVLOW | ADS1015_REG_CONFIG_CMODE_TRAD | ADS1015_REG_CONFIG_DR_1600SPS | ADS1015_REG_CONFIG_MODE_SINGLE | ADS1015_REG_CONFIG_GAIN_ONE | ADS1015_REG_CONFIG_CHANNELS[channel] | ADS1015_REG_CONFIG_OS_SINGLE
configdata = [configword >> 8, configword & 0xFF]
#print("Setting config byte = 0x%02X%02X" % (configdata[0], configdata[1]))
bus.write_i2c_block_data(ADS1x15_DEFAULT_ADDRESS, ADS1x15_POINTER_CONFIG, configdata)
configdata = bus.read_i2c_block_data(ADS1x15_DEFAULT_ADDRESS, ADS1x15_POINTER_CONFIG, 2)
#print("Getting config byte = 0x%02X%02X" % (configdata[0], configdata[1]))
while True:
try:
configdata = bus.read_i2c_block_data(ADS1x15_DEFAULT_ADDRESS, ADS1x15_POINTER_CONFIG, 2)
#print("Getting config byte = 0x%02X%02X" % (configdata[0], configdata[1]))
if (configdata[0] & 0x80):
break
except:
pass
# read data out!
analogdata = bus.read_i2c_block_data(ADS1x15_DEFAULT_ADDRESS, ADS1x15_POINTER_CONVERSION, 2)
#print(analogdata),
retval = (analogdata[0] << 8) | analogdata[1]
retval /= 16
#print("-> %d" %retval)
return retval
######################## main program
os.system("sudo modprobe uinput")
bus = SMBus(1)
# GPIO init
gpio.setwarnings(False)
gpio.setmode(gpio.BCM)
gpio.setup(BUTTONS, gpio.IN, pull_up_down=gpio.PUD_UP)
try:
caps= {e.EV_KEY: KEYS.values(),
e.EV_REL: [e.REL_X, e.REL_Y],
}
ui = UInput(caps, name="retrogame", bustype=e.BUS_USB)
except uinput.UInputError as e:
sys.stdout.write(e.message)
sys.stdout.write("Have you tried running as root? sudo {}".format(sys.argv[0]))
sys.exit(0)
def log(msg):
sys.stdout.write(str(datetime.now()))
sys.stdout.write(": ")
sys.stdout.write(msg)
sys.stdout.write("\n")
sys.stdout.flush()
def handle_button(pin):
key = KEYS[pin]
time.sleep(BOUNCE_TIME)
state = 0 if gpio.input(pin) else 1
ui.write(e.EV_KEY, key, state)
ui.syn()
if DEBUG:
log("Pin: {}, KeyCode: {}, Event: {}".format(pin, key, 'press' if state else 'release'))
for button in BUTTONS:
gpio.add_event_detect(button, gpio.BOTH, callback=handle_button, bouncetime=1)
while True:
try:
y = 800 - ads_read(0)
x = ads_read(1) - 800
except IOError:
continue
# print("(%d , %d)" % (x, y))
if abs(x) < MOUSE_DEADZONE:
x=0
if abs(y) < MOUSE_DEADZONE:
y=0
x = x >> (10 - MOUSE_SENSITIVITY)
y = y >> (10 - MOUSE_SENSITIVITY)
ui.write(e.EV_REL, e.REL_X, x)
ui.write(e.EV_REL, e.REL_Y, -y)
ui.syn()
time.sleep(0.01)