Go2Py/Go2Py/joy.py

225 lines
7.0 KiB
Python

import threading
import time
from time import sleep
import numpy as np
import pygame
from dataclasses import dataclass
from pygame.locals import (
JOYAXISMOTION,
JOYBUTTONDOWN,
JOYBUTTONUP,
K_ESCAPE,
KEYDOWN,
NOEVENT,
QUIT,
K_q,
)
@dataclass
class xKeySwitch:
R1: int
L1: int
start: int
select: int
R2: int
L2: int
F1: int
F2: int
A: int
B: int
X: bool
Y: int
up: int
right: int
down: int
left: int
@dataclass
class xRockerBtn:
head: list
btn: xKeySwitch
lx: float
rx: float
ry: float
L2: float
ly: float
class PyGameJoyManager:
def __init__(self, user_callback=None):
self.dt = 0.01 # Sampling frequence of the joystick
self.running = False
self.joy_thread_handle = threading.Thread(target=self.joy_thread)
# an optional callback that can be used to send the reading values to a user provided function
self.user_callback = user_callback
pygame.init()
self.offset = None
def joy_thread(self):
while self.running:
for event in [
pygame.event.wait(200),
] + pygame.event.get():
# QUIT none
# ACTIVEEVENT gain, state
# KEYDOWN unicode, key, mod
# KEYUP key, mod
# MOUSEMOTION pos, rel, buttons
# MOUSEBUTTONUP pos, button
# MOUSEBUTTONDOWN pos, button
# JOYAXISMOTION joy, axis, value
# JOYBALLMOTION joy, ball, rel
# JOYHATMOTION joy, hat, value
# JOYBUTTONUP joy, button
# JOYBUTTONDOWN joy, button
# VIDEORESIZE size, w, h
# VIDEOEXPOSE none
# USEREVENT code
if event.type == QUIT:
self.stop_daq()
elif event.type == KEYDOWN and event.key in [K_ESCAPE, K_q]:
self.stop_daq()
elif event.type == JOYAXISMOTION:
self.analog_cmd[event.axis] = event.value
elif event.type == JOYBUTTONUP:
self.digital_cmd[event.button] = 0
elif event.type == JOYBUTTONDOWN:
self.digital_cmd[event.button] = 1
if self.user_callback is not None and event.type != NOEVENT:
if self.offset is None:
self.user_callback(self.analog_cmd, self.digital_cmd)
else:
self.user_callback(
np.array(self.analog_cmd) - self.offset, self.digital_cmd
)
def start_daq(self, joy_idx):
# Get the joy object
assert (
pygame.joystick.get_count() != 0
), "No joysticks detected, you can not start the class"
assert (
pygame.joystick.get_count() >= joy_idx
), "The requested joystick ID exceeds the number of available devices"
self.joy = pygame.joystick.Joystick(joy_idx)
self.analog_cmd = [self.joy.get_axis(i) for i in range(self.joy.get_numaxes())]
self.digital_cmd = [
self.joy.get_button(i) for i in range(self.joy.get_numbuttons())
]
self.running = True
self.joy_thread_handle.start()
def stop_daq(self):
self.running = False
self.joy_thread_handle.join()
def read_raw(self):
return self.analog_cmd, self.digital_cmd
def offset_calibration(self):
analog, _ = self.read_raw()
offset = np.array(analog)
print("Put your stick at zero location and do not touch it")
sleep(1)
for i in range(2000):
sleep(0.001)
analog, _ = self.read_raw()
offset += np.array(analog)
self.offset = offset / 2000
def read_values(self):
analog, digital = self.read_raw()
if self.offset is None:
return analog, digital
else:
return analog - self.offset, digital
class Logitech3DPro:
"""
Handles operations specific to the Logitech 3D Pro joystick.
This class serves as a wrapper around PyGameJoyManager, offering methods
for decoding and interpreting joystick values specific to Logitech 3D Pro.
Attributes:
joymanager (PyGameJoyManager): Instance of PyGameJoyManager to handle
the joystick events.
joy_id (int): The identifier for the joystick.
analog_raw (list of float): The raw analog values from the joystick axes.
digital_raw (list of int): The raw digital values from the joystick buttons.
"""
def __init__(self, joy_id=0):
"""
Initializes a Logitech3DPro object.
Starts data acquisition and performs initial calibration.
Parameters:
joy_id (int): Identifier for the joystick. Default is 0.
"""
self.joymanager = PyGameJoyManager()
self.joymanager.start_daq(joy_id)
print("Calibrating joystick, please do not touch the stick and wait...")
time.sleep(2)
self.joymanager.offset_calibration()
print("Calibration finished.")
self.joy_id = joy_id
def decode(self):
"""
Decode Raw Values from Joystick.
Decodes the raw analog and digital values from the joystick. The
decoded values are stored in the object's attributes.
"""
self.analog_raw, self.digital_raw = self.joymanager.read_values()
def readAnalog(self):
"""
Reads Analog Values.
Retrieves the analog values from the joystick and presents them in a
dictionary format. Performs decoding before fetching the values.
Returns:
dict: A dictionary containing the analog axis values with keys as 'x', 'y', 'z', and 'aux'.
"""
self.decode()
data = {
"x": self.analog_raw[0],
"y": self.analog_raw[1],
"z": self.analog_raw[2],
"aux": self.analog_raw[3],
}
return data
def readDigital(self):
"""
Reads Digital Values.
Retrieves the digital button values from the joystick and presents
them in a dictionary format. Performs decoding before fetching the values.
Returns:
dict: A dictionary containing the digital button states with keys as 'shoot', '2', '3',
up to '12'.
"""
self.decode()
data = {
"shoot": self.digital_raw[0],
"2": self.digital_raw[1],
"3": self.digital_raw[2],
"4": self.digital_raw[3],
"5": self.digital_raw[4],
"6": self.digital_raw[5],
"7": self.digital_raw[6],
"8": self.digital_raw[7],
"9": self.digital_raw[8],
"10": self.digital_raw[9],
"11": self.digital_raw[10],
"12": self.digital_raw[11],
}
return data