forked from real-stanford/diffusion_policy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathspacemouse.py
108 lines (92 loc) · 3.07 KB
/
spacemouse.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
from spnav import spnav_open, spnav_poll_event, spnav_close, SpnavMotionEvent, SpnavButtonEvent
from threading import Thread, Event
from collections import defaultdict
import numpy as np
import time
class Spacemouse(Thread):
def __init__(self, max_value=500, deadzone=(0,0,0,0,0,0), dtype=np.float32):
"""
Continuously listen to 3D connection space naviagtor events
and update the latest state.
max_value: {300, 500} 300 for wired version and 500 for wireless
deadzone: [0,1], number or tuple, axis with value lower than this value will stay at 0
front
z
^ _
| (O) space mouse
|
*----->x right
y
"""
if np.issubdtype(type(deadzone), np.number):
deadzone = np.full(6, fill_value=deadzone, dtype=dtype)
else:
deadzone = np.array(deadzone, dtype=dtype)
assert (deadzone >= 0).all()
super().__init__()
self.stop_event = Event()
self.max_value = max_value
self.dtype = dtype
self.deadzone = deadzone
self.motion_event = SpnavMotionEvent([0,0,0], [0,0,0], 0)
self.button_state = defaultdict(lambda: False)
self.tx_zup_spnav = np.array([
[0,0,-1],
[1,0,0],
[0,1,0]
], dtype=dtype)
def get_motion_state(self):
me = self.motion_event
state = np.array(me.translation + me.rotation,
dtype=self.dtype) / self.max_value
is_dead = (-self.deadzone < state) & (state < self.deadzone)
state[is_dead] = 0
return state
def get_motion_state_transformed(self):
"""
Return in right-handed coordinate
z
*------>y right
| _
| (O) space mouse
v
x
back
"""
state = self.get_motion_state()
tf_state = np.zeros_like(state)
tf_state[:3] = self.tx_zup_spnav @ state[:3]
tf_state[3:] = self.tx_zup_spnav @ state[3:]
return tf_state
def is_button_pressed(self, button_id):
return self.button_state[button_id]
def stop(self):
self.stop_event.set()
self.join()
def __enter__(self):
self.start()
return self
def __exit__(self, exc_type, exc_val, exc_tb):
self.stop()
def run(self):
spnav_open()
try:
while not self.stop_event.is_set():
event = spnav_poll_event()
if isinstance(event, SpnavMotionEvent):
self.motion_event = event
elif isinstance(event, SpnavButtonEvent):
self.button_state[event.bnum] = event.press
else:
time.sleep(1/200)
finally:
spnav_close()
def test():
with Spacemouse(deadzone=0.3) as sm:
for i in range(2000):
# print(sm.get_motion_state())
print(sm.get_motion_state_transformed())
print(sm.is_button_pressed(0))
time.sleep(1/100)
if __name__ == '__main__':
test()