add ps4 controller

This commit is contained in:
2022-09-21 23:25:38 +02:00
parent 5a41c6454a
commit 0f6d3fca5e
7 changed files with 270 additions and 14 deletions

View File

@@ -9,17 +9,21 @@ import adsk.fusion
import adsk.cam
import traceback
import json
from ...lib import hid
CMD_NAME = 'View Controller backend'
app = adsk.core.Application.get()
ui = app.userInterface
myThread = None
stopFlag = None
cameraViewEvent = 'CameraViewEventId'
updateConstantsEvent = "UpdateConstantsEventId"
customEvents = []
local_handlers = []
l_handlers = []
X_ROTATION = 0
Y_ROTATION = 0
@@ -44,6 +48,7 @@ class UpdateConstantsEventHandler(adsk.core.CustomEventHandler):
X_PAN = float(eventArgs["X_pan"])
Y_PAN = float(eventArgs["Y_pan"])
Z_PAN = float(eventArgs["Z_pan"])
myThread.start()
# The event handler that responds to the custom event being fired.
@@ -93,9 +98,9 @@ class CameraViewEventHandler(adsk.core.CustomEventHandler):
eye.transformBy(rotate_matrix)
up.transformBy(rotate_matrix)
X_pan = float(eventArgs.get('X_pan', 0)) * X_pan
Y_pan = float(eventArgs.get('Y_pan', 0)) * Y_pan
Z_pan = float(eventArgs.get('Z_pan', 0)) * Z_pan
X_pan = float(eventArgs.get('X_pan', 0)) * X_PAN
Y_pan = float(eventArgs.get('Y_pan', 0)) * Y_PAN
Z_pan = float(eventArgs.get('Z_pan', 0)) * Z_PAN
if any([X_pan, Z_pan]):
move_vector = adsk.core.Vector3D.create(X_pan, 0, Z_pan)
move_vector.transformBy(rotate_matrix)
@@ -105,7 +110,7 @@ class CameraViewEventHandler(adsk.core.CustomEventHandler):
if Y_pan:
zoom = zoom + Y_pan * 10
camera.isSmoothTransition = True
camera.isSmoothTransition = False
camera.eye = eye
camera.upVector = up
camera.target = target
@@ -128,13 +133,30 @@ class ControllerThread(threading.Thread):
self.stopped = event
def run(self):
futil.log(f'{CMD_NAME} ControllerThread started')
gamepad = hid.Device(vid=1356, pid=1476)
gamepad.nonblocking = True
# Every five seconds fire a custom event, passing a random number.
while not self.stopped.wait(2):
futil.log(f'{CMD_NAME} Thread start sending data')
args = {'X_rotation': 0.0, 'Y_rotation': 0.0, 'Z_rotation': 0.0,
'X_pan': 0.0, 'Y_pan': 0.0, 'Z_pan': 0.0}
while not self.stopped.wait(0.01):
# futil.log(f'{CMD_NAME} Thread start sending data')
report = list(gamepad.read(128))
args = {
'X_rotation': 0.0,
'Y_rotation': self.convert(report[4], 10),
'Z_rotation': self.convert(report[3], 10),
'X_pan': self.convert(report[1], 10),
'Y_pan': self.convert(report[2], 10),
'Z_pan': 0.0
}
app.fireCustomEvent(cameraViewEvent, json.dumps(args))
def convert(self, input, dedzone) -> float:
normal: float = input - 127
if abs(normal) < dedzone:
return 0.0
normal = normal / 128
return normal
def start():
@@ -144,26 +166,25 @@ def start():
customEvents.append(app.registerCustomEvent(cameraViewEvent))
onCameraViewEvent = CameraViewEventHandler()
customEvents[0].add(onCameraViewEvent)
local_handlers.append(onCameraViewEvent)
l_handlers.append(onCameraViewEvent)
futil.log(f'{CMD_NAME} CameraViewEvent register')
customEvents.append(app.registerCustomEvent(updateConstantsEvent))
onUpadateConstantEvent = UpdateConstantsEventHandler()
customEvents[1].add(onUpadateConstantEvent)
local_handlers.append(onUpadateConstantEvent)
l_handlers.append(onUpadateConstantEvent)
futil.log(f'{CMD_NAME} UpdateConstantsEvent register')
# Create a new thread for the other processing.
global stopFlag
global stopFlag, myThread
stopFlag = threading.Event()
myThread = ControllerThread(stopFlag)
# myThread.start()
futil.log(f'{CMD_NAME} ControllerThread started')
def stop():
try:
for event, handler in zip(customEvents, local_handlers):
for event, handler in zip(customEvents, l_handlers):
event.remove(handler)
# if local_handlers.count:
# customEvent.remove(local_handlers[0])

View File

@@ -0,0 +1,235 @@
import os
import ctypes
import atexit
__all__ = ['HIDException', 'DeviceInfo', 'Device', 'enumerate']
dir = os.path.dirname(__file__).replace("/", "\\")
hidapi = None
library_paths = (
'libhidapi-hidraw.so',
'libhidapi-hidraw.so.0',
'libhidapi-libusb.so',
'libhidapi-libusb.so.0',
'libhidapi-iohidmanager.so',
'libhidapi-iohidmanager.so.0',
'libhidapi.dylib',
'hidapi.dll',
'libhidapi-0.dll'
)
for lib in library_paths:
try:
hidapi = ctypes.cdll.LoadLibrary(os.path.join(dir, lib))
break
except OSError:
pass
else:
error = "Unable to load any of the following libraries:{}"\
.format(' '.join(library_paths))
raise ImportError(error)
hidapi.hid_init()
atexit.register(hidapi.hid_exit)
class HIDException(Exception):
pass
class DeviceInfo(ctypes.Structure):
def as_dict(self):
ret = {}
for name, type in self._fields_:
if name == 'next':
continue
ret[name] = getattr(self, name, None)
return ret
DeviceInfo._fields_ = [
('path', ctypes.c_char_p),
('vendor_id', ctypes.c_ushort),
('product_id', ctypes.c_ushort),
('serial_number', ctypes.c_wchar_p),
('release_number', ctypes.c_ushort),
('manufacturer_string', ctypes.c_wchar_p),
('product_string', ctypes.c_wchar_p),
('usage_page', ctypes.c_ushort),
('usage', ctypes.c_ushort),
('interface_number', ctypes.c_int),
('next', ctypes.POINTER(DeviceInfo)),
]
hidapi.hid_init.argtypes = []
hidapi.hid_init.restype = ctypes.c_int
hidapi.hid_exit.argtypes = []
hidapi.hid_exit.restype = ctypes.c_int
hidapi.hid_enumerate.argtypes = [ctypes.c_ushort, ctypes.c_ushort]
hidapi.hid_enumerate.restype = ctypes.POINTER(DeviceInfo)
hidapi.hid_free_enumeration.argtypes = [ctypes.POINTER(DeviceInfo)]
hidapi.hid_free_enumeration.restype = None
hidapi.hid_open.argtypes = [ctypes.c_ushort, ctypes.c_ushort, ctypes.c_wchar_p]
hidapi.hid_open.restype = ctypes.c_void_p
hidapi.hid_open_path.argtypes = [ctypes.c_char_p]
hidapi.hid_open_path.restype = ctypes.c_void_p
hidapi.hid_write.argtypes = [ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t]
hidapi.hid_write.restype = ctypes.c_int
hidapi.hid_read_timeout.argtypes = [
ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t, ctypes.c_int]
hidapi.hid_read_timeout.restype = ctypes.c_int
hidapi.hid_read.argtypes = [ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t]
hidapi.hid_read.restype = ctypes.c_int
hidapi.hid_get_input_report.argtypes = [
ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t]
hidapi.hid_get_input_report.restype = ctypes.c_int
hidapi.hid_set_nonblocking.argtypes = [ctypes.c_void_p, ctypes.c_int]
hidapi.hid_set_nonblocking.restype = ctypes.c_int
hidapi.hid_send_feature_report.argtypes = [
ctypes.c_void_p, ctypes.c_char_p, ctypes.c_int]
hidapi.hid_send_feature_report.restype = ctypes.c_int
hidapi.hid_get_feature_report.argtypes = [
ctypes.c_void_p, ctypes.c_char_p, ctypes.c_size_t]
hidapi.hid_get_feature_report.restype = ctypes.c_int
hidapi.hid_close.argtypes = [ctypes.c_void_p]
hidapi.hid_close.restype = None
hidapi.hid_get_manufacturer_string.argtypes = [
ctypes.c_void_p, ctypes.c_wchar_p, ctypes.c_size_t]
hidapi.hid_get_manufacturer_string.restype = ctypes.c_int
hidapi.hid_get_product_string.argtypes = [
ctypes.c_void_p, ctypes.c_wchar_p, ctypes.c_size_t]
hidapi.hid_get_product_string.restype = ctypes.c_int
hidapi.hid_get_serial_number_string.argtypes = [
ctypes.c_void_p, ctypes.c_wchar_p, ctypes.c_size_t]
hidapi.hid_get_serial_number_string.restype = ctypes.c_int
hidapi.hid_get_indexed_string.argtypes = [
ctypes.c_void_p, ctypes.c_int, ctypes.c_wchar_p, ctypes.c_size_t]
hidapi.hid_get_indexed_string.restype = ctypes.c_int
hidapi.hid_error.argtypes = [ctypes.c_void_p]
hidapi.hid_error.restype = ctypes.c_wchar_p
def enumerate(vid=0, pid=0):
ret = []
info = hidapi.hid_enumerate(vid, pid)
c = info
while c:
ret.append(c.contents.as_dict())
c = c.contents.next
hidapi.hid_free_enumeration(info)
return ret
class Device(object):
def __init__(self, vid=None, pid=None, serial=None, path=None):
if path:
self.__dev = hidapi.hid_open_path(path)
elif serial:
serial = ctypes.create_unicode_buffer(serial)
self.__dev = hidapi.hid_open(vid, pid, serial)
elif vid and pid:
self.__dev = hidapi.hid_open(vid, pid, None)
else:
raise ValueError('specify vid/pid or path')
if not self.__dev:
raise HIDException('unable to open device')
def __enter__(self):
return self
def __exit__(self, exc_type, exc_value, exc_traceback):
self.close()
def __hidcall(self, function, *args, **kwargs):
if not self.__dev:
raise HIDException('device closed')
ret = function(*args, **kwargs)
if ret == -1:
err = hidapi.hid_error(self.__dev)
raise HIDException(err)
return ret
def __readstring(self, function, max_length=255):
buf = ctypes.create_unicode_buffer(max_length)
self.__hidcall(function, self.__dev, buf, max_length)
return buf.value
def write(self, data):
return self.__hidcall(hidapi.hid_write, self.__dev, data, len(data))
def read(self, size, timeout=None):
data = ctypes.create_string_buffer(size)
if timeout is None:
size = self.__hidcall(hidapi.hid_read, self.__dev, data, size)
else:
size = self.__hidcall(
hidapi.hid_read_timeout, self.__dev, data, size, timeout)
return data.raw[:size]
def get_input_report(self, report_id, size):
data = ctypes.create_string_buffer(size)
# Pass the id of the report to be read.
data[0] = bytearray((report_id,))
size = self.__hidcall(
hidapi.hid_get_input_report, self.__dev, data, size)
return data.raw[:size]
def send_feature_report(self, data):
return self.__hidcall(hidapi.hid_send_feature_report,
self.__dev, data, len(data))
def get_feature_report(self, report_id, size):
data = ctypes.create_string_buffer(size)
# Pass the id of the report to be read.
data[0] = bytearray((report_id,))
size = self.__hidcall(
hidapi.hid_get_feature_report, self.__dev, data, size)
return data.raw[:size]
def close(self):
if self.__dev:
hidapi.hid_close(self.__dev)
self.__dev = None
@property
def nonblocking(self):
return getattr(self, '_nonblocking', 0)
@nonblocking.setter
def nonblocking(self, value):
self.__hidcall(hidapi.hid_set_nonblocking, self.__dev, value)
setattr(self, '_nonblocking', value)
@property
def manufacturer(self):
return self.__readstring(hidapi.hid_get_manufacturer_string)
@property
def product(self):
return self.__readstring(hidapi.hid_get_product_string)
@property
def serial(self):
return self.__readstring(hidapi.hid_get_serial_number_string)
def get_indexed_string(self, index, max_length=255):
buf = ctypes.create_unicode_buffer(max_length)
self.__hidcall(hidapi.hid_get_indexed_string,
self.__dev, index, buf, max_length)
return buf.value

Binary file not shown.

Binary file not shown.

Binary file not shown.