TAMSVIZ
Visualization and annotation tool for ROS
mainwindow.cpp
1 // TAMSVIZ
2 // (c) 2020 Philipp Ruppel
3 
4 #include "mainwindow.h"
5 
6 #include "../core/bagplayer.h"
7 #include "../core/history.h"
8 #include "../core/log.h"
9 #include "../core/profiler.h"
10 #include "../core/serialization.h"
11 #include "../core/snapshot.h"
12 #include "../core/workspace.h"
13 #include "displaytree.h"
14 #include "imagewindow.h"
15 #include "propertygrid.h"
16 #include "renderwindow.h"
17 #include "scenewindow.h"
18 #include "searchwidget.h"
19 #include "splitwindow.h"
20 #include "timeline.h"
21 
22 #include <ros/package.h>
23 #include <ros/ros.h>
24 
25 #include <signal.h>
26 
28  PROPERTY(std::shared_ptr<Object>, object);
29  PROPERTY(Handle<Object>, parent);
30 };
31 DECLARE_TYPE(ClipboardItem, Object)
32 
34  PROPERTY(std::vector<std::shared_ptr<ClipboardItem>>, clipboard);
35 };
36 DECLARE_TYPE(ClipboardData, Object)
37 
38 bool MainWindow::event(QEvent *event) { QMainWindow::event(event); }
39 
40 void MainWindow::addRecentFile(const QString &path) {
41  {
42  QSettings settings;
43  QStringList files = settings.value("recent").toStringList();
44  files.removeAll(path);
45  files.prepend(path);
46  while (files.size() > 10) {
47  files.removeLast();
48  }
49  settings.setValue("recent", files);
50  }
51  updateRecentMenu();
52 }
53 
54 void MainWindow::updateRecentMenu() {
55  QSettings settings;
56  QStringList files = settings.value("recent").toStringList();
57  open_recent_menu->clear();
58  open_recent_menu->setEnabled(!files.isEmpty());
59  for (const QString &file : files) {
60  QObject::connect(open_recent_menu->addAction(QFileInfo(file).fileName()),
61  &QAction::triggered, this,
62  [this, file](bool checked) { openAny(file); });
63  }
64  open_recent_menu->addSeparator();
65  QObject::connect(
66  open_recent_menu->addAction(tr("Clear List")), &QAction::triggered, this,
67  [this](bool checked) {
68  if (QMessageBox::question(
69  this, tr("Clear recent file list?"),
70  tr("Do you want to clear the list of recent files?")) ==
71  QMessageBox::Yes) {
72  QSettings settings;
73  settings.setValue("recent", QStringList());
74  updateRecentMenu();
75  }
76  });
77 }
78 
79 void MainWindow::openDocument(const QString &path) {
80  if (!closeDocument()) {
81  return;
82  }
83  LockScope ws;
84  QFile file(path);
85  if (!file.open(QIODevice::ReadOnly)) {
86  QMessageBox::critical(this, "Error", "Failed to open file.");
87  return;
88  }
89  auto contents = file.readAll().toStdString();
90  Variant v;
91  try {
92  v = parseYAML(contents);
93  } catch (const std::exception &ex) {
94  QMessageBox::critical(
95  this, "Error",
96  tr("Failed to load file. Parsing error.\n%1").arg(ex.what()));
97  return;
98  }
99  std::shared_ptr<Document> displays;
100  while (true) {
101  try {
102  deserialize(displays, v);
103  } catch (const SerializationTypeError &ex) {
104  auto button = QMessageBox::critical(
105  this, "Error",
106  tr("Type not found: %1\nIncompatible version or missing plug-ins?")
107  .arg(ex.typeName().c_str()),
108  QMessageBox::Ignore | QMessageBox::Cancel);
109  if (button == QMessageBox::Ignore) {
110  ex.createReplacement();
111  continue;
112  } else {
113  return;
114  }
115  } catch (const std::exception &ex) {
116  QMessageBox::critical(
117  this, "Error",
118  tr("Failed to load file. Deserialization error.\n%1").arg(ex.what()));
119  return;
120  }
121  break;
122  }
123  displays->path = path.toStdString();
124  ws->document() = displays;
125  ws->saved_document = Snapshot<std::shared_ptr<Document>>::save(
126  ws->document(), ws->saved_document, nullptr);
127  ws->history->clear();
128  ws->modified();
129  addRecentFile(path);
130  LOG_SUCCESS("opened " << displays->path);
131 }
132 
133 void MainWindow::findAndOpenBag(const std::string &name) {
134  LockScope ws;
135  if (ws->player && ws->player->fileName() == name) {
136  return;
137  }
138  {
139  QFileInfo f(QFileInfo(QString::fromStdString(ws->document()->path)).dir(),
140  QString::fromStdString(name));
141  if (f.exists()) {
142  openBag(f.absoluteFilePath());
143  return;
144  }
145  }
146  if (auto player = ws->player) {
147  QFileInfo f(QFileInfo(QString::fromStdString(player->path())).dir(),
148  QString::fromStdString(name));
149  if (f.exists()) {
150  openBag(f.absoluteFilePath());
151  return;
152  }
153  }
154  {
155  QString path = QFileDialog::getOpenFileName(this, tr("Locate Bag File"),
156  QString::fromStdString(name),
157  tr("Bags (*.bag)"));
158  if (path.isNull()) {
159  return;
160  }
161  openBag(path);
162  return;
163  }
164 }
165 
166 void MainWindow::openBrowse() {
167  LockScope ws;
168  QString path = QFileDialog::getOpenFileName(
169  this, tr("Open File"), QString(),
170  tr("Supported file types (*.tmv *.bag);;Documents (*.tmv);;Bags "
171  "(*.bag)"));
172  if (path.isNull()) {
173  return;
174  }
175  openAny(path);
176 }
177 
178 void MainWindow::openAny(const QString &path) {
179  if (path.toLower().endsWith(".bag")) {
180  openBag(path);
181  return;
182  }
183  if (path.toLower().endsWith(".tmv")) {
184  if (!closeDocument()) {
185  return;
186  }
187  openDocument(path);
188  return;
189  }
190  LOG_ERROR("unknown file type");
191  QMessageBox::critical(this, "Error", tr("Unknown file type"));
192 }
193 
194 bool MainWindow::closeDocument() {
195  LockScope ws;
196  if (ws->saved_document != Snapshot<std::shared_ptr<Document>>::save(
197  ws->document(), ws->saved_document, nullptr)) {
198  auto rs = QMessageBox::question(
199  this, tr("Save changes?"),
200  tr("The current document has been modified. Do "
201  "you want to save your changes?"),
202  QMessageBox::Yes | QMessageBox::No | QMessageBox::Cancel,
203  QMessageBox::Cancel);
204  if (rs == QMessageBox::Cancel) {
205  return false;
206  }
207  if (rs == QMessageBox::No) {
208  // continue
209  }
210  if (rs == QMessageBox::Yes) {
211  if (!saveDocument()) {
212  return false;
213  }
214  }
215  }
216  ws->document() = std::make_shared<Document>();
217  ws->document()->window() = std::make_shared<SceneWindow>();
218  ws->saved_document = Snapshot<std::shared_ptr<Document>>::save(
219  ws->document(), ws->saved_document, nullptr);
220  ws->history->clear();
221  ws->modified();
222  LOG_DEBUG("closed");
223  return true;
224 }
225 
226 void MainWindow::openBag(const QString &path) {
227  LockScope ws;
228  QProgressDialog progress(tr("Loading bag..."), QString(), 0, 0);
229  progress.setModal(true);
230  progress.setWindowFlags(Qt::Window | Qt::WindowTitleHint |
231  Qt::CustomizeWindowHint);
232  progress.show();
233  std::string error;
234  volatile bool finished = false;
235  std::shared_ptr<BagPlayer> player;
236  std::thread thread([&]() {
237  try {
238  player = std::make_shared<BagPlayer>(path.toStdString());
239  } catch (const std::exception &ex) {
240  LOG_ERROR("failed to load bag " << ex.what());
241  error = ex.what();
242  }
243  if (player) {
244  LOG_DEBUG("bag loaded");
245  player->changed.connect([]() { GlobalEvents::instance()->redraw(); });
246  }
247  LOG_DEBUG("bag loader thread finished");
248  finished = true;
249  });
250  while (!finished) {
251  QThread::msleep(10);
252  QApplication::processEvents();
253  }
254  thread.join();
255  progress.close();
256  if (player) {
257  ws->player = player;
258  ws->modified();
259  addRecentFile(path);
260  player->rewind();
261  } else {
262  progress.close();
263  LOG_ERROR("failed to load bag file " << error);
264  QMessageBox::critical(this, "Error",
265  tr("Failed to load file. Deserialization error.\n%1")
266  .arg(error.c_str()));
267  }
268 }
269 
270 bool MainWindow::closeBag() {
271  LockScope ws;
272  if (!ws->player) {
273  return true;
274  }
275  ws->player = nullptr;
276  ws->modified();
277  return true;
278 }
279 
280 bool MainWindow::saveDocument(const QString &path) {
281  LockScope ws;
282  QSaveFile file(path);
283  file.open(QIODevice::WriteOnly);
284  file.write(toYAML(serialize(ws->document())).c_str());
285  if (file.commit()) {
286  ws->document()->path = path.toStdString();
287  ws->saved_document = Snapshot<std::shared_ptr<Document>>::save(
288  ws->document(), ws->saved_document, nullptr);
289  LOG_SUCCESS("saved " << path.toStdString());
290  addRecentFile(path);
291  ws->modified();
292  return true;
293  } else {
294  QMessageBox::critical(this, tr("Error"), tr("Failed to write file"));
295  return false;
296  }
297 }
298 
299 bool MainWindow::saveDocument() {
300  LockScope ws;
301  if (ws->document()->path.size()) {
302  return saveDocument(ws->document()->path.c_str());
303  } else {
304  return saveDocumentAs();
305  }
306 }
307 
308 bool MainWindow::saveDocumentAs() {
309  LockScope ws;
310  QFileDialog dialog(this, QString(), QString(), tr("Documents (*.tmv)"));
311  dialog.selectFile(ws->document()->path.c_str());
312  dialog.setAcceptMode(QFileDialog::AcceptSave);
313  dialog.setDefaultSuffix(tr(".tmv"));
314  if (dialog.exec() == QDialog::Accepted) {
315  QString fname = dialog.selectedFiles().value(0);
316  return saveDocument(fname);
317  } else {
318  return false;
319  }
320 }
321 
322 MainWindow *g_main_window_instance = nullptr;
323 MainWindow *MainWindow::instance() { return g_main_window_instance; }
324 
325 MainWindow::MainWindow() {
326 
327  g_main_window_instance = this;
328 
329  LockScope ws;
330 
331  setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea);
332  setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);
333  setCorner(Qt::TopLeftCorner, Qt::LeftDockWidgetArea);
334  setCorner(Qt::TopRightCorner, Qt::RightDockWidgetArea);
335  // setTabPosition(Qt::BottomDockWidgetArea, QTabWidget::East);
336  auto *display_tree = new DisplayTreeWidget();
337  auto *property_grid = new PropertyGridWidget();
338  auto *timeline_widget = new TimelineWidget();
339  addDockWidget(Qt::LeftDockWidgetArea, display_tree);
340  addDockWidget(Qt::LeftDockWidgetArea, property_grid);
341  addDockWidget(Qt::BottomDockWidgetArea, timeline_widget);
342  ws->modified.connect(this, [this]() {
343  LockScope ws;
344  /*
345  if (ws->document()->path.empty()) {
346  setWindowTitle(QString("%1").arg(qApp->applicationName()));
347  } else {
348  setWindowTitle(
349  QString("%1 - %2")
350  .arg(QFileInfo(ws->document()->path.c_str()).fileName())
351  .arg(qApp->applicationName()));
352  }
353  */
354  QString title;
355  if (!ws->document()->path.empty()) {
356  title =
357  QFileInfo(QString::fromStdString(ws->document()->path)).fileName();
358  }
359  if (ws->player) {
360  if (!title.isEmpty()) {
361  title += " - ";
362  }
363  title += QString::fromStdString(ws->player->fileName());
364  }
365  if (!title.isEmpty()) {
366  title += " - ";
367  }
368  title += qApp->applicationName();
369  setWindowTitle(title);
370  });
371 
372  menuBar()->setNativeMenuBar(false);
373  auto *toolbar = addToolBar(tr("Tools"));
374  toolbar->setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
375  auto setupAction = [&](QAction *item, const std::function<void()> &callback,
376  const std::function<bool()> &predicate) {
377  if (callback) {
378  QObject::connect(item, &QAction::triggered, this,
379  [item, callback](bool checked) { callback(); });
380  }
381  if (predicate) {
382  ws->modified.connect(
383  item, [item, predicate]() { item->setEnabled(predicate()); });
384  item->setEnabled(predicate());
385  }
386  };
387  auto createMenuItem = [&](QMenu *menu, const char *label,
388  const std::function<void()> &callback = nullptr,
389  const std::function<bool()> &predicate = nullptr) {
390  auto *item = menu->addAction(tr(label));
391  setupAction(item, callback, predicate);
392  return item;
393  };
394  auto createToolbarItem = [&](const char *label, QIcon icon,
395  const std::function<void()> &callback = nullptr,
396  const std::function<bool()> &predicate =
397  nullptr) {
398  auto *item = toolbar->addAction(tr(label));
399  item->setIcon(icon);
400  setupAction(item, callback, predicate);
401  return item;
402  };
403  auto createMenuAndToolbarItem =
404  [&](QMenu *menu, const char *label, QIcon icon,
405  const std::function<void()> &callback = nullptr,
406  const std::function<bool()> &predicate = nullptr) {
407  createToolbarItem(label, icon, callback, predicate);
408  auto *menu_item = createMenuItem(menu, label, callback, predicate);
409  // menu_item->setIcon(icon);
410  return menu_item;
411  };
412  {
413  auto *menu = menuBar()->addMenu(tr("&File"));
414  createMenuItem(menu, "&New", [this]() { closeDocument(); })
415  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_N));
416  createMenuAndToolbarItem(menu, "&Open", QIcon::fromTheme("document-open"),
417  [this]() { openBrowse(); })
418  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_O));
419  open_recent_menu = menu->addMenu(tr("Open &Recent"));
420  createMenuAndToolbarItem(menu, "&Save", QIcon::fromTheme("document-save"),
421  [this]() { saveDocument(); })
422  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_S));
423  createMenuItem(menu, "Save &As...", [this]() { saveDocumentAs(); })
424  ->setShortcut(QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_S));
425  createMenuItem(menu, "&Close Document", [this]() { closeDocument(); })
426  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_W));
427  createMenuItem(menu, "Close &Bag", [this]() { closeBag(); })
428  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_B));
429  createMenuItem(menu, "E&xit", [this]() { close(); })
430  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_Q));
431  }
432  {
433  auto *menu = menuBar()->addMenu(tr("&Edit"));
434  createMenuAndToolbarItem(menu, "&Undo", // FA_S_ICON("undo", 0.15),
435  QIcon::fromTheme("edit-undo"),
436  [this]() {
437  LockScope ws;
438  ws->history->undo(ws());
439  ws->modified();
440  },
441  []() {
442  LockScope ws;
443  return ws->history->canUndo();
444  })
445  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_Z));
446  createMenuAndToolbarItem(menu, "&Redo", // FA_S_ICON("redo", 0.15),
447  QIcon::fromTheme("edit-redo"),
448  [this]() {
449  LockScope ws;
450  ws->history->redo(ws());
451  ws->modified();
452  },
453  []() {
454  LockScope ws;
455  return ws->history->canRedo();
456  })
457  ->setShortcuts({QKeySequence(Qt::CTRL + Qt::Key_Y),
458  QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_Z)});
459  createMenuAndToolbarItem(
460  menu, "&Copy", // FA_R_ICON("copy"),
461  QIcon::fromTheme("edit-copy"),
462  [this]() {
463  LOG_INFO("copy to clipboard");
464  LockScope ws;
465  QGuiApplication::clipboard()->clear();
466  std::unordered_set<std::shared_ptr<Object>> selection_set;
467  for (auto &o : ws->selection().resolve(ws())) {
468  selection_set.insert(o);
469  }
470  auto clipboard_data = std::make_shared<ClipboardData>();
471  std::unordered_set<std::shared_ptr<Object>> clipboard_set;
472  ws()->recurse([&](const std::shared_ptr<Object> &parent,
473  const std::shared_ptr<Object> &child) {
474  if (clipboard_set.find(parent) != clipboard_set.end()) {
475  clipboard_set.insert(child);
476  } else {
477  if (selection_set.find(child) != selection_set.end()) {
478  clipboard_set.insert(child);
479  auto clipboard_item = std::make_shared<ClipboardItem>();
480  clipboard_item->object() = child;
481  clipboard_item->parent() = parent;
482  clipboard_data->clipboard().push_back(clipboard_item);
483  }
484  }
485  });
486  std::string data =
487  "#" ROS_PACKAGE_NAME "\n" + toYAML(serialize(clipboard_data));
488  QGuiApplication::clipboard()->setText(data.c_str());
489  },
490  []() {
491  LockScope ws;
492  return !ws()->selection().empty();
493  })
494  ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_C));
495  {
496  auto paste_callback = [this]() {
497  if (!QGuiApplication::clipboard()->text().startsWith(
498  "#" ROS_PACKAGE_NAME)) {
499  LOG_ERROR("invalid clipboard data");
500  return;
501  }
502  ActionScope ws("Paste");
503  Variant v;
504  try {
505  v = parseYAML(QGuiApplication::clipboard()->text().toStdString());
506  } catch (const std::exception &ex) {
507  QMessageBox::critical(
508  this, "Error", tr("Clipboard parsing error.\n%1").arg(ex.what()));
509  return;
510  }
511  std::shared_ptr<ClipboardData> clipboard_data;
512  try {
513  deserialize(clipboard_data, v);
514  } catch (const std::exception &ex) {
515  QMessageBox::critical(
516  this, "Error",
517  tr("Clipboard deserialization error.\n%1").arg(ex.what()));
518  return;
519  }
520  if (!clipboard_data) {
521  return;
522  }
523  ws->selection().clear();
524  for (auto &item : clipboard_data->clipboard()) {
525  if (!item || !item->object()) {
526  continue;
527  }
528  auto parent = item->parent().resolve(ws());
529  if (!parent) {
530  continue;
531  }
532  for (auto &property : parent->properties()) {
533  std::vector<std::shared_ptr<void>> list;
534  if (property.info()->type()->tryToPointerList(
535  property.valuePointer(), list)) {
536  item->object()->assignNewId();
537  list.push_back(item->object());
538  if (property.info()->type()->tryFromPointerList(
539  property.valuePointer(), list)) {
540  ws->selection().add(item->object());
541  break;
542  }
543  }
544  }
545  }
546  ws->modified();
547  };
548  auto *paste_menu_item = createMenuItem(menu, "&Paste", paste_callback);
549  auto *paste_toolbar_item =
550  createToolbarItem("Paste",
551  // FA_R_ICON("clipboard"),
552  QIcon::fromTheme("edit-paste"), paste_callback);
553  paste_toolbar_item->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_V));
554  auto update_paste = [paste_menu_item, paste_toolbar_item]() {
555  bool ok = QGuiApplication::clipboard()->text().startsWith(
556  "#" ROS_PACKAGE_NAME);
557  paste_menu_item->setEnabled(ok);
558  paste_toolbar_item->setEnabled(ok);
559  };
560  connect(QGuiApplication::clipboard(), &QClipboard::dataChanged, this,
561  update_paste);
562  update_paste();
563  }
564  createMenuAndToolbarItem(menu, "&Delete", // FA_R_ICON("trash-alt"),
565  QIcon::fromTheme("edit-delete"),
566  [this]() {
567  LOG_INFO("delete");
568  ActionScope ws("Delete");
569  for (auto selected :
570  ws->selection().resolve(ws())) {
571  removeObject(ws(), (void *)selected.get());
572  ws->modified();
573  }
574  },
575  []() {
576  LockScope ws;
577  return !ws()->selection().empty();
578  })
579  ->setShortcut(QKeySequence(Qt::Key_Delete));
580  createMenuItem(menu, "Deselect",
581  [this]() {
582  LockScope ws;
583  ws->selection().clear();
584  ws->modified();
585  if (auto *w = qApp->focusWidget()) {
586  w->clearFocus();
587  }
588  },
589  []() {
590  LockScope ws;
591  return !ws()->selection().empty();
592  })
593  ->setShortcut(QKeySequence(Qt::Key_Escape));
594  /*createMenuItem(menu, "Reload",
595  [this]() { ResourceEvents::instance().reload(); })
596  ->setShortcut(QKeySequence(Qt::Key_F5));*/
597  createMenuAndToolbarItem(menu, "Reload", QIcon::fromTheme("view-refresh"),
598  [this]() {
599  ResourceEvents::instance().reload();
600  GlobalEvents::instance()->redraw();
601  })
602  ->setShortcut(QKeySequence(Qt::Key_F5));
603  }
604  auto *search_widget = new SearchWidget();
605  addDockWidget(Qt::RightDockWidgetArea, search_widget);
606  search_widget->hide();
607  search_widget->toggleViewAction()->setIcon(QIcon::fromTheme("edit-find"));
608  toolbar->addAction(search_widget->toggleViewAction());
609 
610  {
611  auto *menu = menuBar()->addMenu(tr("&Windows"));
612  for (auto *dock : findChildren<QDockWidget *>()) {
613  menu->addAction(dock->toggleViewAction());
614  }
615  // menu->addAction("x")->setShortcut(QKeySequence(Qt::Key_Alt));
616  // menu->addAction("x")->setShortcut(QKeySequence(Qt::Key_AltGr));
617  }
618 
619  {
620  auto update = [this]() {
621  LockScope ws;
622  auto *new_central_widget =
623  ws() && ws->document() && ws->document()->window()
624  ? (ContentWindowBase *)ws->document()->window().get()
625  : nullptr;
626  if (centralWidget() != new_central_widget) {
627  LOG_DEBUG(
628  "changing central widget from "
629  << (centralWidget() ? typeid(*centralWidget()).name() : "null")
630  << " to "
631  << (new_central_widget ? typeid(*new_central_widget).name()
632  : "null"));
633  if (centralWidget()) {
634  QWidget *w = takeCentralWidget();
635  w->setParent(this);
636  w->hide();
637  }
638  setCentralWidget(new_central_widget);
639  new_central_widget->show();
640  }
641  };
642  ws->modified.connect(this, update);
643  update();
644  }
645  updateRecentMenu();
646  resize(std::min(QGuiApplication::primaryScreen()->geometry().width() * 3 / 4,
647  1280),
648  std::min(QGuiApplication::primaryScreen()->geometry().height() * 3 / 4,
649  800));
650  ws->saved_document = Snapshot<std::shared_ptr<Document>>::save(
651  ws->document(), ws->saved_document, nullptr);
652  closeDocument();
653  {
654  move(pos() + (QGuiApplication::primaryScreen()->geometry().center() -
655  geometry().center()));
656  show();
657  qApp->processEvents();
658  display_tree->widget()->setFixedSize(
659  QSize(display_tree->width(), height() / 4));
660  qApp->processEvents();
661  display_tree->widget()->setFixedSize(
662  QSize(QWIDGETSIZE_MAX, QWIDGETSIZE_MAX));
663  }
664 }
665 
666 MainWindow::~MainWindow() {
667  /*if (centralWidget()) {
668  auto *central = takeCentralWidget();
669  central->setParent(nullptr);
670  }*/
671  g_main_window_instance = nullptr;
672 }
673 
674 void MainWindow::closeEvent(QCloseEvent *event) {
675  if (!closeBag()) {
676  event->ignore();
677  return;
678  }
679  if (!closeDocument()) {
680  event->ignore();
681  return;
682  }
683  LockScope ws;
684  event->accept();
685  /*if (ws->history->save_counter != 0) {
686  event->ignore();
687  } else {
688  event->accept();
689 }*/
690 }
691 
692 int main(int argc, char **argv) {
693 
694  {
695  QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts);
696  {
697  QSurfaceFormat format;
698  format.setVersion(3, 2);
699  format.setProfile(QSurfaceFormat::CompatibilityProfile);
700  format.setSamples(16);
701  QSurfaceFormat::setDefaultFormat(format);
702  }
703 
704  ros::init(argc, argv, ROS_PACKAGE_NAME,
705  ros::init_options::AnonymousName |
706  ros::init_options::NoSigintHandler);
707 
708  QApplication app(argc, argv);
709  app.setOrganizationName("TAMS");
710  app.setApplicationName(QString(ROS_PACKAGE_NAME).toUpper());
711 
712  QCommandLineParser parser;
713  parser.setApplicationDescription(
714  "TAMSVIZ - Visualization and annotation tool for ROS");
715  parser.addHelpOption();
716 
717  parser.addPositionalArgument("files", "Bag and visualization files to open",
718  "[files...]");
719 
720  QCommandLineOption opt_profiler("profiler", "Run with profiler enabled");
721  parser.addOption(opt_profiler);
722 
723  QCommandLineOption opt_maximize("maximize", "Maximize window");
724  parser.addOption(opt_maximize);
725 
726  parser.process(app);
727 
728  ros::NodeHandle node("~");
729  signal(SIGINT, [](int sig) {
730  LOG_DEBUG("shutting down");
731  if (auto *app = qApp) {
732  app->exit();
733  }
734  ros::shutdown();
735  });
736 
737  console_bridge::noOutputHandler();
738 
739  LOG_DEBUG("package name " << ROS_PACKAGE_NAME);
740  LOG_DEBUG("package path " << ros::package::getPath(ROS_PACKAGE_NAME));
741 
742  std::unique_ptr<ProfilerThread> profiler_thread;
743  if (parser.isSet(opt_profiler)) {
744  profiler_thread.reset(new ProfilerThread());
745  }
746 
747  {
748  {
749  RenderThread::instance();
750  MainWindow main_window;
751  // qDebug() << "styles" << QStyleFactory::keys();
752  if (parser.isSet(opt_maximize)) {
753  main_window.showMaximized();
754  }
755  for (size_t i = 0; i < parser.positionalArguments().size(); i++) {
756  QString file = parser.positionalArguments()[i];
757  if (!file.isEmpty()) {
758  qDebug() << "opening file" << file;
759  main_window.openAny(file);
760  }
761  }
762  {
763  ros::AsyncSpinner spinner(0);
764  spinner.start();
765  app.exec();
766  }
767  RenderThread::instance()->stop();
768 
769  // wait for async cleanup from renderthread to be finished
770  qApp->processEvents();
771 
772  {
773  LockScope ws;
774 
775  // windows will be deleted by the document
776  auto widgets = main_window.findChildren<QWidget *>();
777  for (auto *w : widgets) {
778  if (dynamic_cast<Window *>(w)) {
779  w->setParent(nullptr);
780  }
781  }
782 
783  ws()->document() = std::make_shared<Document>();
784  }
785  }
786  {
787  LockScope ws;
788  ws() = nullptr;
789  }
790  }
791  }
792  LOG_DEBUG("shut down");
793 }
Definition: object.h:8
Definition: type.h:117
Definition: variant.h:9