forked from DiscreteTom/stickdeck
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdeck.py
111 lines (96 loc) · 3.12 KB
/
deck.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
import bluetooth
from pyjoystick.sdl2 import Key, Joystick, run_event_loop
import protocol
import datetime
import sys
# configs
port = 1
# how many times per second to send data, set this to throttle frequency to optimize latency.
# without this the controller will send data in 500+ fps on my machine, congesting the network.
fps = 200
# globals
server_sock = bluetooth.BluetoothSocket(bluetooth.RFCOMM)
client_sock = None
state = protocol.State()
last_send_timestamp = 0
interval = 1 / fps
def print_add(joy):
print('Added', joy)
def print_remove(joy):
print('Removed', joy)
def key_received(key: Key):
if client_sock == None:
return
if key.keytype == Key.KeyTypes.BUTTON:
# print(key)
if key.number == 0:
state.button_A = key.value
elif key.number == 1:
state.button_B = key.value
elif key.number == 2:
state.button_X = key.value
elif key.number == 3:
state.button_Y = key.value
elif key.number == 4:
state.button_L_SHOULDER = key.value
elif key.number == 5:
state.button_R_SHOULDER = key.value
elif key.number == 6:
state.button_BACK = key.value
elif key.number == 7:
state.button_START = key.value
# elif key.number == 8: # no this button on steam deck
# pass
elif key.number == 9:
state.button_L_THUMB = key.value
elif key.number == 10:
state.button_R_THUMB = key.value
elif key.keytype == Key.KeyTypes.AXIS:
if key.number == 0:
state.left_joystick_x = int(
(key.value + 1) / 2 * 65535) # [-1, 1] => [0, 65535]
elif key.number == 1:
state.left_joystick_y = int(
(-key.value + 1) / 2 * 65535) # reverse axis-Y value
elif key.number == 2:
state.left_trigger = int(key.value * 255) # [0, 1] => [0, 255]
elif key.number == 3:
state.right_joystick_x = int(
(key.value + 1) / 2 * 65535) # [-1, 1] => [0, 65535]
elif key.number == 4:
state.right_joystick_y = int(
(-key.value + 1) / 2 * 65535) # reverse axis-Y value
elif key.number == 5:
state.right_trigger = int(key.value * 255) # [0, 1] => [0, 255]
elif key.keytype == Key.KeyTypes.HAT:
state.button_DPAD = key.value
# send data if enough time has passed
global last_send_timestamp
if datetime.datetime.now().timestamp() - last_send_timestamp > interval:
try:
client_sock.send(state.encode())
except:
# if the receiver is down, exit
sys.exit(0)
last_send_timestamp = datetime.datetime.now().timestamp()
# listen
server_sock.bind(("", port))
server_sock.listen(1)
print(f'listening at {bluetooth.read_local_bdaddr()[0]} with port {port}')
# get client
try:
client_sock, address = server_sock.accept()
print("Accepted connection from ", address)
except:
sys.exit(0)
# currently set_packet_timeout is not working
# https://github.com/pybluez/pybluez/issues/465
# bluetooth.set_packet_timeout(
# address[0], 1000 / fps) # drop packets if they are too old to ensure fps
# main loop
try:
run_event_loop(print_add, print_remove, key_received)
except KeyboardInterrupt:
# gracefully exit on ctrl-c
client_sock.close()
server_sock.close()