diff --git a/noether_gui/CMakeLists.txt b/noether_gui/CMakeLists.txt index 954a1023..eea10c7e 100644 --- a/noether_gui/CMakeLists.txt +++ b/noether_gui/CMakeLists.txt @@ -20,6 +20,7 @@ option(NOETHER_ENABLE_TESTING "Enables compilation of unit tests" OFF) qt5_wrap_cpp(${PROJECT_NAME}_widget_mocs include/${PROJECT_NAME}/widgets/collapsible_area_widget.h include/${PROJECT_NAME}/widgets/tpp_pipeline_widget.h + include/${PROJECT_NAME}/widgets/configurable_tpp_pipeline_widget.h include/${PROJECT_NAME}/widgets/tpp_widget.h # Tool Path Planners # Raster @@ -52,6 +53,7 @@ qt5_wrap_ui(${PROJECT_NAME}_widget_ui_mocs ui/linear_approach_modifier_widget.ui ui/raster_planner_widget.ui ui/tpp_pipeline_widget.ui + ui/configurable_tpp_pipeline_widget.ui ui/tpp_widget.ui) include_directories(include ${CMAKE_CURRENT_BINARY_DIR}) @@ -63,6 +65,7 @@ add_library(${PROJECT_NAME} SHARED src/widgets/collapsible_area_widget.cpp src/widgets/plugin_loader_widget.cpp src/widgets/tpp_pipeline_widget.cpp + src/widgets/configurable_tpp_pipeline_widget.cpp src/widgets/tpp_widget.cpp # Tool Path Planners # Raster Planner diff --git a/noether_gui/include/noether_gui/widgets/configurable_tpp_pipeline_widget.h b/noether_gui/include/noether_gui/widgets/configurable_tpp_pipeline_widget.h new file mode 100644 index 00000000..e389040e --- /dev/null +++ b/noether_gui/include/noether_gui/widgets/configurable_tpp_pipeline_widget.h @@ -0,0 +1,42 @@ +#pragma once + +#include +#include +#include + +namespace Ui +{ +class ConfigurableTPPPipeline; +} + +namespace YAML +{ +class Node; +} + +namespace noether +{ +class TPPPipelineWidget; + +class ConfigurableTPPPipelineWidget : public QWidget +{ + Q_OBJECT +public: + ConfigurableTPPPipelineWidget(boost_plugin_loader::PluginLoader loader, QWidget* parent = nullptr); + + void setConfigurationFile(const QString& file); + + ToolPathPlannerPipeline createPipeline() const; + void configure(const YAML::Node& config); + void configure(const QString& file); + void save(YAML::Node& config) const; + +private: + void onLoadConfiguration(const bool /*checked*/); + void onSaveConfiguration(const bool /*checked*/); + + Ui::ConfigurableTPPPipeline* ui_; + TPPPipelineWidget* pipeline_widget_; +}; + +} // namespace noether diff --git a/noether_gui/include/noether_gui/widgets/plugin_loader_widget.h b/noether_gui/include/noether_gui/widgets/plugin_loader_widget.h index 0f8734be..e53e86a8 100644 --- a/noether_gui/include/noether_gui/widgets/plugin_loader_widget.h +++ b/noether_gui/include/noether_gui/widgets/plugin_loader_widget.h @@ -13,6 +13,8 @@ namespace YAML class Node; } +class QVBoxLayout; + namespace noether { /** @@ -35,6 +37,7 @@ class PluginLoaderWidget : public QWidget void addWidget(const QString& plugin_name, const YAML::Node& config); Ui::PluginLoader* ui_; + QVBoxLayout* widgets_layout_; const boost_plugin_loader::PluginLoader loader_; }; diff --git a/noether_gui/include/noether_gui/widgets/plugin_loader_widget.hpp b/noether_gui/include/noether_gui/widgets/plugin_loader_widget.hpp index 0cd86919..6b5f998f 100644 --- a/noether_gui/include/noether_gui/widgets/plugin_loader_widget.hpp +++ b/noether_gui/include/noether_gui/widgets/plugin_loader_widget.hpp @@ -13,10 +13,12 @@ template PluginLoaderWidget::PluginLoaderWidget(boost_plugin_loader::PluginLoader loader, const QString& title, QWidget* parent) - : QWidget(parent), ui_(new Ui::PluginLoader()), loader_(std::move(loader)) + : QWidget(parent), ui_(new Ui::PluginLoader()), widgets_layout_(new QVBoxLayout()), loader_(std::move(loader)) { ui_->setupUi(this); + ui_->contents->setLayout(widgets_layout_); + ui_->group_box->setTitle(title); ui_->combo_box->addItems(getAvailablePlugins(loader_)); @@ -45,7 +47,7 @@ void PluginLoaderWidget::addWidget(const QString& plugin_name, const YA collapsible_area->setWidget(plugin->create(this, config)); collapsible_area->setContextMenuPolicy(Qt::ContextMenuPolicy::CustomContextMenu); - ui_->vertical_layout_plugins->addWidget(collapsible_area); + widgets_layout_->addWidget(collapsible_area); // Add a right click menu option for deleting this tool path modifier widget connect(collapsible_area, &QWidget::customContextMenuRequested, [this, collapsible_area](const QPoint& pos) { @@ -54,7 +56,7 @@ void PluginLoaderWidget::addWidget(const QString& plugin_name, const YA QAction* action = context_menu.exec(collapsible_area->mapToGlobal(pos)); if (action == remove_action) { - ui_->vertical_layout_plugins->removeWidget(collapsible_area); + widgets_layout_->removeWidget(collapsible_area); delete collapsible_area; } }); @@ -67,9 +69,9 @@ template QWidgetList PluginLoaderWidget::getWidgets() const { QWidgetList widgets; - for (int i = 0; i < ui_->vertical_layout_plugins->count(); ++i) + for (int i = 0; i < widgets_layout_->count(); ++i) { - QLayoutItem* item = ui_->vertical_layout_plugins->itemAt(i); + QLayoutItem* item = widgets_layout_->itemAt(i); if (item) { auto collapsible_area = dynamic_cast(item->widget()); @@ -96,9 +98,9 @@ void PluginLoaderWidget::configure(const YAML::Node& config) template void PluginLoaderWidget::save(YAML::Node& config) const { - for (int i = 0; i < ui_->vertical_layout_plugins->count(); ++i) + for (int i = 0; i < widgets_layout_->count(); ++i) { - QLayoutItem* item = ui_->vertical_layout_plugins->itemAt(i); + QLayoutItem* item = widgets_layout_->itemAt(i); if (item) { auto collapsible_area = dynamic_cast(item->widget()); @@ -121,9 +123,9 @@ void PluginLoaderWidget::save(YAML::Node& config) const template void PluginLoaderWidget::removeWidgets() { - for (int i = 0; i < ui_->vertical_layout_plugins->count(); ++i) + for (int i = 0; i < widgets_layout_->count(); ++i) { - ui_->vertical_layout_plugins->itemAt(i)->widget()->deleteLater(); + widgets_layout_->itemAt(i)->widget()->deleteLater(); } } diff --git a/noether_gui/include/noether_gui/widgets/tpp_widget.h b/noether_gui/include/noether_gui/widgets/tpp_widget.h index 613a91ab..a58cdd63 100644 --- a/noether_gui/include/noether_gui/widgets/tpp_widget.h +++ b/noether_gui/include/noether_gui/widgets/tpp_widget.h @@ -26,7 +26,7 @@ class TPP; namespace noether { -class TPPPipelineWidget; +class ConfigurableTPPPipelineWidget; /** * @brief Basic tool path planning widget @@ -51,8 +51,6 @@ class TPPWidget : public QWidget private: void onLoadMesh(const bool /*checked*/); - void onLoadConfiguration(const bool /*checked*/); - void onSaveConfiguration(const bool /*checked*/); void onPlan(const bool /*checked*/); void onShowOriginalMesh(const bool); void onShowModifiedMesh(const bool); @@ -60,7 +58,7 @@ class TPPWidget : public QWidget void onShowModifiedToolPath(const bool); Ui::TPP* ui_; - TPPPipelineWidget* pipeline_widget_; + ConfigurableTPPPipelineWidget* pipeline_widget_; // Viewer rendering QVTKWidget* render_widget_; diff --git a/noether_gui/src/tpp_app.cpp b/noether_gui/src/tpp_app.cpp index 0b787b13..33cb69b3 100644 --- a/noether_gui/src/tpp_app.cpp +++ b/noether_gui/src/tpp_app.cpp @@ -49,7 +49,7 @@ int main(int argc, char** argv) // Set the TPP widget as the central widget and show w.setCentralWidget(widget); - w.show(); + w.showMaximized(); return app.exec(); } diff --git a/noether_gui/src/widgets/collapsible_area_widget.cpp b/noether_gui/src/widgets/collapsible_area_widget.cpp index 6fb93d2e..44752bbd 100644 --- a/noether_gui/src/widgets/collapsible_area_widget.cpp +++ b/noether_gui/src/widgets/collapsible_area_widget.cpp @@ -44,6 +44,8 @@ CollapsibleArea::CollapsibleArea(const QString& label, QWidget* parent) content->setStyleSheet("QScrollArea { border: none; }"); content->setMaximumHeight(0); content->setMinimumHeight(0); + content->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); + content->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); animation_->addAnimation(new QPropertyAnimation(this, "minimumHeight")); animation_->addAnimation(new QPropertyAnimation(this, "maximumHeight")); diff --git a/noether_gui/src/widgets/configurable_tpp_pipeline_widget.cpp b/noether_gui/src/widgets/configurable_tpp_pipeline_widget.cpp new file mode 100644 index 00000000..77805730 --- /dev/null +++ b/noether_gui/src/widgets/configurable_tpp_pipeline_widget.cpp @@ -0,0 +1,123 @@ +#include +#include +#include +#include "ui_configurable_tpp_pipeline_widget.h" + +#include +#include +#include +#include +#include +#include + +namespace noether +{ +ConfigurableTPPPipelineWidget::ConfigurableTPPPipelineWidget(boost_plugin_loader::PluginLoader loader, QWidget* parent) + : QWidget(parent) + , ui_(new Ui::ConfigurableTPPPipeline()) + , pipeline_widget_(new TPPPipelineWidget(std::move(loader), this)) +{ + ui_->setupUi(this); + layout()->addWidget(pipeline_widget_); + + // Connect + connect(ui_->push_button_load, &QPushButton::clicked, this, &ConfigurableTPPPipelineWidget::onLoadConfiguration); + connect(ui_->push_button_save, &QPushButton::clicked, this, &ConfigurableTPPPipelineWidget::onSaveConfiguration); +} + +ToolPathPlannerPipeline ConfigurableTPPPipelineWidget::createPipeline() const +{ + return pipeline_widget_->createPipeline(); +} + +void ConfigurableTPPPipelineWidget::configure(const YAML::Node& config) { return pipeline_widget_->configure(config); } + +void ConfigurableTPPPipelineWidget::configure(const QString& file) +{ + try + { + configure(YAML::LoadFile(file.toStdString())); + ui_->line_edit->setText(file); + } + catch (const YAML::BadFile&) + { + QString message; + QTextStream ss(&message); + ss << "Failed to open YAML file at '" << file << "'"; + QMessageBox::warning(this, "Configuration Error", message); + } + catch (const std::exception& ex) + { + std::stringstream ss; + noether::printException(ex, ss); + QMessageBox::warning(this, "Configuration Error", QString::fromStdString(ss.str())); + } +} + +void ConfigurableTPPPipelineWidget::save(YAML::Node& config) const { return pipeline_widget_->save(config); } + +void ConfigurableTPPPipelineWidget::setConfigurationFile(const QString& file) +{ + ui_->line_edit->setText(file); + + try + { + configure(YAML::LoadFile(file.toStdString())); + } + catch (const YAML::BadFile&) + { + QString message; + QTextStream ss; + ss << "Failed to open YAML file at '" << file << "'"; + QMessageBox::warning(this, "Configuration Error", message); + } + catch (const std::exception& ex) + { + std::stringstream ss; + printException(ex, ss); + QMessageBox::warning(this, "Configuration Error", QString::fromStdString(ss.str())); + } +} + +void ConfigurableTPPPipelineWidget::onLoadConfiguration(const bool /*checked*/) +{ + QString file = ui_->line_edit->text(); + if (file.isEmpty()) + file = QStandardPaths::standardLocations(QStandardPaths::HomeLocation).at(0); + + file = QFileDialog::getOpenFileName(this, "Load configuration file", file, "YAML files (*.yaml)"); + if (!file.isEmpty()) + setConfigurationFile(file); +} + +void ConfigurableTPPPipelineWidget::onSaveConfiguration(const bool /*checked*/) +{ + try + { + QString file = ui_->line_edit->text(); + if (file.isEmpty()) + file = QStandardPaths::standardLocations(QStandardPaths::HomeLocation).at(0); + + file = QFileDialog::getSaveFileName(this, "Save configuration file", file, "YAML files (*.yaml)"); + if (file.isEmpty()) + return; + + YAML::Node config; + save(config); + + std::ofstream ofh(file.toStdString()); + if (!ofh) + throw std::runtime_error("Failed to open output file at '" + file.toStdString() + "'"); + + ofh << config; + QMessageBox::information(this, "Configuration", "Successfully saved tool path planning pipeline configuration"); + } + catch (const std::exception& ex) + { + std::stringstream ss; + printException(ex, ss); + QMessageBox::warning(this, "Save Error", QString::fromStdString(ss.str())); + } +} + +} // namespace noether diff --git a/noether_gui/src/widgets/tpp_pipeline_widget.cpp b/noether_gui/src/widgets/tpp_pipeline_widget.cpp index d6ef56ba..60006b40 100644 --- a/noether_gui/src/widgets/tpp_pipeline_widget.cpp +++ b/noether_gui/src/widgets/tpp_pipeline_widget.cpp @@ -68,12 +68,12 @@ TPPPipelineWidget::TPPPipelineWidget(boost_plugin_loader::PluginLoader loader, Q ui_->group_box_tpp->setTitle(text); if (text.isEmpty()) { - overwriteWidget(ui_->group_box_tpp->layout(), ui_->widget_tpp, new QWidget(this)); + ui_->scroll_area->setWidget(new QWidget(this)); } else { auto plugin = loader_.createInstance(text.toStdString()); - overwriteWidget(ui_->group_box_tpp->layout(), ui_->widget_tpp, plugin->create(this)); + ui_->scroll_area->setWidget(plugin->create(this)); } } catch (const std::exception& ex) @@ -98,7 +98,7 @@ void TPPPipelineWidget::configure(const YAML::Node& config) auto tpp_config = config[TOOL_PATH_PLANNER_KEY]; auto name = getEntry(tpp_config, "name"); auto plugin = loader_.createInstance(name); - overwriteWidget(ui_->group_box_tpp->layout(), ui_->widget_tpp, plugin->create(this, tpp_config)); + ui_->scroll_area->setWidget(plugin->create(this, tpp_config)); ui_->group_box_tpp->setTitle(QString::fromStdString(name)); } catch (const std::exception&) @@ -112,7 +112,7 @@ void TPPPipelineWidget::configure(const YAML::Node& config) catch (const std::exception&) { // Clear the widgets - overwriteWidget(ui_->group_box_tpp->layout(), ui_->widget_tpp, new QWidget(this)); + ui_->scroll_area->setWidget(new QWidget(this)); mesh_modifier_loader_widget_->removeWidgets(); tool_path_modifier_loader_widget_->removeWidgets(); @@ -131,7 +131,7 @@ void TPPPipelineWidget::save(YAML::Node& config) const // Tool path planner { - auto tpp_widget = dynamic_cast(ui_->widget_tpp); + auto tpp_widget = dynamic_cast(ui_->scroll_area->widget()); if (tpp_widget) { YAML::Node tpp_config; @@ -151,7 +151,7 @@ void TPPPipelineWidget::save(YAML::Node& config) const ToolPathPlannerPipeline TPPPipelineWidget::createPipeline() const { - auto widget_tpp = dynamic_cast(ui_->widget_tpp); + auto widget_tpp = dynamic_cast(ui_->scroll_area->widget()); if (!widget_tpp) throw std::runtime_error("No tool path planner specified"); diff --git a/noether_gui/src/widgets/tpp_widget.cpp b/noether_gui/src/widgets/tpp_widget.cpp index c2b12386..08ef5f73 100644 --- a/noether_gui/src/widgets/tpp_widget.cpp +++ b/noether_gui/src/widgets/tpp_widget.cpp @@ -1,5 +1,6 @@ #include #include "ui_tpp_widget.h" +#include #include #include @@ -34,7 +35,7 @@ namespace noether TPPWidget::TPPWidget(boost_plugin_loader::PluginLoader loader, QWidget* parent) : QWidget(parent) , ui_(new Ui::TPP()) - , pipeline_widget_(new TPPPipelineWidget(std::move(loader), this)) + , pipeline_widget_(new ConfigurableTPPPipelineWidget(std::move(loader), this)) , render_widget_(new QVTKWidget(this)) , renderer_(vtkSmartPointer::New()) , mesh_mapper_(vtkSmartPointer::New()) @@ -47,7 +48,7 @@ TPPWidget::TPPWidget(boost_plugin_loader::PluginLoader loader, QWidget* parent) { ui_->setupUi(this); - ui_->scroll_area->setWidget(pipeline_widget_); + overwriteWidget(ui_->group_box_configuration->layout(), ui_->widget, pipeline_widget_); ui_->splitter->addWidget(render_widget_); // Set up the VTK objects @@ -80,8 +81,6 @@ TPPWidget::TPPWidget(boost_plugin_loader::PluginLoader loader, QWidget* parent) // Connect signals connect(ui_->push_button_load_mesh, &QPushButton::clicked, this, &TPPWidget::onLoadMesh); - connect(ui_->push_button_load_configuration, &QPushButton::clicked, this, &TPPWidget::onLoadConfiguration); - connect(ui_->push_button_save_configuration, &QPushButton::clicked, this, &TPPWidget::onSaveConfiguration); connect(ui_->check_box_show_original_mesh, &QCheckBox::clicked, this, &TPPWidget::onShowOriginalMesh); connect(ui_->check_box_show_modified_mesh, &QCheckBox::clicked, this, &TPPWidget::onShowModifiedMesh); connect(ui_->check_box_show_original_tool_path, &QCheckBox::clicked, this, &TPPWidget::onShowUnmodifiedToolPath); @@ -160,69 +159,7 @@ void TPPWidget::onLoadMesh(const bool /*checked*/) setMeshFile(file); } -void TPPWidget::setConfigurationFile(const QString& file) -{ - ui_->line_edit_configuration->setText(file); - - try - { - pipeline_widget_->configure(YAML::LoadFile(file.toStdString())); - } - catch (const YAML::BadFile&) - { - QString message; - QTextStream ss; - ss << "Failed to open YAML file at '" << file << "'"; - QMessageBox::warning(this, "Configuration Error", message); - } - catch (const std::exception& ex) - { - std::stringstream ss; - printException(ex, ss); - QMessageBox::warning(this, "Configuration Error", QString::fromStdString(ss.str())); - } -} - -void TPPWidget::onLoadConfiguration(const bool /*checked*/) -{ - QString file = ui_->line_edit_configuration->text(); - if (file.isEmpty()) - file = QStandardPaths::standardLocations(QStandardPaths::HomeLocation).at(0); - - file = QFileDialog::getOpenFileName(this, "Load configuration file", file, "YAML files (*.yaml)"); - if (!file.isEmpty()) - setConfigurationFile(file); -} - -void TPPWidget::onSaveConfiguration(const bool /*checked*/) -{ - try - { - QString file = ui_->line_edit_configuration->text(); - if (file.isEmpty()) - file = QStandardPaths::standardLocations(QStandardPaths::HomeLocation).at(0); - - file = QFileDialog::getSaveFileName(this, "Save configuration file", file, "YAML files (*.yaml)"); - if (file.isEmpty()) - return; - - YAML::Node config; - pipeline_widget_->save(config); - - std::ofstream ofh(file.toStdString()); - if (!ofh) - throw std::runtime_error("Failed to open output file at '" + file.toStdString() + "'"); - - ofh << config; - QMessageBox::information(this, "Configuration", "Successfully saved tool path planning pipeline configuration"); - } - catch (const std::exception& ex) - { - std::stringstream ss; - printException(ex, ss); - QMessageBox::warning(this, "Save Error", QString::fromStdString(ss.str())); - } -} +void TPPWidget::setConfigurationFile(const QString& file) { pipeline_widget_->setConfigurationFile(file); } vtkSmartPointer toVTK(const Eigen::Isometry3d& mat) { diff --git a/noether_gui/ui/configurable_tpp_pipeline_widget.ui b/noether_gui/ui/configurable_tpp_pipeline_widget.ui new file mode 100644 index 00000000..734ed46a --- /dev/null +++ b/noether_gui/ui/configurable_tpp_pipeline_widget.ui @@ -0,0 +1,49 @@ + + + ConfigurableTPPPipeline + + + + 0 + 0 + 400 + 79 + + + + Form + + + + + + + + + + + Load + + + + + + + + + Save + + + + + + + Qt::Horizontal + + + + + + + + diff --git a/noether_gui/ui/plugin_loader_widget.ui b/noether_gui/ui/plugin_loader_widget.ui index 7a554bcd..7720f4fe 100644 --- a/noether_gui/ui/plugin_loader_widget.ui +++ b/noether_gui/ui/plugin_loader_widget.ui @@ -6,8 +6,8 @@ 0 0 - 400 - 300 + 340 + 159 @@ -47,24 +47,25 @@ - + + + true + + + + + 0 + 0 + 296 + 68 + + + + - - - - Qt::Vertical - - - - 20 - 40 - - - - diff --git a/noether_gui/ui/tpp_pipeline_widget.ui b/noether_gui/ui/tpp_pipeline_widget.ui index 13420e3a..37caaf5e 100644 --- a/noether_gui/ui/tpp_pipeline_widget.ui +++ b/noether_gui/ui/tpp_pipeline_widget.ui @@ -61,9 +61,29 @@ - + - + + + Qt::ScrollBarAlwaysOff + + + true + + + Qt::AlignHCenter|Qt::AlignTop + + + + + 0 + 0 + 364 + 289 + + + + diff --git a/noether_gui/ui/tpp_widget.ui b/noether_gui/ui/tpp_widget.ui index c26caec0..b40e5d52 100644 --- a/noether_gui/ui/tpp_widget.ui +++ b/noether_gui/ui/tpp_widget.ui @@ -6,8 +6,8 @@ 0 0 - 1070 - 614 + 215 + 344 @@ -63,61 +63,7 @@ - - - - - true - - - - - - - Load - - - - - - - - - Save - - - - - - - - 0 - 0 - - - - - 16777215 - 1024 - - - - Qt::ScrollBarAlwaysOff - - - true - - - - - 0 - 0 - 1024 - 220 - - - - + @@ -211,10 +157,6 @@ line_edit_mesh push_button_load_mesh - line_edit_configuration - push_button_load_configuration - push_button_save_configuration - scroll_area push_button_plan double_spin_box_axis_size check_box_show_original_mesh