-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDIPPID_pyqtnode.py
188 lines (141 loc) · 5.5 KB
/
DIPPID_pyqtnode.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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
#!/usr/bin/env python3
# coding: utf-8
# -*- coding: utf-8 -*-
import time
from pyqtgraph.flowchart import Flowchart, Node
from pyqtgraph.flowchart.library.common import CtrlNode
import pyqtgraph.flowchart.library as fclib
from pyqtgraph.Qt import QtGui, QtCore
import pyqtgraph as pg
import numpy as np
from DIPPID import SensorUDP
import sys
class BufferNode(Node):
"""
Buffers the last n samples provided on input and provides them as a list of
length n on output.
A spinbox widget allows for setting the size of the buffer.
Default size is 32 samples.
"""
nodeName = "Buffer"
def __init__(self, name):
terminals = {
'dataIn': dict(io='in'),
'dataOut': dict(io='out'),
}
self.buffer_size = 32
self._buffer = np.array([])
Node.__init__(self, name, terminals=terminals)
def process(self, **kwds):
self._buffer = np.append(self._buffer, kwds['dataIn'])[-self.buffer_size:]
return {'dataOut': self._buffer}
fclib.registerNodeType(BufferNode, [('Data',)])
class DIPPIDNode(Node):
"""
Outputs sensor data from DIPPID supported hardware.
Supported sensors: accelerometer (3 axis)
Text input box allows for setting a Bluetooth MAC address or Port.
Pressing the "connect" button tries connecting to the DIPPID device.
Update rate can be changed via a spinbox widget. Setting it to "0"
activates callbacks every time a new sensor value arrives (which is
quite often -> performance hit)
"""
nodeName = "DIPPID"
def __init__(self, name):
terminals = {
'accelX': dict(io='out'),
'accelY': dict(io='out'),
'accelZ': dict(io='out'),
}
self.dippid = None
self._acc_vals = []
self._init_ui()
self.update_timer = QtCore.QTimer()
self.update_timer.timeout.connect(self.update_all_sensors)
Node.__init__(self, name, terminals=terminals)
def _init_ui(self):
self.ui = QtGui.QWidget()
self.layout = QtGui.QGridLayout()
label = QtGui.QLabel("Port:")
self.layout.addWidget(label)
self.text = QtGui.QLineEdit()
self.addr = "5700"
self.text.setText(self.addr)
self.layout.addWidget(self.text)
label2 = QtGui.QLabel("Update rate (Hz)")
self.layout.addWidget(label2)
self.update_rate_input = QtGui.QSpinBox()
self.update_rate_input.setMinimum(0)
self.update_rate_input.setMaximum(60)
self.update_rate_input.setValue(20)
self.update_rate_input.valueChanged.connect(self.set_update_rate)
self.layout.addWidget(self.update_rate_input)
self.connect_button = QtGui.QPushButton("connect")
self.connect_button.clicked.connect(self.connect_device)
self.layout.addWidget(self.connect_button)
self.ui.setLayout(self.layout)
def update_all_sensors(self):
if self.dippid is None or not self.dippid.has_capability('accelerometer'):
return
v = self.dippid.get_value('accelerometer')
self._acc_vals = [v['x'], v['y'], v['z']]
self.update()
def update_accel(self, acc_vals):
if not self.dippid.has_capability('accelerometer'):
return
self._acc_vals = [acc_vals['x'], acc_vals['y'], acc_vals['z']]
self.update()
def ctrlWidget(self):
return self.ui
def connect_device(self):
if self.connect_button.text() != "connect" and self.connect_button.text() != "try again":
return
self.connect_button.setText("connecting...")
self.dippid = SensorUDP(int(self.text.text().strip()))
if self.dippid is None:
self.connect_button.setText("try again")
return
self.connect_button.setText("connected")
self.set_update_rate(self.update_rate_input.value())
self.connect_button.setEnabled(False)
def set_update_rate(self, rate):
if self.dippid is None:
return
self.dippid.unregister_callback('accelerometer', self.update_accel)
if rate == 0:
self.update_timer.stop()
else:
self.update_timer.start(1000 / rate)
def callback(self):
time.sleep(10)
self.get_sensor()
# method for returning the sensor
def get_sensor(self):
return self.dippid
def process(self, **kwdargs):
return {'accelX': np.array([self._acc_vals[0]]), 'accelY': np.array([self._acc_vals[1]]), 'accelZ': np.array([self._acc_vals[2]])}
fclib.registerNodeType(DIPPIDNode, [('Sensor',)])
if __name__ == '__main__':
app = QtGui.QApplication([])
win = QtGui.QMainWindow()
win.setWindowTitle('DIPPIDNode demo')
cw = QtGui.QWidget()
win.setCentralWidget(cw)
layout = QtGui.QGridLayout()
cw.setLayout(layout)
# Create an empty flowchart with a single input and output
fc = Flowchart(terminals={})
w = fc.widget()
layout.addWidget(fc.widget(), 0, 0, 2, 1)
pw1 = pg.PlotWidget()
layout.addWidget(pw1, 0, 1)
pw1.setYRange(0, 1)
pw1Node = fc.createNode('PlotWidget', pos=(0, -150))
pw1Node.setPlot(pw1)
dippidNode = fc.createNode("DIPPID", pos=(0, 0))
bufferNode = fc.createNode('Buffer', pos=(150, 0))
fc.connectTerminals(dippidNode['accelX'], bufferNode['dataIn'])
fc.connectTerminals(bufferNode['dataOut'], pw1Node['In'])
win.show()
if (sys.flags.interactive != 1) or not hasattr(QtCore, 'PYQT_VERSION'):
sys.exit(QtGui.QApplication.instance().exec_())