TAMSVIZ
Visualization and annotation tool for ROS
splitwindow.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "splitwindow.h"
5 
6 #include "../core/log.h"
7 #include "../core/workspace.h"
8 #include "mainwindow.h"
9 
10 WindowBase::WindowBase() {
11  LOG_DEBUG("new split window created");
12  setParent(MainWindow::instance());
13 }
14 WindowBase::~WindowBase() { LOG_DEBUG("closing split window"); }
15 void WindowBase::changeEvent(QEvent *event) {
16  if (event->type() == QEvent::ParentChange) {
17  LOG_DEBUG("parent of " << typeid(*this).name() << " changed to "
18  << (parent() != nullptr ? typeid(*parent()).name()
19  : "nullptr"));
20  }
21 }
22 
23 static void replaceWindow(const std::shared_ptr<Object> &old_window,
24  const std::shared_ptr<Window> &new_window) {
25  LockScope ws;
26  std::shared_ptr<Object> parent;
27  ws->recurse(
28  [&](const std::shared_ptr<Object> &p, const std::shared_ptr<Object> &c) {
29  if (c == old_window) {
30  parent = p;
31  }
32  });
33  if (parent) {
34  for (auto &property : parent->properties()) {
35  if (property.typeId() == typeid(std::shared_ptr<Window>)) {
36  if (property.read().value<std::shared_ptr<Window>>() == old_window) {
37  property.assign(Variant(new_window));
38  }
39  }
40  }
41  }
42 }
43 
44 SplitWindowBase::SplitWindowBase(Qt::Orientation orientation)
45  : orientation(orientation) {
46  class Splitter : public QWidget {
47  SplitWindowBase *parent = nullptr;
48  bool aggregate = false;
49 
50  public:
51  Splitter(SplitWindowBase *parent) : QWidget(parent), parent(parent) {}
52  virtual void mousePressEvent(QMouseEvent *event) override {
53  event->accept();
54  aggregate = false;
55  }
56  virtual void mouseMoveEvent(QMouseEvent *event) override {
57  event->accept();
58  if (event->buttons() == Qt::LeftButton) {
59  ActionScope ws("Drag Splitter", nullptr, aggregate);
60  aggregate = true;
61  QPoint pos = parent->mapFromGlobal(event->globalPos());
62  double p = 0.5;
63  if (parent->orientation == Qt::Vertical) {
64  p = (pos.x()) * 1.0 / parent->width();
65  } else {
66  p = (pos.y()) * 1.0 / parent->height();
67  }
68  parent->position() = std::max(0.05, std::min(0.95, p));
69  LOG_DEBUG("splitter position " << parent->position());
70  ws->modified();
71  }
72  }
73  virtual void mouseReleaseEvent(QMouseEvent *event) override {
74  event->accept();
75  }
76  };
77  QBoxLayout *layout = nullptr;
78  Splitter *splitter = new Splitter(this);
79  if (orientation == Qt::Horizontal) {
80  layout = new QVBoxLayout(this);
81  splitter->setCursor(Qt::SplitVCursor);
82  } else {
83  layout = new QHBoxLayout(this);
84  splitter->setCursor(Qt::SplitHCursor);
85  }
86  layout->setContentsMargins(0, 0, 0, 0);
87  layout->setSpacing(0);
88  LockScope ws;
89  a() = std::make_shared<EmptyWindow>();
90  b() = std::make_shared<EmptyWindow>();
91  splitter->setMinimumSize(8, 8);
92  this->splitter = splitter;
93  sync();
94  ws->modified.connect(this, [this, layout, splitter]() { sync(); });
95  layout->addWidget(splitter);
96  setLayout(layout);
97 }
98 SplitWindowBase::~SplitWindowBase() {
99  LOG_DEBUG("~SplitWindowBase");
100  LOG_DEBUG("mainwindow: " << MainWindow::instance());
101  LockScope ws;
102  std::dynamic_pointer_cast<WindowBase>(a())->setParent(MainWindow::instance());
103  std::dynamic_pointer_cast<WindowBase>(b())->setParent(MainWindow::instance());
104 }
105 void SplitWindowBase::sync() {
106  LockScope ws;
107  auto *layout = (QBoxLayout *)this->layout();
108  if (!layout->itemAt(0) ||
109  (dynamic_cast<WindowBase *>(a().get()) != layout->itemAt(0)->widget())) {
110  if (auto *item = layout->takeAt(0)) {
111  item->widget()->setParent(MainWindow::instance());
112  }
113  layout->insertWidget(0, (WindowBase *)a().get());
114  }
115  if (layout->count() != 3) {
116  layout->takeAt(1);
117  layout->insertWidget(1, splitter);
118  }
119  if (!layout->itemAt(2) ||
120  (dynamic_cast<WindowBase *>(b().get()) != layout->itemAt(2)->widget())) {
121  if (auto *item = layout->takeAt(2)) {
122  item->widget()->setParent(MainWindow::instance());
123  }
124  layout->insertWidget(2, (WindowBase *)b().get());
125  }
126  ((WindowBase *)a().get())
127  ->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
128  splitter->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
129  ((WindowBase *)b().get())
130  ->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
131  int range = (orientation == Qt::Vertical) ? width() : height();
132  std::array<int, 3> s;
133  s[0] = (int)std::round(position() * range);
134  s[1] = 0;
135  s[2] = range - s[0];
136  for (size_t i = 0; i < s.size(); i++) {
137  if (layout->stretch(i) != s[i]) {
138  layout->setStretch(i, s[i]);
139  }
140  }
141 }
142 
143 void ContentWindowBase::replace(const std::shared_ptr<Window> &new_window) {
144  LOG_DEBUG("replacing window with " << typeid(*new_window).name());
145  LockScope ws;
146  auto me = std::dynamic_pointer_cast<Window>(this->shared_from_this());
147  auto snapshot = Snapshot<std::shared_ptr<Window>>::save(me, nullptr, nullptr);
148  QTimer::singleShot(0, this, [snapshot, new_window, this]() {
149  ActionScope ws("Change Window Type");
150  auto me = std::dynamic_pointer_cast<Window>(this->shared_from_this());
151  replaceWindow(me, new_window);
152  if (auto split_window =
153  std::dynamic_pointer_cast<SplitWindowBase>(new_window)) {
154  // TODO: handle recursive ids, maybe just use serializer
155  snapshot->apply(split_window->a());
156  snapshot->apply(split_window->b());
157  split_window->b()->assignNewId();
158  }
159  ws->modified();
160  });
161 }
162 ContentWindowBase::ContentWindowBase() {
163  layout = new QVBoxLayout(this);
164  layout->setContentsMargins(0, 0, 0, 0);
165  bar = new QHBoxLayout(this);
166  layout->addLayout(bar);
167  bar->setSpacing(0);
168  layout->setSpacing(0);
169  {
170  auto *button = new FlatButton();
171  button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
172  QTimer::singleShot(0, this, [button, this]() {
173  QMenu *menu = new QMenu(this);
174  auto types = Type::find<Window>()->list();
175  for (auto &type : types) {
176  if (type->typeId() == typeid(SplitWindowHorizontal) ||
177  type->typeId() == typeid(SplitWindowVertical)) {
178  continue;
179  }
180  if (!type->constructable()) {
181  continue;
182  }
183  QString label = type->name().c_str();
184  label = label.replace("Window", "");
185  if (type->typeId() == typeid(*this)) {
186  button->setText(label);
187  }
188  connect(menu->addAction(label), &QAction::triggered, this,
189  [type, this](bool checked) {
190  LockScope ws;
191  replace(type->instantiate<Window>());
192  });
193  }
194  button->setMenu(menu);
195  });
196  bar->addWidget(button);
197  }
198  {
199  class Spacer : public QWidget {
200  ContentWindowBase *_parent = nullptr;
201 
202  public:
203  Spacer(ContentWindowBase *parent) : QWidget(parent), _parent(parent) {}
204 
205  protected:
206  virtual void mousePressEvent(QMouseEvent *event) override {
207  QWidget::mousePressEvent(event);
208  LockScope ws;
209  ws->selection() = _parent->shared_from_this();
210  ws->modified();
211  }
212  };
213  spacer = new Spacer(this);
214  spacer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Minimum);
215  bar->addWidget(spacer);
216  }
217 
218  {
219  auto *button = new FlatButton();
220  button->setIcon(MATERIAL_ICON("border_vertical"));
221  button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
222  connect(button, &QPushButton::clicked, this, [this]() {
223  LockScope ws;
224  replace(std::make_shared<SplitWindowVertical>());
225  });
226  bar->addWidget(button);
227  }
228  {
229  auto *button = new FlatButton();
230  button->setIcon(MATERIAL_ICON("border_horizontal"));
231  button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
232  connect(button, &QPushButton::clicked, this, [this]() {
233  LockScope ws;
234  replace(std::make_shared<SplitWindowHorizontal>());
235  });
236  bar->addWidget(button);
237  }
238  {
239 
240  auto *button = new FlatButton();
241 
242  button->setIcon(style()->standardIcon(QStyle::SP_DockWidgetCloseButton));
243  button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
244  bar->addWidget(button);
245  connect(button, &QAbstractButton::clicked, this, [this]() {
246  QTimer::singleShot(0, this, [this]() {
247 
248  LockScope ws;
249  auto me = std::dynamic_pointer_cast<Window>(this->shared_from_this());
250  std::shared_ptr<Object> window_old;
251  std::shared_ptr<Window> window_new;
252  ws->recurse([&](const std::shared_ptr<Object> &window) {
253  if (auto split = std::dynamic_pointer_cast<SplitWindowBase>(window)) {
254  if (split->a() == me) {
255  window_old = split;
256  window_new = split->b();
257  split->b() = std::make_shared<EmptyWindow>();
258  }
259  if (split->b() == me) {
260  window_old = split;
261  window_new = split->a();
262  split->a() = std::make_shared<EmptyWindow>();
263  }
264  }
265  if (auto parent = std::dynamic_pointer_cast<Document>(window)) {
266  if (typeid(*me) != typeid(EmptyWindow)) {
267  window_old = me;
268  window_new = std::make_shared<EmptyWindow>();
269  }
270  }
271  });
272  if (window_old) {
273  ActionScope action("Close Split Window");
274  replaceWindow(window_old, window_new);
275  ws->modified();
276  }
277 
278  });
279  });
280  }
281  setContentWidget(new QWidget(this));
282  setLayout(layout);
283 }
284 void ContentWindowBase::addToolWidget(QWidget *widget) {
285  widget->setParent(this);
286  bar->insertWidget(bar->indexOf(spacer), widget);
287 }
288 void ContentWindowBase::addToolWidgetRight(QWidget *widget) {
289  widget->setParent(this);
290  bar->insertWidget(bar->indexOf(spacer) + 1, widget);
291 }
292 
293 void ContentWindowBase::setContentWidget(QWidget *widget) {
294  if (content_window) {
295  layout->removeWidget(content_window);
296  delete content_window;
297  }
298  content_window = widget;
299  content_window->setParent(this);
300  content_window->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
301  layout->addWidget(content_window);
302  layout->setStretch(0, 0);
303  layout->setStretch(1, 100);
304 }
305 
306 EmptyWindow::EmptyWindow() {
307  auto *view = new QWidget(this);
308  view->setStyleSheet("background: #000;");
309  setContentWidget(view);
310 }
311 
312 void ContentWindowBase::paintAnnotationHUD(
313  QPainter *painter, const std::shared_ptr<const Type> &type) {
314  if (type != nullptr) {
315  double color = 0;
316  {
317  LockScope ws;
318  if (auto track = ws->currentAnnotationTrack().resolve(ws())) {
319  color = track->color();
320  }
321  }
322  QSize size = painter->window().size();
323  size = QSize(size.width(), size.height());
324  if (type->name() != _annotation_hud_string) {
325  static auto font = []() {
326  QFont font;
327  font.setPixelSize(32);
328  return font;
329  }();
330  _annotation_hud_string = type->name();
331  /*
332  _annotation_hud_text = QStaticText(_annotation_hud_string.c_str());
333  _annotation_hud_text.setPerformanceHint(QStaticText::AggressiveCaching);
334  _annotation_hud_text.prepare(QTransform(), font);
335  */
336  }
337  int padding_x = 8;
338  int padding_y = 4;
339  auto c = QColor::fromHsvF(color, 1, 0.7, 0.9);
340  int frame = 5;
341  // double text_right = padding_x * 2 + _annotation_hud_text.size().width();
342  // double text_bottom = padding_y * 2 +
343  // _annotation_hud_text.size().height();
344  QString str = QString::fromStdString(_annotation_hud_string);
345  double text_right =
346  padding_x * 2 + painter->fontMetrics().boundingRect(str).width();
347  double text_bottom = padding_y * 2 + painter->fontMetrics().height();
348  painter->fillRect(0, 0, text_right, text_bottom, QBrush(c));
349  painter->fillRect(text_right, 0, size.width() - text_right, frame,
350  QBrush(c));
351  painter->fillRect(0, text_bottom, frame, size.height() - text_bottom,
352  QBrush(c));
353  painter->fillRect(size.width() - frame, frame, frame, size.height() - frame,
354  QBrush(c));
355  painter->fillRect(frame, size.height() - frame, size.width() - 2 * frame,
356  frame, QBrush(c));
357  painter->setPen(QPen(QBrush(Qt::white), 0));
358  // painter->drawStaticText(padding_x, padding_y, _annotation_hud_text);
359  painter->drawText(QRectF(0, 0, text_right, text_bottom), Qt::AlignCenter,
360  str);
361  }
362 }
Definition: variant.h:9