scope_collection_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 
26 
27 #include <corbo-core/console.h>
29 #include <corbo-gui/scope_widget.h>
31 #include <QGroupBox>
32 #include <QHBoxLayout>
33 #include <QPushButton>
34 #include <QScrollArea>
35 
36 namespace corbo {
37 namespace gui {
38 
39 ScopeCollectionWidget::ScopeCollectionWidget(SignalHelper::ConstPtr signal_helper, QWidget* parent) : QWidget(parent), _signal_helper(signal_helper)
40 {
41  setSizePolicy(QSizePolicy::Expanding, QSizePolicy::MinimumExpanding);
42 
43  _main_layout = new QVBoxLayout(this);
44  // _layout->setContentsMargins(0, 0, 0, 0);
45  _main_layout->setAlignment(Qt::AlignTop);
46 
47  createMenu();
48  createScopeArea();
49 }
50 
51 ScopeCollectionWidget::~ScopeCollectionWidget() {}
52 
53 QSize ScopeCollectionWidget::sizeHint() const { return QSize(1500, 800); }
54 
55 void ScopeCollectionWidget::createMenu()
56 {
57  QHBoxLayout* menu_layout = new QHBoxLayout;
58 
59  // create button for adding new scopes
60  QPushButton* new_scope_btn = new QPushButton(tr("Add Scope"));
61  new_scope_btn->setMaximumWidth(100);
62 
63  QFont font = new_scope_btn->font();
64  font.setPointSize(10);
65  new_scope_btn->setFont(font);
66  connect(new_scope_btn, &QPushButton::clicked, [this](bool) { addScope(); });
67 
68  menu_layout->addWidget(new_scope_btn);
69 
70  menu_layout->addStretch();
71 
72  // add text field to select the current preview time
73  LabelEditWidget* tfs_edit = new LabelEditWidget(tr("Preview time"), "0");
74  tfs_edit->setMaximumWidth(130);
75  auto tfs_edit_changed = [tfs_edit, this]() {
76  // get values from line edit
77  bool ok;
78  double value = tfs_edit->widgetLineEdit()->text().toDouble(&ok);
79  if (!ok || value < 0)
80  {
81  value = 0;
82  tfs_edit->setText("0");
83  }
84  _current_preview_time = value;
85  emit previewTimeUpdate(_current_preview_time);
86  };
87  connect(tfs_edit->widgetLineEdit(), &QLineEdit::editingFinished, tfs_edit_changed);
88  menu_layout->addWidget(tfs_edit);
89 
90  // add slider widget to provide the user with an oppertunity to scroll through time (e.g. for plotting TimeSeriesSequences)
91  SliderCenterStuck* slider = new SliderCenterStuck(Qt::Horizontal);
92  // slider->setMaximumHeight(10);
93  slider->setMaximumWidth(80);
94  slider->setToCenter();
95  slider->setStuckToCenter(10);
96  connect(slider, &SliderCenterStuck::valueChanged, this, [slider, tfs_edit, this](int value) {
97  int speed = value - slider->getCenter();
98 
99  if (_dial_preview_time_timer)
100  {
101  _dial_preview_time_timer->stop();
102  delete _dial_preview_time_timer;
103  _dial_preview_time_timer = nullptr;
104  }
105  if (speed != 0)
106  {
107  _dial_preview_time_timer = new QTimer(this);
108 
109  double dt = (double)speed * (double)std::abs(speed) * 0.0001; // create quadratic scaling behavior
110  dt = std::trunc(dt * 100) / 100; // only two decimal places
111  connect(_dial_preview_time_timer, &QTimer::timeout, tfs_edit, [tfs_edit, dt, this]() {
112  bool ok;
113  double time = tfs_edit->widgetLineEdit()->text().toDouble(&ok);
114  if (!ok) time = 0;
115  time += dt;
116  if (time < 0)
117  {
118  time = 0;
119  _dial_preview_time_timer->stop();
120  }
121  tfs_edit->setText(QString::number(time));
122  _current_preview_time = time;
123  emit previewTimeUpdate(_current_preview_time);
124  });
125  _dial_preview_time_timer->start(250);
126  }
127  });
128  menu_layout->addWidget(slider);
129 
130  menu_layout->addSpacing(20);
131 
132  // checkbox for remembering signal association
133  QCheckBox* inherit_signals = new QCheckBox(tr("Inherit Signals"));
134  inherit_signals->setChecked(true);
135  inherit_signals->setToolTip(tr("If selected, the signal-to-scope association is remembered for future tasks."));
136  connect(inherit_signals, &QCheckBox::toggled, [this](bool checked) { _inherit_signals = checked; });
137 
138  menu_layout->addWidget(inherit_signals);
139 
140  _main_layout->addLayout(menu_layout);
141 
142  QFrame* hline = new QFrame;
143  hline->setFrameShape(QFrame::HLine);
144  hline->setFrameShadow(QFrame::Sunken);
145  hline->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Maximum);
146  _main_layout->addWidget(hline);
147 }
148 
149 void ScopeCollectionWidget::createScopeArea()
150 {
151  _scope_layout = new QVBoxLayout;
152  _scope_layout->setAlignment(Qt::AlignTop | Qt::AlignLeft);
153  _scope_layout->setContentsMargins(0, 0, 0, 0);
154 
155  // add scroll area for the content
156  QScrollArea* scroll_area = new QScrollArea;
157  // we need to create a group since the scrollarea only works with a singla widget rather than a whole layout
158  QWidget* group = new QWidget;
159  group->setLayout(_scope_layout);
160  // scroll_area->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::MinimumExpanding);
161  scroll_area->setWidgetResizable(true);
162  scroll_area->setWidget(group);
163  _main_layout->addWidget(scroll_area);
164 }
165 
166 void ScopeCollectionWidget::addScope()
167 {
168  if (!_signal_helper) return;
169 
170  ScopeWidget* scope_widget = new ScopeWidget(_signal_helper);
171  scope_widget->setPreviewTime(_current_preview_time);
172 
173  // connect scope signal update
174  connect(this, &ScopeCollectionWidget::scopeMeasurementUpdate, scope_widget,
175  [scope_widget](const QString& key, Measurement::ConstPtr measurement, SignalHelper::SignalData& data, bool /*first*/) {
176  scope_widget->addMeasurement(key, measurement, data);
177  // else
178  // scope_widget->addSignal(key, signal);
179  // TODO(roesmann) we have no value_idx here, since we currently
180  // only support drag and drop insertion
181  });
182  // conect scope signal removal
183  connect(this, &ScopeCollectionWidget::scopeSignalRemoval, scope_widget, &ScopeWidget::removeSignal);
184  // connect scope task update
185  connect(this, &ScopeCollectionWidget::scopeTaskInitialization, scope_widget, &ScopeWidget::initializeTask);
186  // connect scope axes resize
187  connect(this, &ScopeCollectionWidget::requestScopeAxesResize, scope_widget, &ScopeWidget::rescaleAxes);
188  // connect preview time / time from start
189  connect(this, &ScopeCollectionWidget::previewTimeUpdate, scope_widget, &ScopeWidget::setPreviewTime);
190 
191  _scope_layout->addWidget(scope_widget);
192 }
193 
194 void ScopeCollectionWidget::closeAllScopes()
195 {
196  // for (auto widget : _scope_layout->findChildren<QWidget*>(QString(), Qt::FindDirectChildrenOnly))
197  // {
198  // // This does not seem to work right now
199  // widget->deleteLater();
200  // }
201  QLayoutItem* item;
202  while ((item = _scope_layout->takeAt(0)) != nullptr)
203  {
204  delete item->widget();
205  delete item;
206  }
207 }
208 
209 void ScopeCollectionWidget::addMeasurement(const QString& key, Measurement::ConstPtr measurement, SignalHelper::SignalData& signal_data, bool first)
210 {
211  emit scopeMeasurementUpdate(key, measurement, signal_data, first);
212 }
213 
214 void ScopeCollectionWidget::removeSignal(const QString& key, int value_idx) { emit scopeSignalRemoval(key, value_idx); }
215 
216 void ScopeCollectionWidget::initializeTask(int task_id) { emit scopeTaskInitialization(task_id, _inherit_signals); }
217 
218 void ScopeCollectionWidget::resizeScopeAxes() { emit requestScopeAxesResize(); }
219 
220 } // namespace gui
221 } // namespace corbo
corbo::gui::ScopeCollectionWidget::ScopeCollectionWidget
ScopeCollectionWidget(SignalHelper::ConstPtr signal_helper, QWidget *parent=0)
Definition: scope_collection_widget.cpp:83
Eigen::Horizontal
@ Horizontal
Definition: Constants.h:268
corbo
Definition: communication/include/corbo-communication/utilities.h:37
scope_widget.h
console.h
slider_center_stuck_widget.h
scope_collection_widget.h
abs
EIGEN_DEVICE_FUNC const EIGEN_STRONG_INLINE AbsReturnType abs() const
Definition: ArrayCwiseUnaryOps.h:43
label_edit_widget.h
corbo::gui::SignalHelper::SignalData
Definition: signal_helper.h:116
corbo::ok
bool ok()
global method to check whether to proceed or cancel the current action
Definition: global.cpp:54
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
relicense.text
text
Definition: relicense.py:59


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