4 from distutils.version
import LooseVersion
7 if not hasattr(sys,
'maxint'):
8 sys.maxint = sys.maxsize
11 from matplotlib.collections
import LineCollection
12 from matplotlib.collections
import PathCollection
13 from matplotlib.collections
import PolyCollection
14 from matplotlib.colors
import colorConverter
15 from matplotlib.figure
import Figure
16 import matplotlib.pyplot
as plt
17 from mpl_toolkits.mplot3d
import axes3d
18 from mpl_toolkits.mplot3d
import Axes3D
20 import python_qt_binding
21 from python_qt_binding
import loadUi
22 from python_qt_binding.QtCore
import Qt
23 from python_qt_binding.QtCore
import QTimer
24 from python_qt_binding.QtCore
import qWarning
25 from python_qt_binding.QtCore
import Slot
26 from python_qt_binding.QtGui
import QColor
27 from python_qt_binding.QtGui
import QIcon
37 if LooseVersion(python_qt_binding.QT_BINDING_VERSION).version[0] >= 5:
38 from python_qt_binding.QtWidgets
import QAction
39 from python_qt_binding.QtWidgets
import QMenu
40 from python_qt_binding.QtWidgets
import QSizePolicy
41 from python_qt_binding.QtWidgets
import QVBoxLayout
42 from python_qt_binding.QtWidgets
import QWidget
44 from matplotlib.backends.backend_qt5agg
import FigureCanvasQTAgg \
49 sys.modules[
'_thread'] = thread
50 from matplotlib.backends.backend_qt5agg
import FigureCanvasQTAgg \
53 from matplotlib.backends.backend_qt5agg \
54 import NavigationToolbar2QTAgg
as NavigationToolbar
56 from matplotlib.backends.backend_qt5agg
import NavigationToolbar2QT \
60 from python_qt_binding.QtGui
import QAction
61 from python_qt_binding.QtGui
import QMenu
62 from python_qt_binding.QtGui
import QSizePolicy
63 from python_qt_binding.QtGui
import QVBoxLayout
64 from python_qt_binding.QtGui
import QWidget
66 from matplotlib.backends.backend_qt4agg
import FigureCanvasQTAgg \
71 sys.modules[
'_thread'] = thread
72 from matplotlib.backends.backend_qt4agg
import FigureCanvasQTAgg \
75 from matplotlib.backends.backend_qt4agg \
76 import NavigationToolbar2QTAgg
as NavigationToolbar
78 from matplotlib.backends.backend_qt4agg
import NavigationToolbar2QT \
84 """Ultimately, this is a QWidget (as well as a FigureCanvasAgg, etc.).
89 self.
axes = self.figure.add_subplot(111, projection=
'3d')
92 self.
axes.set_xlabel(
't')
93 self.
axes.set_xlim3d(0, 10)
94 self.
axes.set_ylabel(
'Y')
95 self.
axes.set_ylim3d(-1, 1)
96 self.
axes.set_zlabel(
'Z')
97 self.
axes.set_zlim3d(0, 1)
99 self.figure.tight_layout()
100 self.setSizePolicy(QSizePolicy.Expanding, QSizePolicy.Expanding)
101 self.updateGeometry()
105 self.figure.tight_layout()
107 _colors = [QColor(c)
for c
in [
108 Qt.red, Qt.blue, Qt.magenta, Qt.cyan, Qt.green, Qt.darkYellow,
109 Qt.black, Qt.darkRed, Qt.gray, Qt.darkCyan]]
111 def __init__(self, parent=None, buffer_length=100, use_poly=True,
138 self.
_curves[curve_id] = [[], [], line, [
None,
None],
139 (color.red() / 255.0,
140 color.green() / 255.0,
141 color.blue() / 255.0,
147 curve_id = str(curve_id)
156 labels = list(self.
_curves.keys())
158 plt.Rectangle((0, 0), 1, 1, fc=self.
_curves[labels[i]][4])
159 for i
in range(len(labels))]
160 self.
_canvas.axes.legend(handles, labels, loc=
'upper left')
162 @Slot(str, list, list)
164 data_x, data_y, line, range_y, c = self.
_curves[curve_id]
170 self.
_curves[curve_id][0] = data_x
171 self.
_curves[curve_id][1] = data_y
176 ymin = min(ymin, range_y[0])
180 ymax = max(ymax, range_y[1])
184 self.
_canvas.axes.grid(
True, color=
'gray')
189 for curve
in self.
_curves.values():
190 data_x, _, _, range_y, c = curve
193 xmax = max(xmax, data_x[-1])
194 xmin = min(xmin, data_x[0])
199 ymin = min(range_y[0], ymin)
200 ymax = max(range_y[1], ymax)
208 self.
_canvas.axes.set_xbound(lower=xmin, upper=xmax)
209 self.
_canvas.axes.set_zbound(lower=ymin, upper=ymax)
210 self.
_canvas.axes.set_ybound(lower=0,
211 upper=len(self.
_curves.keys()))
217 colors.append(self.
_curves[curve_id][4])
219 verts.append([(xmin, ymin)] + list(zip(data_x, data_y))
222 verts.append(list(zip(data_x, data_y)))
223 line_num = len(self.
_curves.keys())
225 poly = PolyCollection(verts, facecolors=colors, closed=
False)
227 poly = LineCollection(verts, colors=colors)
230 self.
_canvas.axes.add_collection3d(poly,
231 zs=range(line_num), zdir=
'y')
238 super(Plot3D, self).
__init__(context)
239 self.setObjectName(
'Plot3D')
242 initial_topics=self.
_args.topics,
243 start_paused=self.
_args.start_paused,
244 buffer_length=self.
_args.buffer,
245 use_poly=
not self.
_args.show_line,
246 no_legend=self.
_args.no_legend)
247 context.add_widget(self.
_widget)
250 parser = argparse.ArgumentParser(prog=
'rqt_3d_plot', add_help=
False)
251 Plot3D.add_arguments(parser)
252 args = parser.parse_args(argv)
254 for t
in args.topics:
258 for sub_t
in [x
for x
in t.split(
',')
if x]:
261 base = sub_t[:sub_t.find(
':')]
264 c_topics.append(base)
267 "%s must contain a topic and field name" % sub_t)
268 base = base[:base.rfind(
'/')]
271 fields = sub_t.split(
':')[1:]
272 c_topics.extend([
"%s/%s" % (base, f)
for f
in fields
if f])
274 c_topics.append(sub_t)
277 c_topics = [rosgraph.names.script_resolve_name(
'rqt_plot', n)
279 if type(c_topics) == list:
280 topic_list.extend(c_topics)
282 topic_list.append(c_topics)
283 args.topics = topic_list
289 group = parser.add_argument_group(
'Options for rqt_plot plugin')
291 '-P',
'--pause', action=
'store_true', dest=
'start_paused',
292 help=
'Start in paused state')
294 '-L',
'--line', action=
'store_true', dest=
'show_line',
295 help=
'Show lines rather than polygon representation')
297 '--no-legend', action=
'store_true', dest=
'no_legend',
298 help=
'do not show legend')
300 '-B',
'--buffer', dest=
'buffer', action=
"store",
301 help=
'the length of the buffer', default=100, type=int)
306 'topics', nargs=
'*', default=[], help=
'Topics to plot')
310 _redraw_interval = 40
312 def __init__(self, initial_topics=None, start_paused=False,
313 buffer_length=100, use_poly=True, no_legend=False):
314 super(Plot3DWidget, self).
__init__()
315 self.setObjectName(
'Plot3DWidget')
319 rp = rospkg.RosPack()
320 ui_file = os.path.join(rp.get_path(
'jsk_rqt_plugins'),
321 'resource',
'plot3d.ui')
322 loadUi(ui_file, self)
323 self.subscribe_topic_button.setIcon(QIcon.fromTheme(
'add'))
324 self.remove_topic_button.setIcon(QIcon.fromTheme(
'remove'))
325 self.pause_button.setIcon(QIcon.fromTheme(
'media-playback-pause'))
326 self.clear_button.setIcon(QIcon.fromTheme(
'edit-clear'))
329 self.data_plot_layout.addWidget(self.
data_plot)
330 self.
data_plot.autoscroll(self.autoscroll_checkbox.isChecked())
334 self.subscribe_topic_button.setEnabled(
False)
336 self.pause_button.setChecked(
True)
354 @Slot(
'QDragEnterEvent*')
357 if not event.mimeData().hasText():
358 if not hasattr(event.source(),
'selectedItems')
or \
359 len(event.source().selectedItems()) == 0:
361 'Plot.dragEnterEvent(): not hasattr(event.source(), selectedItems) or len(event.source().selectedItems()) == 0')
363 item = event.source().selectedItems()[0]
364 topic_name = item.data(0, Qt.UserRole)
365 if topic_name ==
None:
367 'Plot.dragEnterEvent(): not hasattr(item, ros_topic_name_)')
370 topic_name = str(event.mimeData().text())
373 is_numeric, is_array, message = is_slot_numeric(topic_name)
374 if is_numeric
and not is_array:
375 event.acceptProposedAction()
377 qWarning(
'Plot.dragEnterEvent(): rejecting: "%s"' % (message))
381 if event.mimeData().hasText():
382 topic_name = str(event.mimeData().text())
384 droped_item = event.source().selectedItems()[0]
385 topic_name = str(droped_item.data(0, Qt.UserRole))
391 if topic_name
in (
'',
'/'):
394 is_numeric, is_array, message = is_slot_numeric(topic_name)
395 self.subscribe_topic_button.setEnabled(is_numeric
and not is_array)
396 self.subscribe_topic_button.setToolTip(message)
400 if self.subscribe_topic_button.isEnabled():
401 self.
add_topic(str(self.topic_edit.text()))
405 self.
add_topic(str(self.topic_edit.text()))
422 for topic_name, rosdata
in self.
_rosdata.items():
424 data_x, data_y = rosdata.next()
427 topic_name, data_x, data_y)
429 except RosPlotException
as e:
431 'PlotWidget.update_plot(): error in rosplot: %s' % e)
437 if not self.pause_button.isChecked():
443 def make_remove_topic_function(x):
447 for topic_name
in sorted(self.
_rosdata.keys()):
449 action.triggered.connect(make_remove_topic_function(topic_name))
456 qWarning(
'PlotWidget.add_topic(): topic already subscribed: %s' % topic_name)
460 if self.
_rosdata[topic_name].error
is not None:
461 qWarning(str(self.
_rosdata[topic_name].error))
464 data_x, data_y = self.
_rosdata[topic_name].next()
465 self.
data_plot.add_curve(topic_name, topic_name, data_x, data_y)
477 for topic_name, rosdata
in self.
_rosdata.items():