scope_widget.cpp
Go to the documentation of this file.
1 /*********************************************************************
2  *
3  * Software License Agreement
4  *
5  * Copyright (c) 2020,
6  * TU Dortmund - Institute of Control Theory and Systems Engineering.
7  * All rights reserved.
8  *
9  * This program is free software: you can redistribute it and/or modify
10  * it under the terms of the GNU General Public License as published by
11  * the Free Software Foundation, either version 3 of the License, or
12  * (at your option) any later version.
13  *
14  * This program is distributed in the hope that it will be useful,
15  * but WITHOUT ANY WARRANTY; without even the implied warranty of
16  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17  * GNU General Public License for more details.
18  *
19  * You should have received a copy of the GNU General Public License
20  * along with this program. If not, see <https://www.gnu.org/licenses/>.
21  *
22  * Authors: Christoph Rösmann
23  *********************************************************************/
24 
25 #include <corbo-core/console.h>
27 #include <corbo-gui/scope_widget.h>
28 #include <corbo-gui/utilities.h>
29 #include <QVBoxLayout>
30 #include <QVector>
31 
32 #include <algorithm>
33 #include <vector>
34 
35 namespace corbo {
36 namespace gui {
37 
38 ScopeWidget::ScopeWidget(SignalHelper::ConstPtr signal_helper, QWidget* parent) : QWidget(parent), _signal_helper(signal_helper)
39 {
40  setSizePolicy(QSizePolicy::Expanding, QSizePolicy::MinimumExpanding);
41 
42  _layout = new QVBoxLayout(this);
43  _layout->setAlignment(Qt::AlignTop | Qt::AlignLeft);
44 
45  // setup plot widget
46  _plot = new QCustomPlot;
47  _plot->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::MinimumExpanding);
48  _plot->setMinimumHeight(225); // minimum height
49  // _plot->setInteractions(QCP::iRangeDrag | QCP::iRangeZoom | QCP::iSelectLegend | QCP::iSelectPlottables | QCP::iSelectAxes);
51 
52  // include this section to fully disable antialiasing for higher performance:
53  /*
54  _plot->setNotAntialiasedElements(QCP::aeAll);
55  QFont font;
56  font.setStyleStrategy(QFont::NoAntialias);
57  _plot->xAxis->setTickLabelFont(font);
58  _plot->yAxis->setTickLabelFont(font);
59  _plot->legend->setFont(font);
60  */
61 
62  _layout->addWidget(_plot);
63 
64  setupLegend();
65 
66  // connect slots that takes care that when an axis is selected, only that direction can be dragged and zoomed:
67  connect(_plot, SIGNAL(mousePress(QMouseEvent*)), this, SLOT(scopeMousePress()));
68  connect(_plot, SIGNAL(mouseWheel(QWheelEvent*)), this, SLOT(scopeMouseWheel()));
69  connect(_plot, SIGNAL(mouseDoubleClick(QMouseEvent*)), this, SLOT(rescaleAxes()));
70 
71  // setup policy and connect slot for context menu popup:
72  _plot->setContextMenuPolicy(Qt::CustomContextMenu);
73  connect(_plot, SIGNAL(customContextMenuRequested(const QPoint&)), this, SLOT(scopeContextMenuRequest(const QPoint&)));
74 
75  // enable drag and drop of signals
76  setAcceptDrops(true);
77 
78  // for resizing the QWidget
79  setMouseTracking(true);
80 }
81 
83 
84 QSize ScopeWidget::sizeHint() const { return QSize(800, 200); }
85 
87 {
89  {
90  // _plot->setAutoAddPlottableToLegend(false);
91  // _plot->setLocale(QLocale(QLocale::English, QLocale::UnitedKingdom)); // period as decimal separator and comma as thousand
92  // separator
93  _plot->legend->setVisible(true);
94  QFont legend_font = font(); // start out with MainWindow's font..
95  legend_font.setPointSize(8); // and make a bit smaller for legend
96  _plot->legend->setFont(legend_font);
97  _plot->legend->setBrush(QBrush(QColor(255, 255, 255, 230)));
98  _plot->legend->setIconSize(17, 10);
99  QPen border_pen = _plot->legend->borderPen();
100  border_pen.setColor(QColor("white"));
101  _plot->legend->setBorderPen(border_pen);
102 
103  // move legend into the axis rect of the main grid layout.
104  // create sub layout to generate a small gap between plot cell border and legend border
105  QCPLayoutGrid* sub_layout = new QCPLayoutGrid;
106  sub_layout->setMargins(QMargins(5, 0, 5, 5));
107  sub_layout->addElement(0, 0, _plot->legend);
108  sub_layout->addElement(0, 1, new QCPLayoutElement(_plot)); // dummy element in order to force to fillout the complete width
109  sub_layout->setColumnStretchFactor(1, 0.0000001); // set dummy element width to a neglectable width
110 
111  _plot->plotLayout()->addElement(1, 0, sub_layout);
112 
113  // change fill order to: left -> right
115  // set legend's row stretch factor very small so it ends up with minimum height:
116  _plot->plotLayout()->setRowStretchFactor(1, 0.001);
117 
118  _plot->legend->setSelectableParts(QCPLegend::spItems); // legend box shall not be selectable, only legend items
119 
120  _plot->replot();
121 
122  _legend_initialized = true;
123  }
124 }
125 
126 void ScopeWidget::dragEnterEvent(QDragEnterEvent* event)
127 {
128  if (event->mimeData()->hasFormat("text/plain")) event->acceptProposedAction();
129 }
130 
131 void ScopeWidget::dropEvent(QDropEvent* event)
132 {
133  QString message = event->mimeData()->text();
134 
135  QString key;
136  int value_idx = 0;
137  fromValueEncodedKey(message, key, value_idx);
138 
139  addSignal(key, value_idx);
140  event->acceptProposedAction();
141 }
142 
143 void ScopeWidget::mousePressEvent(QMouseEvent* event)
144 {
145  if (event->button() == Qt::LeftButton)
146  {
147  _resize_drag_start_position = event->pos();
148  _resize_drag_start_geometry = _plot->geometry();
149  }
150 }
151 
152 void ScopeWidget::mouseMoveEvent(QMouseEvent* event)
153 {
154  if (!(event->buttons() & Qt::LeftButton))
155  {
156  // No drag, just change the cursor and return
157  if (event->y() >= height() - 5)
158  {
160  setCursor(Qt::SizeVerCursor);
161  }
162  else
163  {
164  // default cursor
166  setCursor(Qt::ArrowCursor);
167  }
168  return;
169  }
170 
171  // drag to change height
172 
173  // consider minimum height of plot field including margin
174  if (event->y() <= _plot->minimumHeight() + _layout->margin() + 20) return;
175 
176  switch (_drag_start_pos)
177  {
178  case bottom:
179  {
180  QPoint window_coords = mapFrom(window(), QPoint(0, window()->height()));
181  if (event->y() < window_coords.y() - 3) // add a small margin which we assume as window bottom
182  {
183  // setGeometry(geometry().left(), geometry().top(), width(), event->y());
184  setFixedHeight(event->y());
185  }
186  else
187  {
188  // cursor is at the bottom of the window (parent widget)
189  // increase height as long as drag is active
190  setFixedHeight(height() + 2); // otherwise, I was not able to update the parant layout correctly...
191  }
192 
193  break;
194  }
195  default:
196  break;
197  }
198 }
199 
200 void ScopeWidget::addSignal(const QString& signal_key, int value_idx)
201 {
202  // new signal
203  const SignalHelper::SignalData* signal = _signal_helper->signalData(signal_key);
204  if (signal)
205  {
206  addSignal(signal_key, value_idx, *signal);
207  }
208  else
209  PRINT_WARNING("ScopeWidget::addSignal(): signal " << signal_key.toStdString() << " not found.");
210 }
211 
212 void ScopeWidget::addSignal(const QString& signal_key, int value_idx, const SignalHelper::SignalData& signal_data)
213 {
214  QString value_encoded_key = toValueEncodedKey(signal_key, value_idx);
215  auto previous_data = getActiveSignal(value_encoded_key);
216  if (previous_data != _active_signals.end())
217  {
218  // clear potential old graph with same key
219  _active_signals.erase(previous_data);
220  }
221 
222  SignalData& data = _active_signals[value_encoded_key];
223  data.graph_color = ColorManager::getColor(_plot->plottableCount());
224  data.legend_text = signal_data.name;
225  data.zero_order_hold = signal_data.zero_order_hold;
226  if (signal_data.dimension > 1)
227  data.legend_text += "/" + ((value_idx < signal_data.value_labels.size()) ? signal_data.value_labels[value_idx] : QString::number(value_idx));
228  data.value_idx = value_idx;
229  data.task_id = signal_data.task_id;
230 
231  // signal type dependent
232  data.plottable = nullptr;
233 
234  switch (signal_data.signal->getType())
235  {
237  {
238  const TimeSeries* ts = static_cast<const TimeSeriesSignal*>(signal_data.signal.get())->getTimeSeries();
239  data.plottable = addTimeSeriesGraph(*ts, value_idx, data.graph_color, data.legend_text, signal_data.zero_order_hold, true);
240  break;
241  }
243  {
244  data.ts_sequence = std::static_pointer_cast<const TimeSeriesSequenceSignal>(signal_data.signal)->getSequencePtr();
245  updateTimeSeriesSequenceGraph(data); // this method also adds the current TimeSeriesGraph
246  break;
247  }
249  {
250  const IndexedValuesSetSignal* set = static_cast<const IndexedValuesSetSignal*>(signal_data.signal.get());
251  data.plottable = addBoxPlot(*set, data.graph_color, data.legend_text, true);
252  break;
253  }
254  default:
255  {
256  QMessageBox::warning(this, tr("Cannot plot Signal"), "Signal type not supported yet.");
257  }
258  }
259 }
260 
261 QCPAbstractPlottable* ScopeWidget::addTimeSeriesGraph(const TimeSeries& time_series, int value_idx, const QColor& color, const QString& legend_text,
262  bool zero_order_hold, bool replot)
263 {
264  // if (time_series.timeDimension() == 0) return nullptr;
265 
266  if (value_idx < 0 || value_idx >= time_series.getValueDimension())
267  {
268  return nullptr;
269  }
270 
271  QVector<double> time;
272  for (const double& t : time_series.getTime()) time.push_back(t + time_series.getTimeFromStart());
273 
274  QVector<double> values;
275  TimeSeries::ValuesMatConstMap values_mat = time_series.getValuesMatrixView();
276  for (int t = 0; t < values_mat.cols(); ++t)
277  {
278  values.push_back(values_mat(value_idx, t));
279  }
280 
281  QCPGraph* graph = _plot->addGraph();
282 
283  graph->setPen(QPen(color));
284  // graph->setSelectedPen(QPen(QColor("blue"),2));
286  graph->setAntialiasedFill(false);
287  graph->setData(time, values, true);
288  graph->setName(legend_text);
289 
290  if (zero_order_hold) graph->setLineStyle(QCPGraph::lsStepLeft); // lsStepLeft
291 
292  if (time.empty())
293  setGraphActive(graph, false); // not active resp. no values
294  else
295  setGraphActive(graph, true);
296 
297  graph->rescaleAxes(_plot->graphCount() > 1);
298  // if (replot) _plot->replot(QCustomPlot::rpQueuedReplot);
299  if (replot) _plot->replot();
300  return graph;
301 }
302 
303 void ScopeWidget::updateTimeSeriesGraph(SignalData& data, double t, double value, bool enlarge_axis, bool replot)
304 {
305  QCPGraph* graph = dynamic_cast<QCPGraph*>(data.plottable);
306  if (!graph)
307  {
308  PRINT_ERROR_NAMED("cannot update plottable since is not of type OCPGraph");
309  return;
310  }
311 
312  graph->addData(t, value);
313  if (enlarge_axis) graph->rescaleAxes(true);
314  // TODO(roesmann) do not replot for every update -> create a timer in a separate thread to trigger replot with a slower rate!!
315  // if (replot) _plot->replot(QCustomPlot::rpQueuedReplot);
316  if (replot) _plot->replot();
317 }
318 
319 void ScopeWidget::updateTimeSeriesGraph(SignalData& data, const TimeSeries& time_series, bool enlarge_axis, bool replace_data, bool replot)
320 {
321  if (!data.plottable) return;
322 
323  QCPGraph* graph = dynamic_cast<QCPGraph*>(data.plottable);
324  if (!graph)
325  {
326  PRINT_ERROR_NAMED("cannot update plottable since is not of type OCPGraph");
327  return;
328  }
329 
330  if (data.value_idx < 0 || data.value_idx >= time_series.getValueDimension()) return;
331 
332  QVector<double> time;
333  for (const double& t : time_series.getTime()) time.push_back(t + time_series.getTimeFromStart());
334 
335  QVector<double> values;
336  TimeSeries::ValuesMatConstMap values_mat = time_series.getValuesMatrixView();
337  for (int t = 0; t < values_mat.cols(); ++t)
338  {
339  values.push_back(values_mat(data.value_idx, t));
340  }
341 
342  // data.graph->addData(t, value);
343  if (replace_data)
344  {
345  graph->setData(time, values);
346  }
347  else
348  {
349  graph->addData(time, values);
350  }
351 
352  if (graph->data()->isEmpty())
353  setGraphActive(graph, false); // not active resp. no values
354  else
355  setGraphActive(graph, true);
356 
357  if (enlarge_axis) graph->rescaleAxes(true);
358  // if (replot) _plot->replot(QCustomPlot::rpQueuedReplot);
359  if (replot) _plot->replot();
360 }
361 
362 void ScopeWidget::addMeasurement(const QString& key, Measurement::ConstPtr measurement, const SignalHelper::SignalData& data)
363 {
364  // get all graphs that correspond to that signal
365  QVector<QHash<QString, ScopeWidget::SignalData>::iterator> active_signals = findActiveSignal(key, SearchType::Exact);
366  for (int i = 0; i < active_signals.size(); ++i)
367  {
368  // first check if the current signal is already initialized or just reserved
369  if (active_signals[i].value().plottable == nullptr)
370  {
371  // just reserved: we have to initialize the new signal
372  int value_idx;
373  QString singal_key;
374  if (!fromValueEncodedKey(active_signals[i].key(), singal_key, value_idx)) continue;
375  addSignal(singal_key, value_idx, data);
376  return;
377  }
378 
379  QCPGraph* graph = dynamic_cast<QCPGraph*>(active_signals[i].value().plottable);
380  if (!graph) continue;
381 
382  if (active_signals[i].value().value_idx < measurement->getValues().size())
383  {
384  updateTimeSeriesGraph(active_signals[i].value(), measurement->getTime(), measurement->getValues()[active_signals[i].value().value_idx],
385  false, false);
386 
387  if (!isGraphActive(graph)) setGraphActive(graph, true);
388  }
389  else
390  {
391  PRINT_ERROR(
392  "Received values for plotting a time-series to scope, but the dimension is smaller than the values "
393  "dimension.");
394  }
395  }
396  // double dur = (measurement->header.time - _last_signal_header_time).toSec();
397  // if (dur > 0.05)
398  //{
399  // _last_signal_header_time = measurement->header.time;
400  _plot->rescaleAxes(true);
402  //}
403 }
404 
405 void ScopeWidget::setPreviewTime(double preview_time)
406 {
407  _current_preview_time = preview_time;
408 
409  for (auto it = _active_signals.begin(); it != _active_signals.end(); ++it)
410  {
411  if (it->ts_sequence) updateTimeSeriesSequenceGraph(*it, false);
412  }
413  _plot->replot();
414 }
415 
416 void ScopeWidget::updateTimeSeriesSequenceGraph(SignalData& data, bool replot)
417 {
418  if (!data.ts_sequence) return;
419 
420  // we assume that data has already been filled appropriately (everything except graph must be set, e.g. color, value_idx, legend_text,
421  // ...).
422 
423  TimeSeries::ConstPtr current_ts;
424 
425  // find TimeSeries to be plotted (w.r.t. _current_preview_time)
426  // we assume that the TimeSeries sequence is already ordered according to its time_from_start members.
427  auto found_it = std::find_if(data.ts_sequence->getSequence().rbegin(), data.ts_sequence->getSequence().rend(),
428  [this](const TimeSeries::Ptr& ts) { return ts->getTimeFromStart() <= _current_preview_time; }); // reverse iterators
429  if (found_it != data.ts_sequence->getSequence().rend())
430  {
431  // found
432  current_ts = *found_it;
433  }
434  else
435  {
436  // plot first available time series:
437  if (data.ts_sequence->getSequence().empty())
438  {
439  PRINT_ERROR("ScopeWidget::updateTimeSeriesSequenceGraph(): time series sequence is empty.");
440  return;
441  }
442  current_ts = data.ts_sequence->getSequence().front();
443  }
444 
445  // check if we have plotted a TimeSeries of this sequence before
446  if (data.plottable)
447  {
448  // just update the values in order to keep color, labels and legend key.
449  updateTimeSeriesGraph(data, *current_ts, true, true, replot);
450  }
451  else
452  {
453  // add a complete new TimeSeriesPlot
454  data.plottable = addTimeSeriesGraph(*current_ts, data.value_idx, data.graph_color, data.legend_text, data.zero_order_hold, replot);
455  }
456 }
457 
458 QCPAbstractPlottable* ScopeWidget::addBoxPlot(const IndexedValuesSetSignal& indexed_values_set, const QColor& color, const QString& legend_text,
459  bool replot)
460 {
461  if (indexed_values_set.isEmpty()) return nullptr;
462 
464 
465  for (const auto& item : indexed_values_set.getData())
466  {
467  // QVector<double> data = QVector<double>::fromStdVector(item.second);
468  std::vector<double> data = item.second;
469 
470  // get statistic properties of the data
471  double minimum = *std::min_element(data.begin(), data.end());
472  double maximum = *std::max_element(data.begin(), data.end());
473 
474  int num_quartile1 = data.size() / 4;
475  int num_quartile2 = data.size() / 2;
476  int num_quartile3 = num_quartile1 + num_quartile2;
477 
478  std::nth_element(data.begin(), data.begin() + num_quartile1, data.end());
479  std::nth_element(data.begin() + num_quartile1 + 1, data.begin() + num_quartile2, data.end());
480  std::nth_element(data.begin() + num_quartile2 + 1, data.begin() + num_quartile3, data.end());
481 
482  double lower_quartile = data[num_quartile1];
483  double median = data[num_quartile2];
484  double upper_quartile = data[num_quartile3];
485 
486  box->addData(item.first, minimum, lower_quartile, median, upper_quartile, maximum); // no outliers yet
487  }
488 
489  box->setName(legend_text);
490  box->setPen(QPen(color));
491 
492  // QBrush box_brush(QColor(60, 60, 255, 100));
493  // box_brush.setStyle(Qt::Dense6Pattern); // make it look oldschool
494  // box->setBrush(box_brush);
495 
496  _plot->rescaleAxes();
497 
498  if (replot) _plot->replot();
499 
500  return box;
501 }
502 
503 void ScopeWidget::removeSignal(const QString& key, int value_idx)
504 {
505  int removed = 0;
506  if (value_idx != SignalHelper::ALL_VALUES)
507  {
508  QString value_encoded_key = toValueEncodedKey(key, value_idx);
509  removed = _active_signals.remove(value_encoded_key);
510  }
511  else
512  {
513  auto iterators = findActiveSignal(key, SearchType::Exact);
514  for (auto& it : iterators) _active_signals.erase(it);
515  removed = (int)iterators.size();
516  }
517 
518  if (removed > 0)
519  {
520  _plot->rescaleAxes(true);
521  _plot->replot();
522  }
523 }
524 
525 void ScopeWidget::initializeTask(int task_id, bool inherit_signals)
526 {
527  if (inherit_signals)
528  {
529  QString signal_key;
530  for (auto it = _active_signals.begin(); it != _active_signals.end(); ++it)
531  {
532  int value_idx;
533  QString signal_key;
534  if (!fromValueEncodedKey(it.key(), signal_key, value_idx)) continue;
535  QString name;
536  int prev_task_id;
537  if (!SignalHelper::key2Name(signal_key, name, prev_task_id)) continue;
538 
539  if (prev_task_id == task_id - 1) // only inherit if also obtained in the last task run
540  {
541  // new task found (this might only happen if search_type == SearchType::IgnorePrefix)
542  QString new_key = SignalHelper::name2Key(name, task_id);
543  // reserve field which should be initialized in one of the add-signals methods
544  SignalData& data = _active_signals[toValueEncodedKey(new_key, value_idx)];
545  return;
546  }
547  }
548  }
549 }
550 
552 {
553  _plot->rescaleAxes(false);
554  // _plot->replot(QCustomPlot::rpQueuedReplot);
555  _plot->replot();
556 }
557 
558 bool ScopeWidget::hasSignal(const QString& key, int value_idx) const
559 {
560  auto it = _active_signals.find(toValueEncodedKey(key, value_idx));
561  if (it != _active_signals.end()) return true;
562 
563  return false;
564 }
565 
566 QHash<QString, ScopeWidget::SignalData>::iterator ScopeWidget::getActiveSignal(const QString& value_encoded_key)
567 {
568  auto it = _active_signals.find(value_encoded_key);
569  if (it != _active_signals.end()) return it;
570 
571  return _active_signals.end();
572 }
573 
574 QHash<QString, ScopeWidget::SignalData>::iterator ScopeWidget::getActiveSignal(const QString& key, int value_idx)
575 {
576  return getActiveSignal(toValueEncodedKey(key, value_idx));
577 }
578 
579 QVector<QHash<QString, ScopeWidget::SignalData>::iterator> ScopeWidget::findActiveSignal(const QString& key, SearchType search_type)
580 {
581  QVector<QHash<QString, ScopeWidget::SignalData>::iterator> active_list;
582 
583  QString signal_key;
584  int value_idx = 0;
585  for (auto it = _active_signals.begin(); it != _active_signals.end(); ++it)
586  {
587  if (fromValueEncodedKey(it.key(), signal_key, value_idx))
588  {
589  bool found = false;
590  switch (search_type)
591  {
592  case SearchType::Exact:
593  {
594  found = (signal_key.compare(key) == 0);
595  break;
596  }
598  {
599  QStringList token_source = key.split(util::SIGNAL_NAMESPACE_PREFIX_DELIMITER);
600  QStringList token_active = signal_key.split(util::SIGNAL_NAMESPACE_PREFIX_DELIMITER);
601  found = (token_active.back().compare(token_source.back()) == 0);
602  break;
603  }
604  default:
605  {
606  PRINT_WARNING("ScopeWidget::findActiveSignal(): selected search type not implemented.");
607  }
608  }
609  if (found) active_list.push_back(it);
610  }
611  }
612  return active_list;
613 }
614 
615 QString ScopeWidget::toValueEncodedKey(const QString& key, int value_idx) const
616 {
617  return key + util::SIGNAL_NAMESPACE_SUFFIX_DELIMITER + QString::number(value_idx);
618 }
619 
620 bool ScopeWidget::fromValueEncodedKey(const QString& value_encoded_key, QString& key, int& value_idx)
621 {
622  QStringList tokens = value_encoded_key.split(util::SIGNAL_NAMESPACE_SUFFIX_DELIMITER);
623  PRINT_WARNING_COND(tokens.size() > 2, "QString::fromValueEnvodedKey: more than 2 tokens found...");
624 
625  bool ret_val = true;
626 
627  key = tokens.front();
628  if (tokens.size() > 1)
629  {
630  bool ok;
631  value_idx = tokens.back().toInt(&ok);
632  if (!ok)
633  {
634  PRINT_ERROR("QString::fromValueEnvodedKey: cannot detect value index in " << value_encoded_key.toStdString());
635  value_idx = 0;
636  ret_val = false;
637  }
638  }
639  else
640  {
641  value_idx = 0;
642  ret_val = false;
643  PRINT_WARNING("QString::fromValueEnvodedKey: no value_idx found in " << value_encoded_key.toStdString() << ". Setting value_idx to 0");
644  }
645  return ret_val;
646 }
647 
649 {
650  // if an axis is selected, only allow the direction of that axis to be dragged
651  // if no axis is selectedor or if both axes are selected (multi-select), both directions may be dragged
652 
653  if (_plot->xAxis->selectedParts().testFlag(QCPAxis::spAxis) && !_plot->yAxis->selectedParts().testFlag(QCPAxis::spAxis))
655  else if (!_plot->xAxis->selectedParts().testFlag(QCPAxis::spAxis) && _plot->yAxis->selectedParts().testFlag(QCPAxis::spAxis))
657  else
659 }
660 
662 {
663  // if an axis is selected, only allow the direction of that axis to be zoomed
664  // if no axis is selected or if both axes are selected (multi-select), both directions may be zoomed
665 
666  if (_plot->xAxis->selectedParts().testFlag(QCPAxis::spAxis) && !_plot->yAxis->selectedParts().testFlag(QCPAxis::spAxis))
668  else if (!_plot->xAxis->selectedParts().testFlag(QCPAxis::spAxis) && _plot->yAxis->selectedParts().testFlag(QCPAxis::spAxis))
670  else
672 }
673 
674 void ScopeWidget::scopeContextMenuRequest(const QPoint& point)
675 {
676  // QMenu* menu = new QMenu(this);
677  // menu->setAttribute(Qt::WA_DeleteOnClose);
678  QMenu menu;
679 
680  // context menu on legend requested
681  if (_plot->legend->selectTest(point, false) >= 0)
682  {
683  // find legend for active graphs
684  auto it = _active_signals.begin();
685  while (it != _active_signals.end())
686  {
687  QCPPlottableLegendItem* graph_item = _plot->legend->itemWithPlottable(it.value().plottable);
688  if (graph_item)
689  {
690  if (graph_item->selectTest(point, false) >= 0)
691  {
692  QAction* remove_action = new QAction(tr("Remove signal"), this);
693  connect(remove_action, &QAction::triggered, [this, it]() {
694  _active_signals.erase(it);
695  _plot->replot();
696  });
697  menu.addAction(remove_action);
698  break;
699  }
700  }
701  ++it;
702  }
703  }
704  else // general context menu
705  {
706  QAction* rescale_axes = new QAction(tr("Rescale axes"), this);
707  connect(rescale_axes, &QAction::triggered, [this]() { rescaleAxes(); });
708  menu.addAction(rescale_axes);
709 
710  menu.addSeparator();
711 
712  bool signal_options = false;
713  if (_plot->selectedGraphs().size() > 0)
714  {
715  menu.addAction("Remove selected signals", this, SLOT(removeSelectedSignals()));
716  signal_options = true;
717  }
718  if (_plot->graphCount() > 0)
719  {
720  menu.addAction("Remove all signals", this, SLOT(removeAllSignals()));
721  signal_options = true;
722  }
723 
724  if (signal_options) menu.addSeparator();
725 
726  QAction* close_scope = new QAction(tr("Close scope"), this);
727  connect(close_scope, &QAction::triggered, [this]() { deleteLater(); });
728  menu.addAction(close_scope);
729  }
730 
731  // menu->popup(_plot->mapToGlobal(point));
732  menu.exec(_plot->mapToGlobal(point));
733 }
734 
736 {
737  QMutableHashIterator<QString, SignalData> it(_active_signals);
738  bool update = false;
739  while (it.hasNext())
740  {
741  it.next();
742  if (it.value().plottable && it.value().plottable->selected())
743  {
744  it.remove();
745  update = true;
746  }
747  }
748  if (update) _plot->replot();
749 }
750 
752 {
753  _active_signals.clear();
754  _plot->replot();
755 }
756 
757 bool ScopeWidget::isGraphActive(const QCPGraph* graph) const
758 {
759  if (!graph) return false;
760 
761  QVariant active = graph->property("active");
762 
763  if (active.isValid() && active.toBool()) return true;
764  return false;
765 }
766 
767 void ScopeWidget::setGraphActive(QCPGraph* graph, bool active)
768 {
769  // set to active (adjust legend font)
771  if (item)
772  {
773  if (active)
774  item->setTextColor(QColor(0, 0, 0));
775  else
776  item->setTextColor(QColor(0, 0, 0, 50));
777  graph->setProperty("active", active);
778  }
779 }
780 
781 } // namespace gui
782 } // namespace corbo
corbo::gui::ScopeWidget::removeSelectedSignals
void removeSelectedSignals()
Definition: scope_widget.cpp:779
QCustomPlot::xAxis
QCPAxis * xAxis
Definition: qcustomplot.h:3739
corbo::gui::util::SIGNAL_NAMESPACE_PREFIX_DELIMITER
constexpr const char SIGNAL_NAMESPACE_PREFIX_DELIMITER[]
Definition: gui/include/corbo-gui/utilities.h:93
corbo::gui::ScopeWidget::_layout
QVBoxLayout * _layout
Definition: scope_widget.h:200
corbo::TimeSeries::ValuesMatConstMap
Eigen::Map< const Eigen::Matrix< double, Eigen::Dynamic, Eigen::Dynamic, Eigen::ColMajor > > ValuesMatConstMap
Definition: time_series.h:111
QCPLayoutGrid::addElement
bool addElement(int row, int column, QCPLayoutElement *element)
Definition: qcustomplot.cpp:4134
QCustomPlot::axisRect
QCPAxisRect * axisRect(int index=0) const
Definition: qcustomplot.cpp:14215
QCPGraph::setScatterStyle
void setScatterStyle(const QCPScatterStyle &style)
Definition: qcustomplot.cpp:20181
QCPAbstractPlottable::setPen
void setPen(const QPen &pen)
Definition: qcustomplot.cpp:10770
relicense.update
def update(text)
Definition: relicense.py:46
corbo::gui::ScopeWidget::scopeContextMenuRequest
void scopeContextMenuRequest(const QPoint &point)
Definition: scope_widget.cpp:718
QCPLayoutElement
The abstract base class for all objects that form the layout system.
Definition: qcustomplot.h:1184
corbo::gui::ScopeWidget::isGraphActive
bool isGraphActive(const QCPGraph *graph) const
Definition: scope_widget.cpp:801
corbo::gui::ScopeWidget::scopeMousePress
void scopeMousePress()
Definition: scope_widget.cpp:692
QCPLegend::spItems
@ spItems
0x002 Legend items individually (see selectedItems)
Definition: qcustomplot.h:4834
QCPGraph::data
QSharedPointer< QCPGraphDataContainer > data() const
Definition: qcustomplot.h:5193
QCPAxisRect::setRangeDrag
void setRangeDrag(Qt::Orientations orientations)
Definition: qcustomplot.cpp:17492
PRINT_WARNING
#define PRINT_WARNING(msg)
Print msg-stream.
Definition: console.h:145
PRINT_ERROR_NAMED
#define PRINT_ERROR_NAMED(msg)
Definition: console.h:260
corbo::gui::ScopeWidget::ScopeWidget
ScopeWidget(SignalHelper::ConstPtr signal_helper, QWidget *parent=0)
Definition: scope_widget.cpp:82
corbo::SignalType::IndexedValuesSet
@ IndexedValuesSet
corbo::SignalType::TimeSeriesSequence
@ TimeSeriesSequence
corbo::gui::ColorManager::getColor
static const QColor & getColor(int index)
Definition: color_manager.h:105
QCPAbstractLegendItem::selectTest
virtual double selectTest(const QPointF &pos, bool onlySelectable, QVariant *details=0) const Q_DECL_OVERRIDE
Definition: qcustomplot.cpp:18094
corbo::gui::ScopeWidget::getActiveSignal
QHash< QString, SignalData >::iterator getActiveSignal(const QString &value_encoded_key)
Definition: scope_widget.cpp:610
QCustomPlot::rpQueuedReplot
@ rpQueuedReplot
Queues the entire replot for the next event loop iteration. This way multiple redundant replots can b...
Definition: qcustomplot.h:3624
QCPLegend::borderPen
QPen borderPen() const
Definition: qcustomplot.h:4844
Eigen::Horizontal
@ Horizontal
Definition: Constants.h:268
corbo
Definition: communication/include/corbo-communication/utilities.h:37
corbo::gui::ScopeWidget::removeAllSignals
void removeAllSignals()
Definition: scope_widget.cpp:795
corbo::gui::ScopeWidget::dragEnterEvent
void dragEnterEvent(QDragEnterEvent *event) override
Definition: scope_widget.cpp:170
corbo::gui::ScopeWidget::removeSignal
void removeSignal(const QString &key, int value_idx)
Definition: scope_widget.cpp:547
QCustomPlot::replot
Q_SLOT void replot(QCustomPlot::RefreshPriority refreshPriority=QCustomPlot::rpRefreshHint)
Definition: qcustomplot.cpp:14423
QCustomPlot::setInteractions
void setInteractions(const QCP::Interactions &interactions)
Definition: qcustomplot.cpp:13128
QCPLegend::setBorderPen
void setBorderPen(const QPen &pen)
Definition: qcustomplot.cpp:18382
corbo::TimeSeries::ConstPtr
std::shared_ptr< const TimeSeries > ConstPtr
Definition: time_series.h:109
corbo::SignalType::TimeSeries
@ TimeSeries
corbo::gui::ScopeWidget::setGraphActive
void setGraphActive(QCPGraph *graph, bool active)
Definition: scope_widget.cpp:811
scope_widget.h
color_manager.h
console.h
corbo::gui::SignalHelper::key2Name
static bool key2Name(const QString &key, QString &name, int &id)
Definition: signal_helper.cpp:435
QCPGraph
A plottable representing a graph in a plot.
Definition: qcustomplot.h:5163
corbo::gui::ScopeWidget::initializeTask
void initializeTask(int task_id, bool inherit_signals)
Definition: scope_widget.cpp:569
QCPScatterStyle
Represents the visual appearance of scatter points.
Definition: qcustomplot.h:2296
QCP::iSelectAxes
@ iSelectAxes
0x010 Axes are selectable (or parts of them, see QCPAxis::setSelectableParts)
Definition: qcustomplot.h:260
QCPAbstractPlottable::setAntialiasedFill
void setAntialiasedFill(bool enabled)
Definition: qcustomplot.cpp:10746
QCPLegend::setSelectableParts
Q_SLOT void setSelectableParts(const SelectableParts &selectableParts)
Definition: qcustomplot.cpp:18481
corbo::gui::ScopeWidget::dropEvent
void dropEvent(QDropEvent *event) override
Definition: scope_widget.cpp:175
QCPLayoutElement::setMargins
void setMargins(const QMargins &margins)
Definition: qcustomplot.cpp:3234
corbo::gui::ScopeWidget::_resize_drag_start_geometry
QRect _resize_drag_start_geometry
Definition: scope_widget.h:196
QCustomPlot::yAxis
QCPAxis * yAxis
Definition: qcustomplot.h:3739
QCP::iMultiSelect
@ iMultiSelect
0x004 The user can select multiple objects by holding the modifier set by QCustomPlot::setMultiSelect...
Definition: qcustomplot.h:258
corbo::gui::ScopeWidget::_current_preview_time
double _current_preview_time
determine which TimeSeries of a TimeSeriesSequence should be plotted
Definition: scope_widget.h:198
QCPLegend::itemWithPlottable
QCPPlottableLegendItem * itemWithPlottable(const QCPAbstractPlottable *plottable) const
Definition: qcustomplot.cpp:18620
QCPScatterStyle::ssDisc
@ ssDisc
\enumimage{ssDisc.png} a circle which is filled with the pen's color (not the brush as with ssCircle)
Definition: qcustomplot.h:2331
corbo::gui::ScopeWidget::~ScopeWidget
virtual ~ScopeWidget()
Definition: scope_widget.cpp:126
QCPLegend::setFont
void setFont(const QFont &font)
Definition: qcustomplot.cpp:18404
corbo::gui::ScopeWidget::_plot
QCustomPlot * _plot
Definition: scope_widget.h:184
QCPLegend::setBrush
void setBrush(const QBrush &brush)
Definition: qcustomplot.cpp:18390
QCustomPlot::plottableCount
int plottableCount() const
Definition: qcustomplot.cpp:13595
corbo::TimeSeries::getValueDimension
int getValueDimension() const
Return dimension of the value vector.
Definition: time_series.h:123
QCPGraph::lsStepLeft
@ lsStepLeft
line is drawn as steps where the step height is the value of the left data point
Definition: qcustomplot.h:5182
corbo::TimeSeries::getValuesMatrixView
ValuesMatConstMap getValuesMatrixView() const
Read access to the complete values matrix in Eigen matrix format [getValueDimension() x getTimeDimens...
Definition: time_series.h:208
QCPAbstractPlottable::rescaleAxes
void rescaleAxes(bool onlyEnlarge=false) const
Definition: qcustomplot.cpp:10994
QCPAxis::spAxis
@ spAxis
The axis backbone and tick marks.
Definition: qcustomplot.h:2002
corbo::gui::SignalHelper::name2Key
static QString name2Key(const QString &name, int id)
Definition: signal_helper.cpp:433
QCPGraph::addData
void addData(const QVector< double > &keys, const QVector< double > &values, bool alreadySorted=false)
Definition: qcustomplot.cpp:20279
corbo::gui::ScopeWidget::replot
void replot()
Definition: scope_widget.h:132
corbo::gui::ScopeWidget::_active_signals
QHash< QString, SignalData > _active_signals
Definition: scope_widget.h:186
QCPLegend::setIconSize
void setIconSize(const QSize &size)
Definition: qcustomplot.cpp:18437
PRINT_WARNING_COND
#define PRINT_WARNING_COND(cond, msg)
Print msg-stream only if cond == true.
Definition: console.h:159
QCP::iSelectPlottables
@ iSelectPlottables
0x008 Plottables are selectable (e.g. graphs, curves, bars,... see QCPAbstractPlottable)
Definition: qcustomplot.h:259
QCPPlottableLegendItem
A legend item representing a plottable with an icon and the plottable name.
Definition: qcustomplot.h:4783
corbo::IndexedValuesSetSignal
Signal containing values indexed by an integer (int to double[] map)
Definition: signals.h:496
QCustomPlot::legend
QCPLegend * legend
Definition: qcustomplot.h:3740
QCPAbstractPlottable::setName
void setName(const QString &name)
Definition: qcustomplot.cpp:10735
QCPLayoutGrid
A layout that arranges child elements in a grid.
Definition: qcustomplot.h:1327
corbo::gui::ScopeWidget::inactive
@ inactive
Definition: scope_widget.h:117
corbo::TimeSeries::getTime
const std::vector< double > & getTime() const
Read access to the underlying time values [getTimeDimension() x 1].
Definition: time_series.h:213
QCPAbstractLegendItem::setTextColor
void setTextColor(const QColor &color)
Definition: qcustomplot.cpp:18035
corbo::gui::SearchType::IgnorePrefix
@ IgnorePrefix
corbo::gui::ScopeWidget::setPreviewTime
void setPreviewTime(double preview_time)
Definition: scope_widget.cpp:449
QCPAxis::orientation
Qt::Orientation orientation() const
Definition: qcustomplot.h:2108
Eigen::Vertical
@ Vertical
Definition: Constants.h:265
QCPAxis::selectedParts
SelectableParts selectedParts() const
Definition: qcustomplot.h:2045
QCustomPlot::addGraph
QCPGraph * addGraph(QCPAxis *keyAxis=0, QCPAxis *valueAxis=0)
Definition: qcustomplot.cpp:13709
int
return int(ret)+1
QCPAbstractPlottable
The abstract base class for all data representing objects in a plot.
Definition: qcustomplot.h:3295
corbo::gui::ScopeWidget::_signal_helper
SignalHelper::ConstPtr _signal_helper
Definition: scope_widget.h:188
corbo::gui::ScopeWidget::rescaleAxes
void rescaleAxes()
Definition: scope_widget.cpp:595
corbo::TimeSeries
Time Series (trajectory resp. sequence of values w.r.t. time)
Definition: time_series.h:76
QCPLayoutGrid::setRowStretchFactor
void setRowStretchFactor(int row, double factor)
Definition: qcustomplot.cpp:4269
QCPStatisticalBox::addData
void addData(const QVector< double > &keys, const QVector< double > &minimum, const QVector< double > &lowerQuartile, const QVector< double > &median, const QVector< double > &upperQuartile, const QVector< double > &maximum, bool alreadySorted=false)
Definition: qcustomplot.cpp:24680
QCustomPlot::rescaleAxes
Q_SLOT void rescaleAxes(bool onlyVisiblePlottables=false)
Definition: qcustomplot.cpp:14466
QCustomPlot
The central class of the library. This is the QWidget which displays the plot and interacts with the ...
Definition: qcustomplot.h:3590
corbo::gui::ScopeWidget::addBoxPlot
QCPAbstractPlottable * addBoxPlot(const IndexedValuesSetSignal &indexed_values_set, const QColor &color, const QString &legend_text, bool replot=true)
Definition: scope_widget.cpp:502
QCustomPlot::plotLayout
QCPLayoutGrid * plotLayout() const
Definition: qcustomplot.h:3637
corbo::gui::ScopeWidget::mousePressEvent
void mousePressEvent(QMouseEvent *event) override
Definition: scope_widget.cpp:187
corbo::gui::SearchType
SearchType
Definition: gui/include/corbo-gui/utilities.h:89
corbo::gui::ScopeWidget::findActiveSignal
QVector< QHash< QString, SignalData >::iterator > findActiveSignal(const QString &key, SearchType search_type)
Definition: scope_widget.cpp:623
corbo::gui::ScopeWidget::addMeasurement
void addMeasurement(const QString &key, Measurement::ConstPtr measurement, const SignalHelper::SignalData &data)
Definition: scope_widget.cpp:406
corbo::gui::ScopeWidget::mouseMoveEvent
void mouseMoveEvent(QMouseEvent *event) override
Definition: scope_widget.cpp:196
QCPStatisticalBox
A plottable representing a single statistical box in a plot.
Definition: qcustomplot.h:5613
QCPGraph::setData
void setData(QSharedPointer< QCPGraphDataContainer > data)
Definition: qcustomplot.cpp:20142
corbo::gui::ScopeWidget::_drag_start_pos
ResizeDragStratPositions _drag_start_pos
Definition: scope_widget.h:194
corbo::gui::SearchType::Exact
@ Exact
corbo::gui::ScopeWidget::setupLegend
void setupLegend()
Definition: scope_widget.cpp:130
corbo::gui::ScopeWidget::_resize_drag_start_position
QPoint _resize_drag_start_position
Definition: scope_widget.h:195
QCustomPlot::graphCount
int graphCount() const
Definition: qcustomplot.cpp:13777
corbo::gui::ScopeWidget::hasSignal
bool hasSignal(const QString &key, int value_idx) const
Definition: scope_widget.cpp:602
QCPLayoutGrid::foColumnsFirst
@ foColumnsFirst
Columns are filled first, and a new element is wrapped to the next row if the column count would exce...
Definition: qcustomplot.h:1350
corbo::gui::ScopeWidget::scopeMouseWheel
void scopeMouseWheel()
Definition: scope_widget.cpp:705
corbo::gui::ScopeWidget::updateTimeSeriesGraph
void updateTimeSeriesGraph(SignalData &data, double t, double value, bool enlarge_axis, bool replot=true)
Definition: scope_widget.cpp:347
QCPGraph::setLineStyle
void setLineStyle(LineStyle ls)
Definition: qcustomplot.cpp:20170
QCPLayerable::setVisible
void setVisible(bool on)
Definition: qcustomplot.cpp:1370
QCPAxisRect::setRangeZoom
void setRangeZoom(Qt::Orientations orientations)
Definition: qcustomplot.cpp:17512
QCustomPlot::selectedGraphs
QList< QCPGraph * > selectedGraphs() const
Definition: qcustomplot.cpp:13790
corbo::gui::util::SIGNAL_NAMESPACE_SUFFIX_DELIMITER
constexpr const char SIGNAL_NAMESPACE_SUFFIX_DELIMITER[]
Definition: gui/include/corbo-gui/utilities.h:94
corbo::gui::ScopeWidget::addSignal
void addSignal(const QString &signal_key, int value_idx)
Definition: scope_widget.cpp:244
QCPLegend::selectTest
virtual double selectTest(const QPointF &pos, bool onlySelectable, QVariant *details=0) const Q_DECL_OVERRIDE
Definition: qcustomplot.cpp:18815
corbo::gui::ScopeWidget::sizeHint
QSize sizeHint() const override
Definition: scope_widget.cpp:128
corbo::TimeSeries::Ptr
std::shared_ptr< TimeSeries > Ptr
Definition: time_series.h:108
corbo::gui::ScopeWidget::bottom
@ bottom
Definition: scope_widget.h:117
corbo::ok
bool ok()
global method to check whether to proceed or cancel the current action
Definition: global.cpp:54
QCPLayoutGrid::setColumnStretchFactor
void setColumnStretchFactor(int column, double factor)
Definition: qcustomplot.cpp:4216
corbo::gui::ScopeWidget::toValueEncodedKey
QString toValueEncodedKey(const QString &key, int value_idx) const
Definition: scope_widget.cpp:659
corbo::gui::ScopeWidget::updateTimeSeriesSequenceGraph
void updateTimeSeriesSequenceGraph(SignalData &data, bool replot=true)
Definition: scope_widget.cpp:460
QCP::iRangeZoom
@ iRangeZoom
0x002 Axis ranges are zoomable with the mouse wheel (see QCPAxisRect::setRangeZoom,...
Definition: qcustomplot.h:257
corbo::Measurement::ConstPtr
std::shared_ptr< const Measurement > ConstPtr
Definition: signals.h:197
corbo::gui::SignalHelper::ConstPtr
std::shared_ptr< const SignalHelper > ConstPtr
Definition: signal_helper.h:112
corbo::gui::ScopeWidget::_legend_initialized
bool _legend_initialized
Definition: scope_widget.h:190
corbo::TimeSeries::getTimeFromStart
const double & getTimeFromStart() const
Get time from start (offset to all time stamps in time())
Definition: time_series.h:222
PRINT_ERROR
#define PRINT_ERROR(msg)
Print msg-stream as error msg.
Definition: console.h:173
QCPLayoutGrid::setFillOrder
void setFillOrder(FillOrder order, bool rearrange=true)
Definition: qcustomplot.cpp:4377
QCP::iRangeDrag
@ iRangeDrag
0x001 Axis ranges are draggable (see QCPAxisRect::setRangeDrag, QCPAxisRect::setRangeDragAxes)
Definition: qcustomplot.h:256
corbo::gui::ScopeWidget::fromValueEncodedKey
bool fromValueEncodedKey(const QString &value_encoded_key, QString &key, int &value_idx)
Definition: scope_widget.cpp:664
corbo::gui::ScopeWidget::addTimeSeriesGraph
QCPAbstractPlottable * addTimeSeriesGraph(const TimeSeries &time_series, int value_idx, const QColor &color, const QString &legend_text, bool zero_order_hold, bool replot)
Definition: scope_widget.cpp:305
corbo::gui::SignalHelper::ALL_VALUES
static constexpr const int ALL_VALUES
Definition: signal_helper.h:114
utilities.h


control_box_rst
Author(s): Christoph Rösmann
autogenerated on Wed Mar 2 2022 00:06:10