4 #include "mainwindow.h" 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" 22 #include <ros/package.h> 28 PROPERTY(std::shared_ptr<Object>,
object);
34 PROPERTY(std::vector<std::shared_ptr<ClipboardItem>>, clipboard);
38 bool MainWindow::event(QEvent *event) { QMainWindow::event(event); }
40 void MainWindow::addRecentFile(
const QString &path) {
43 QStringList files = settings.value(
"recent").toStringList();
44 files.removeAll(path);
46 while (files.size() > 10) {
49 settings.setValue(
"recent", files);
54 void MainWindow::updateRecentMenu() {
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); });
64 open_recent_menu->addSeparator();
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?")) ==
73 settings.setValue(
"recent", QStringList());
79 void MainWindow::openDocument(
const QString &path) {
80 if (!closeDocument()) {
85 if (!file.open(QIODevice::ReadOnly)) {
86 QMessageBox::critical(
this,
"Error",
"Failed to open file.");
89 auto contents = file.readAll().toStdString();
92 v = parseYAML(contents);
93 }
catch (
const std::exception &ex) {
94 QMessageBox::critical(
96 tr(
"Failed to load file. Parsing error.\n%1").arg(ex.what()));
99 std::shared_ptr<Document> displays;
102 deserialize(displays, v);
104 auto button = QMessageBox::critical(
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();
115 }
catch (
const std::exception &ex) {
116 QMessageBox::critical(
118 tr(
"Failed to load file. Deserialization error.\n%1").arg(ex.what()));
123 displays->path = path.toStdString();
124 ws->document() = displays;
126 ws->document(), ws->saved_document,
nullptr);
127 ws->history->clear();
130 LOG_SUCCESS(
"opened " << displays->path);
133 void MainWindow::findAndOpenBag(
const std::string &name) {
135 if (ws->player && ws->player->fileName() == name) {
139 QFileInfo f(QFileInfo(QString::fromStdString(ws->document()->path)).dir(),
140 QString::fromStdString(name));
142 openBag(f.absoluteFilePath());
146 if (
auto player = ws->player) {
147 QFileInfo f(QFileInfo(QString::fromStdString(player->path())).dir(),
148 QString::fromStdString(name));
150 openBag(f.absoluteFilePath());
155 QString path = QFileDialog::getOpenFileName(
this, tr(
"Locate Bag File"),
156 QString::fromStdString(name),
166 void MainWindow::openBrowse() {
168 QString path = QFileDialog::getOpenFileName(
169 this, tr(
"Open File"), QString(),
170 tr(
"Supported file types (*.tmv *.bag);;Documents (*.tmv);;Bags " 178 void MainWindow::openAny(
const QString &path) {
179 if (path.toLower().endsWith(
".bag")) {
183 if (path.toLower().endsWith(
".tmv")) {
184 if (!closeDocument()) {
190 LOG_ERROR(
"unknown file type");
191 QMessageBox::critical(
this,
"Error", tr(
"Unknown file type"));
194 bool MainWindow::closeDocument() {
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) {
207 if (rs == QMessageBox::No) {
210 if (rs == QMessageBox::Yes) {
211 if (!saveDocument()) {
216 ws->document() = std::make_shared<Document>();
217 ws->document()->window() = std::make_shared<SceneWindow>();
219 ws->document(), ws->saved_document,
nullptr);
220 ws->history->clear();
226 void MainWindow::openBag(
const QString &path) {
228 QProgressDialog progress(tr(
"Loading bag..."), QString(), 0, 0);
229 progress.setModal(
true);
230 progress.setWindowFlags(Qt::Window | Qt::WindowTitleHint |
231 Qt::CustomizeWindowHint);
234 volatile bool finished =
false;
235 std::shared_ptr<BagPlayer> player;
236 std::thread thread([&]() {
238 player = std::make_shared<BagPlayer>(path.toStdString());
239 }
catch (
const std::exception &ex) {
240 LOG_ERROR(
"failed to load bag " << ex.what());
244 LOG_DEBUG(
"bag loaded");
245 player->changed.connect([]() { GlobalEvents::instance()->redraw(); });
247 LOG_DEBUG(
"bag loader thread finished");
252 QApplication::processEvents();
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()));
270 bool MainWindow::closeBag() {
275 ws->player =
nullptr;
280 bool MainWindow::saveDocument(
const QString &path) {
282 QSaveFile file(path);
283 file.open(QIODevice::WriteOnly);
284 file.write(toYAML(serialize(ws->document())).c_str());
286 ws->document()->path = path.toStdString();
288 ws->document(), ws->saved_document,
nullptr);
289 LOG_SUCCESS(
"saved " << path.toStdString());
294 QMessageBox::critical(
this, tr(
"Error"), tr(
"Failed to write file"));
299 bool MainWindow::saveDocument() {
301 if (ws->document()->path.size()) {
302 return saveDocument(ws->document()->path.c_str());
304 return saveDocumentAs();
308 bool MainWindow::saveDocumentAs() {
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);
323 MainWindow *MainWindow::instance() {
return g_main_window_instance; }
325 MainWindow::MainWindow() {
327 g_main_window_instance =
this;
331 setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea);
332 setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);
333 setCorner(Qt::TopLeftCorner, Qt::LeftDockWidgetArea);
334 setCorner(Qt::TopRightCorner, Qt::RightDockWidgetArea);
339 addDockWidget(Qt::LeftDockWidgetArea, display_tree);
340 addDockWidget(Qt::LeftDockWidgetArea, property_grid);
341 addDockWidget(Qt::BottomDockWidgetArea, timeline_widget);
342 ws->modified.connect(
this, [
this]() {
355 if (!ws->document()->path.empty()) {
357 QFileInfo(QString::fromStdString(ws->document()->path)).fileName();
360 if (!title.isEmpty()) {
363 title += QString::fromStdString(ws->player->fileName());
365 if (!title.isEmpty()) {
368 title += qApp->applicationName();
369 setWindowTitle(title);
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) {
378 QObject::connect(item, &QAction::triggered,
this,
379 [item, callback](
bool checked) { callback(); });
382 ws->modified.connect(
383 item, [item, predicate]() { item->setEnabled(predicate()); });
384 item->setEnabled(predicate());
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);
394 auto createToolbarItem = [&](
const char *label, QIcon icon,
395 const std::function<void()> &callback =
nullptr,
396 const std::function<bool()> &predicate =
398 auto *item = toolbar->addAction(tr(label));
400 setupAction(item, callback, predicate);
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);
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));
433 auto *menu = menuBar()->addMenu(tr(
"&Edit"));
434 createMenuAndToolbarItem(menu,
"&Undo",
435 QIcon::fromTheme(
"edit-undo"),
438 ws->history->undo(ws());
443 return ws->history->canUndo();
445 ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_Z));
446 createMenuAndToolbarItem(menu,
"&Redo",
447 QIcon::fromTheme(
"edit-redo"),
450 ws->history->redo(ws());
455 return ws->history->canRedo();
457 ->setShortcuts({QKeySequence(Qt::CTRL + Qt::Key_Y),
458 QKeySequence(Qt::CTRL + Qt::SHIFT + Qt::Key_Z)});
459 createMenuAndToolbarItem(
461 QIcon::fromTheme(
"edit-copy"),
463 LOG_INFO(
"copy to clipboard");
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);
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);
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);
487 "#" ROS_PACKAGE_NAME
"\n" + toYAML(serialize(clipboard_data));
488 QGuiApplication::clipboard()->setText(data.c_str());
492 return !ws()->selection().empty();
494 ->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_C));
496 auto paste_callback = [
this]() {
497 if (!QGuiApplication::clipboard()->text().startsWith(
498 "#" ROS_PACKAGE_NAME)) {
499 LOG_ERROR(
"invalid clipboard data");
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()));
511 std::shared_ptr<ClipboardData> clipboard_data;
513 deserialize(clipboard_data, v);
514 }
catch (
const std::exception &ex) {
515 QMessageBox::critical(
517 tr(
"Clipboard deserialization error.\n%1").arg(ex.what()));
520 if (!clipboard_data) {
523 ws->selection().clear();
524 for (
auto &item : clipboard_data->clipboard()) {
525 if (!item || !item->object()) {
528 auto parent = item->parent().resolve(ws());
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());
548 auto *paste_menu_item = createMenuItem(menu,
"&Paste", paste_callback);
549 auto *paste_toolbar_item =
550 createToolbarItem(
"Paste",
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);
560 connect(QGuiApplication::clipboard(), &QClipboard::dataChanged,
this,
564 createMenuAndToolbarItem(menu,
"&Delete",
565 QIcon::fromTheme(
"edit-delete"),
570 ws->selection().resolve(ws())) {
571 removeObject(ws(), (
void *)selected.get());
577 return !ws()->selection().empty();
579 ->setShortcut(QKeySequence(Qt::Key_Delete));
580 createMenuItem(menu,
"Deselect",
583 ws->selection().clear();
585 if (
auto *w = qApp->focusWidget()) {
591 return !ws()->selection().empty();
593 ->setShortcut(QKeySequence(Qt::Key_Escape));
597 createMenuAndToolbarItem(menu,
"Reload", QIcon::fromTheme(
"view-refresh"),
599 ResourceEvents::instance().reload();
600 GlobalEvents::instance()->redraw();
602 ->setShortcut(QKeySequence(Qt::Key_F5));
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());
611 auto *menu = menuBar()->addMenu(tr(
"&Windows"));
612 for (
auto *dock : findChildren<QDockWidget *>()) {
613 menu->addAction(dock->toggleViewAction());
620 auto update = [
this]() {
622 auto *new_central_widget =
623 ws() && ws->document() && ws->document()->window()
626 if (centralWidget() != new_central_widget) {
628 "changing central widget from " 629 << (centralWidget() ?
typeid(*centralWidget()).name() :
"null")
631 << (new_central_widget ?
typeid(*new_central_widget).name()
633 if (centralWidget()) {
634 QWidget *w = takeCentralWidget();
638 setCentralWidget(new_central_widget);
639 new_central_widget->show();
642 ws->modified.connect(
this, update);
646 resize(std::min(QGuiApplication::primaryScreen()->geometry().width() * 3 / 4,
648 std::min(QGuiApplication::primaryScreen()->geometry().height() * 3 / 4,
651 ws->document(), ws->saved_document,
nullptr);
654 move(pos() + (QGuiApplication::primaryScreen()->geometry().center() -
655 geometry().center()));
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));
666 MainWindow::~MainWindow() {
671 g_main_window_instance =
nullptr;
674 void MainWindow::closeEvent(QCloseEvent *event) {
679 if (!closeDocument()) {
692 int main(
int argc,
char **argv) {
695 QCoreApplication::setAttribute(Qt::AA_ShareOpenGLContexts);
697 QSurfaceFormat format;
698 format.setVersion(3, 2);
699 format.setProfile(QSurfaceFormat::CompatibilityProfile);
700 format.setSamples(16);
701 QSurfaceFormat::setDefaultFormat(format);
704 ros::init(argc, argv, ROS_PACKAGE_NAME,
705 ros::init_options::AnonymousName |
706 ros::init_options::NoSigintHandler);
708 QApplication app(argc, argv);
709 app.setOrganizationName(
"TAMS");
710 app.setApplicationName(QString(ROS_PACKAGE_NAME).toUpper());
712 QCommandLineParser parser;
713 parser.setApplicationDescription(
714 "TAMSVIZ - Visualization and annotation tool for ROS");
715 parser.addHelpOption();
717 parser.addPositionalArgument(
"files",
"Bag and visualization files to open",
720 QCommandLineOption opt_profiler(
"profiler",
"Run with profiler enabled");
721 parser.addOption(opt_profiler);
723 QCommandLineOption opt_maximize(
"maximize",
"Maximize window");
724 parser.addOption(opt_maximize);
728 ros::NodeHandle node(
"~");
729 signal(SIGINT, [](
int sig) {
730 LOG_DEBUG(
"shutting down");
731 if (
auto *app = qApp) {
737 console_bridge::noOutputHandler();
739 LOG_DEBUG(
"package name " << ROS_PACKAGE_NAME);
740 LOG_DEBUG(
"package path " << ros::package::getPath(ROS_PACKAGE_NAME));
742 std::unique_ptr<ProfilerThread> profiler_thread;
743 if (parser.isSet(opt_profiler)) {
749 RenderThread::instance();
752 if (parser.isSet(opt_maximize)) {
753 main_window.showMaximized();
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);
763 ros::AsyncSpinner spinner(0);
767 RenderThread::instance()->stop();
770 qApp->processEvents();
776 auto widgets = main_window.findChildren<QWidget *>();
777 for (
auto *w : widgets) {
778 if (dynamic_cast<Window *>(w)) {
779 w->setParent(
nullptr);
783 ws()->document() = std::make_shared<Document>();
792 LOG_DEBUG(
"shut down");