From 9e510cd88610509ca2a8b1fdfce95e510d5565f3 Mon Sep 17 00:00:00 2001 From: aliakseis Date: Mon, 14 Mar 2022 14:09:00 +0300 Subject: [PATCH 1/7] CMakeLists.txt --- .gitignore | 222 +++++++++++++++++++++++++++++++++++++++++++++++++ CMakeLists.txt | 59 +++++++++++++ 2 files changed, 281 insertions(+) create mode 100644 .gitignore create mode 100644 CMakeLists.txt diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..ee99c1d --- /dev/null +++ b/.gitignore @@ -0,0 +1,222 @@ +################# +## Eclipse +################# + +*.pydevproject +.project +.metadata +bin/ +tmp/ +*.tmp +*.bak +*.swp +*~.nib +local.properties +.classpath +.settings/ +.loadpath + +# External tool builders +.externalToolBuilders/ + +# Locally stored "Eclipse launch configurations" +*.launch + +# CDT-specific +.cproject + +# PDT-specific +.buildpath + + +################# +## Visual Studio +################# + +## Ignore Visual Studio temporary files, build results, and +## files generated by popular Visual Studio add-ons. + +# User-specific files +*.suo +*.user +*.sln.docstates + +# Build results + +[Dd]ebug/ +[Rr]elease/ +x64/ +build/ +[Bb]in/ +[Oo]bj/ + +# MSTest test Results +[Tt]est[Rr]esult*/ +[Bb]uild[Ll]og.* + +*_i.c +*_p.c +*.ilk +*.meta +*.obj +*.pch +*.pdb +*.pgc +*.pgd +*.rsp +*.sbr +*.tlb +*.tli +*.tlh +*.tmp +*.tmp_proj +*.log +*.vspscc +*.vssscc +.builds +*.pidb +*.log +*.scc + +# Visual C++ cache files +ipch/ +*.aps +*.ncb +*.opensdf +*.sdf +*.cachefile + +# Visual Studio profiler +*.psess +*.vsp +*.vspx + +# Guidance Automation Toolkit +*.gpState + +# ReSharper is a .NET coding add-in +_ReSharper*/ +*.[Rr]e[Ss]harper + +# TeamCity is a build add-in +_TeamCity* + +# DotCover is a Code Coverage Tool +*.dotCover + +# NCrunch +*.ncrunch* +.*crunch*.local.xml + +# Installshield output folder +[Ee]xpress/ + +# DocProject is a documentation generator add-in +DocProject/buildhelp/ +DocProject/Help/*.HxT +DocProject/Help/*.HxC +DocProject/Help/*.hhc +DocProject/Help/*.hhk +DocProject/Help/*.hhp +DocProject/Help/Html2 +DocProject/Help/html + +# Click-Once directory +publish/ + +# Publish Web Output +*.Publish.xml +*.pubxml +*.publishproj + +# NuGet Packages Directory +## TODO: If you have NuGet Package Restore enabled, uncomment the next line +#packages/ + +# Windows Azure Build Output +csx +*.build.csdef + +# Windows Store app package directory +AppPackages/ + +# Others +sql/ +*.Cache +ClientBin/ +[Ss]tyle[Cc]op.* +~$* +*~ +*.dbmdl +*.[Pp]ublish.xml +*.pfx +*.publishsettings + +# RIA/Silverlight projects +Generated_Code/ + +# Backup & report files from converting an old project file to a newer +# Visual Studio version. Backup files are not needed, because we have git ;-) +_UpgradeReport_Files/ +Backup*/ +UpgradeLog*.XML +UpgradeLog*.htm + +# SQL Server files +App_Data/*.mdf +App_Data/*.ldf + +############# +## Windows detritus +############# + +# Windows image file caches +Thumbs.db +ehthumbs.db + +# Folder config file +Desktop.ini + +# Recycle Bin used on file shares +$RECYCLE.BIN/ + +# Mac crap +.DS_Store + + +############# +## Python +############# + +*.py[cod] + +# Packages +*.egg +*.egg-info +dist/ +build/ +eggs/ +parts/ +var/ +sdist/ +develop-eggs/ +.installed.cfg + +# Installer logs +pip-log.txt + +# Unit test / coverage reports +.coverage +.tox + +#Translations +*.mo + +#Mr Developer +.mr.developer.cfg + +ipch/ + +*.exe + +GeneratedFiles/ diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000..468872e --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,59 @@ +cmake_minimum_required(VERSION 3.5) + +project(segmentation-depthmap-3d-opencv LANGUAGES CXX) + +set(CMAKE_INCLUDE_CURRENT_DIR ON) + +set(CMAKE_AUTOUIC ON) +set(CMAKE_AUTOMOC ON) +set(CMAKE_AUTORCC ON) + +set(CMAKE_CXX_STANDARD 11) +set(CMAKE_CXX_STANDARD_REQUIRED ON) + +# QtCreator supports the following variables for Android, which are identical to qmake Android variables. +# Check http://doc.qt.io/qt-5/deployment-android.html for more information. +# They need to be set before the find_package(Qt5 ...) call. + +#if(ANDROID) +# set(ANDROID_PACKAGE_SOURCE_DIR "${CMAKE_CURRENT_SOURCE_DIR}/android") +# if (ANDROID_ABI STREQUAL "armeabi-v7a") +# set(ANDROID_EXTRA_LIBS +# ${CMAKE_CURRENT_SOURCE_DIR}/path/to/libcrypto.so +# ${CMAKE_CURRENT_SOURCE_DIR}/path/to/libssl.so) +# endif() +#endif() + +find_package(QT NAMES Qt6 Qt5 COMPONENTS Widgets OpenGL REQUIRED) +find_package(Qt${QT_VERSION_MAJOR} COMPONENTS Widgets OpenGL REQUIRED) + +# Requires OpenCV +FIND_PACKAGE( OpenCV 4 REQUIRED ) + +if(ANDROID) + add_library(segmentation-depthmap-3d-opencv SHARED + main.cpp + mainwindow.cpp + mainwindow.h + mainwindow.ui + mat-image-tools.cpp + openglwidget.cpp + ) +else() + add_executable(segmentation-depthmap-3d-opencv + main.cpp + mainwindow.cpp + mainwindow.h + mainwindow.ui + mat-image-tools.cpp + openglwidget.cpp + ${CMAKE_SOURCE_DIR}/resources.qrc + ) +endif() + +target_link_libraries(segmentation-depthmap-3d-opencv PRIVATE + Qt${QT_VERSION_MAJOR}::Widgets + Qt${QT_VERSION_MAJOR}::OpenGL + ${OpenCV_LIBRARIES} + opengl32 +) From 471461057f160ebaba270a6f30a095b6764a6885 Mon Sep 17 00:00:00 2001 From: aliakseis Date: Mon, 14 Mar 2022 19:33:04 +0300 Subject: [PATCH 2/7] 3d window auto resizing --- mainwindow.cpp | 14 ++++++++++++++ mainwindow.h | 3 +++ 2 files changed, 17 insertions(+) diff --git a/mainwindow.cpp b/mainwindow.cpp index a868211..b1f7462 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -53,10 +53,12 @@ MainWindow::MainWindow(QWidget *parent) : setFocusPolicy(Qt::StrongFocus); // catch keyboard and mouse in priority // add size grip to openGL widget + /* ui->openGLWidget_3d->setWindowFlags(Qt::SubWindow); QSizeGrip * sizeGrip = new QSizeGrip(ui->openGLWidget_3d); QGridLayout * layout = new QGridLayout(ui->openGLWidget_3d); layout->addWidget(sizeGrip, 0,0,1,1,Qt::AlignBottom | Qt::AlignRight); + */ // populate gray gradient curve combobox ui->listWidget_gradient_curve->blockSignals(true); // don't trigger automatic actions for these widgets @@ -2011,3 +2013,15 @@ void MainWindow::ChangeLabelGradient() // update depthmap mask with gradient updateVertices3D = true; Render(); // update view } + +void MainWindow::resizeEvent(QResizeEvent* event) +{ + QMainWindow::resizeEvent(event); + + const int w = width() - ui->openGLWidget_3d->pos().x(); + const int h = height() - ui->openGLWidget_3d->pos().y(); + if (w > 0 && h > 0) + { + ui->openGLWidget_3d->setFixedSize(w, h); + } +} diff --git a/mainwindow.h b/mainwindow.h index b794aec..f2a7811 100644 --- a/mainwindow.h +++ b/mainwindow.h @@ -148,6 +148,9 @@ private slots: void UpdateViewportDimensions(); // calculate width and height of the viewport cv::Point Viewport2Image(const cv::Point &p); // calculate coordinates in the image from the viewport + void resizeEvent(QResizeEvent* event) override; + + // the UI object, to access the UI elements created with Qt Designer Ui::MainWindow *ui; From e9d8143fb58c776d01310aac1a739a843e722e85 Mon Sep 17 00:00:00 2001 From: aliakseis Date: Tue, 15 Mar 2022 15:01:47 +0300 Subject: [PATCH 3/7] fullscreen mode fixed --- mainwindow.cpp | 29 ++++++++++++++++++++--------- openglwidget.cpp | 19 +++++++++++++++++++ openglwidget.h | 1 + 3 files changed, 40 insertions(+), 9 deletions(-) diff --git a/mainwindow.cpp b/mainwindow.cpp index b1f7462..357ab65 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -1317,6 +1317,7 @@ void MainWindow::on_checkBox_3d_fullscreen_clicked() // view 3D scene fullscreen saveWidthOpenGL = ui->openGLWidget_3d->width(); saveHeightOpenGL = ui->openGLWidget_3d->height(); + /* QRect screenSize = qApp->desktop()->availableGeometry(qApp->desktop()->primaryScreen()); // get screen size in which app is run int newW = screenSize.width(); int newH = round(newW * (double(ui->openGLWidget_3d->width()) / ui->openGLWidget_3d->height())); // keep aspect ratio of widget using new width @@ -1326,6 +1327,11 @@ void MainWindow::on_checkBox_3d_fullscreen_clicked() // view 3D scene fullscreen ui->openGLWidget_3d->move(QPoint(0, 0)); // move widget to upper-left position in window ui->openGLWidget_3d->resize(QSize(newW, newH)); // resize openGL widget ui->openGLWidget_3d->raise(); // bring the 3d widget to front, above all other objects + */ + + ui->openGLWidget_3d->setWindowFlags(Qt::Window); + ui->openGLWidget_3d->setContentsMargins(0, 0, 0, 0); + ui->openGLWidget_3d->showFullScreen(); } void MainWindow::on_button_3d_reset_clicked() // recenter position and reset zoom for 3D scene @@ -1634,9 +1640,13 @@ void MainWindow::keyPressEvent(QKeyEvent *keyEvent) // special keys qApp->processEvents(); } else if ((keyEvent->key() == Qt::Key_Escape) & (ui->checkBox_3d_fullscreen->isChecked())) { // get out of fullscreen view - this->setWindowFlags((((windowFlags() | Qt::CustomizeWindowHint) - & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); - show(); + //this->setWindowFlags((((windowFlags() | Qt::CustomizeWindowHint) + // & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); + //show(); + + ui->openGLWidget_3d->setWindowFlags(Qt::SubWindow); + ui->openGLWidget_3d->showNormal(); + ui->openGLWidget_3d->move(QPoint(saveXOpenGL, saveYOpenGL)); // restore openGL widget to its previous position ui->openGLWidget_3d->resize(QSize(saveWidthOpenGL, saveHeightOpenGL)); // ... and size ui->checkBox_3d_fullscreen->setChecked(false); // uncheck fullscreen button @@ -2017,11 +2027,12 @@ void MainWindow::ChangeLabelGradient() // update depthmap mask with gradient void MainWindow::resizeEvent(QResizeEvent* event) { QMainWindow::resizeEvent(event); - - const int w = width() - ui->openGLWidget_3d->pos().x(); - const int h = height() - ui->openGLWidget_3d->pos().y(); - if (w > 0 && h > 0) - { - ui->openGLWidget_3d->setFixedSize(w, h); + if (!ui->checkBox_3d_fullscreen->isChecked()) { + const int w = width() - ui->openGLWidget_3d->pos().x(); + const int h = height() - ui->openGLWidget_3d->pos().y(); + if (w > 0 && h > 0) + { + ui->openGLWidget_3d->setFixedSize(w, h); + } } } diff --git a/openglwidget.cpp b/openglwidget.cpp index f3fecf9..dd6c989 100644 --- a/openglwidget.cpp +++ b/openglwidget.cpp @@ -646,3 +646,22 @@ void openGLWidget::Capture() // take a snapshot of rendered 3D scene { capture3D = grabFramebuffer(); // slow because it relies on glReadPixels() to read back the pixels } + +class QMainWindowKeyPressEventAccessor : public QMainWindow +{ +public: + void callKeyPressEvent(QKeyEvent *keyEvent) + { + keyPressEvent(keyEvent); + } +}; + +void openGLWidget::keyPressEvent(QKeyEvent *keyEvent) // special keys +{ + for (QWidget* widget : QApplication::topLevelWidgets()) { + if (auto *mainWindow = qobject_cast(widget)) { + static_cast(mainWindow)->callKeyPressEvent(keyEvent); + break; + } + } +} diff --git a/openglwidget.h b/openglwidget.h index 4541942..4e75218 100644 --- a/openglwidget.h +++ b/openglwidget.h @@ -90,6 +90,7 @@ class openGLWidget : public QOpenGLWidget void mousePressEvent(QMouseEvent *event); // save initial mouse position for move and rotate void mouseMoveEvent(QMouseEvent *event); // move and rotate view with mouse buttons void wheelEvent(QWheelEvent *event); // zoom + void keyPressEvent(QKeyEvent *keyEvent); void ComputeVertices(); // create vertices void ComputeIndexes(); // create indexes void UpdateVertices(); // update vertices z From c94b301de0b62c531058492e4114fbfd7104edf9 Mon Sep 17 00:00:00 2001 From: aliakseis Date: Tue, 15 Mar 2022 15:42:22 +0300 Subject: [PATCH 4/7] main window position saving/restoring --- main.cpp | 1 + mainwindow.cpp | 45 ++++++++++++++++++++++++++++++++++++++++++++- mainwindow.h | 21 ++++++++++++--------- openglwidget.h | 8 ++++---- 4 files changed, 61 insertions(+), 14 deletions(-) diff --git a/main.cpp b/main.cpp index f2c7a78..9242c5e 100644 --- a/main.cpp +++ b/main.cpp @@ -15,6 +15,7 @@ int main(int argc, char *argv[]) { QApplication a(argc, argv); MainWindow w; + w.readPositionSettings(); w.show(); return a.exec(); diff --git a/mainwindow.cpp b/mainwindow.cpp index 357ab65..f65392d 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -19,6 +19,7 @@ #include #include #include +#include #include "mat-image-tools.h" #include "dispersion3D.h" @@ -49,7 +50,7 @@ MainWindow::MainWindow(QWidget *parent) : // window setWindowFlags((((windowFlags() | Qt::CustomizeWindowHint) & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); // don't show buttons in title bar - this->setWindowState(Qt::WindowMaximized); // maximize window + //this->setWindowState(Qt::WindowMaximized); // maximize window setFocusPolicy(Qt::StrongFocus); // catch keyboard and mouse in priority // add size grip to openGL widget @@ -144,6 +145,8 @@ void MainWindow::on_button_quit_clicked() if (quit == QMessageBox::No) // don't quit ! return; + writePositionSettings(); + QCoreApplication::quit(); } @@ -2024,6 +2027,45 @@ void MainWindow::ChangeLabelGradient() // update depthmap mask with gradient Render(); // update view } +// https://stackoverflow.com/questions/74690/how-do-i-store-the-window-size-between-sessions-in-qt +void MainWindow::writePositionSettings() +{ + QSettings qsettings("noname", "Tiles"); + + qsettings.beginGroup("mainwindow"); + + qsettings.setValue("geometry", saveGeometry()); + qsettings.setValue("savestate", saveState()); + qsettings.setValue("maximized", isMaximized()); + if (!isMaximized()) { + qsettings.setValue("pos", pos()); + qsettings.setValue("size", size()); + } + + qsettings.endGroup(); +} + +void MainWindow::readPositionSettings() +{ + QSettings qsettings("noname", "Tiles"); + + qsettings.beginGroup("mainwindow"); + + restoreGeometry(qsettings.value("geometry", saveGeometry()).toByteArray()); + restoreState(qsettings.value("savestate", saveState()).toByteArray()); + move(qsettings.value("pos", pos()).toPoint()); + resize(qsettings.value("size", size()).toSize()); + if (qsettings.value("maximized", isMaximized()).toBool()) + showMaximized(); + + qsettings.endGroup(); +} + +void MainWindow::moveEvent(QMoveEvent*) +{ + writePositionSettings(); +} + void MainWindow::resizeEvent(QResizeEvent* event) { QMainWindow::resizeEvent(event); @@ -2035,4 +2077,5 @@ void MainWindow::resizeEvent(QResizeEvent* event) ui->openGLWidget_3d->setFixedSize(w, h); } } + writePositionSettings(); } diff --git a/mainwindow.h b/mainwindow.h index f2a7811..37f1976 100644 --- a/mainwindow.h +++ b/mainwindow.h @@ -38,6 +38,9 @@ class MainWindow : public QMainWindow explicit MainWindow(QWidget *parent = 0); ~MainWindow(); + void writePositionSettings(); + void readPositionSettings(); + public slots: private slots: @@ -123,13 +126,16 @@ private slots: void DeleteAllLabels(); // delete ALL labels and create one new if wanted //// Keyboard & mouse events - void keyPressEvent(QKeyEvent *keyEvent); // for the create cell mode - void keyReleaseEvent(QKeyEvent *keyEvent); + void keyPressEvent(QKeyEvent *keyEvent) override; // for the create cell mode + void keyReleaseEvent(QKeyEvent *keyEvent) override; + + void mouseReleaseEvent(QMouseEvent *eventRelease) override; // when the mouse button is released + void mousePressEvent(QMouseEvent *eventPress) override; // mouse events = zoom, set cell color etc + void mouseMoveEvent(QMouseEvent *eventPress) override; + void wheelEvent(QWheelEvent *wheelEvent) override; - void mouseReleaseEvent(QMouseEvent *eventRelease); // when the mouse button is released - void mousePressEvent(QMouseEvent *eventPress); // mouse events = zoom, set cell color etc - void mouseMoveEvent(QMouseEvent *eventPress); - void wheelEvent(QWheelEvent *wheelEvent); + void moveEvent(QMoveEvent*) override; + void resizeEvent(QResizeEvent*) override; //// Display void Render(); // display image in viewport with depthmap and selection @@ -148,9 +154,6 @@ private slots: void UpdateViewportDimensions(); // calculate width and height of the viewport cv::Point Viewport2Image(const cv::Point &p); // calculate coordinates in the image from the viewport - void resizeEvent(QResizeEvent* event) override; - - // the UI object, to access the UI elements created with Qt Designer Ui::MainWindow *ui; diff --git a/openglwidget.h b/openglwidget.h index 4e75218..72bcfc2 100644 --- a/openglwidget.h +++ b/openglwidget.h @@ -87,10 +87,10 @@ class openGLWidget : public QOpenGLWidget void initializeGL(); // launched when the widget is initialized void paintGL(); // 3D rendering void resizeGL(int width, int height); // called when the widget is resized - void mousePressEvent(QMouseEvent *event); // save initial mouse position for move and rotate - void mouseMoveEvent(QMouseEvent *event); // move and rotate view with mouse buttons - void wheelEvent(QWheelEvent *event); // zoom - void keyPressEvent(QKeyEvent *keyEvent); + void mousePressEvent(QMouseEvent *event) override; // save initial mouse position for move and rotate + void mouseMoveEvent(QMouseEvent *event) override; // move and rotate view with mouse buttons + void wheelEvent(QWheelEvent *event) override; // zoom + void keyPressEvent(QKeyEvent *keyEvent) override; void ComputeVertices(); // create vertices void ComputeIndexes(); // create indexes void UpdateVertices(); // update vertices z From 7ff69fc0e07dfb87b480622815c5dbddb85172c9 Mon Sep 17 00:00:00 2001 From: aliakseis Date: Tue, 15 Mar 2022 18:48:16 +0300 Subject: [PATCH 5/7] install section initial revision --- CMakeLists.txt | 42 +++++++++++++++++++++++++++++++++++++++++- 1 file changed, 41 insertions(+), 1 deletion(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 468872e..2aada2c 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -51,9 +51,49 @@ else() ) endif() +if(WIN32) + set(OPENGL_LIBRARY "opengl32") +endif() + target_link_libraries(segmentation-depthmap-3d-opencv PRIVATE Qt${QT_VERSION_MAJOR}::Widgets Qt${QT_VERSION_MAJOR}::OpenGL ${OpenCV_LIBRARIES} - opengl32 + ${OPENGL_LIBRARY} ) + +if(MSVC) + set(CMAKE_INSTALL_PREFIX "${CMAKE_BINARY_DIR}/bin") + + set(BINARY_INSTALL_DIR .) + + install(TARGETS ${PROJECT_NAME} DESTINATION ${BINARY_INSTALL_DIR}) + + set(BINARY_FILE "${CMAKE_INSTALL_PREFIX}/${PROJECT_NAME}.exe") + + get_filename_component(CMAKE_TOOLCHAIN_DIRECTORY ${CMAKE_TOOLCHAIN_FILE} DIRECTORY) + + if (CMAKE_SIZEOF_VOID_P EQUAL 8) + set( CMAKE_INSTALL_PLATFORM "x64-windows" ) + else () + set( CMAKE_INSTALL_PLATFORM "x86-windows" ) + endif () + + install(CODE "execute_process(COMMAND \"powershell\" \"-noprofile\" \"-executionpolicy\" \"Bypass\" \"-file\" \"${CMAKE_TOOLCHAIN_DIRECTORY}/msbuild/applocal.ps1\" \"-targetBinary\" \"${BINARY_FILE}\" \"-installedDir\" \"${CMAKE_TOOLCHAIN_DIRECTORY}/../../installed/${CMAKE_INSTALL_PLATFORM}/bin\" \"-OutVariable\" \"out\")") + + # Install CRT + set(CMAKE_INSTALL_SYSTEM_RUNTIME_DESTINATION .) + #if(DEVELOPER_FEATURES) + # set(CMAKE_INSTALL_DEBUG_LIBRARIES ON) + #endif(DEVELOPER_FEATURES) + set(CMAKE_INSTALL_UCRT_LIBRARIES TRUE) + set(CMAKE_INSTALL_OPENMP_LIBRARIES TRUE) + include (InstallRequiredSystemLibraries) + + if (QT_QMAKE_EXECUTABLE) + get_filename_component(QT_BIN_DIRECTORY ${QT_QMAKE_EXECUTABLE} DIRECTORY) + install(CODE "execute_process(COMMAND \"${QT_BIN_DIRECTORY}/windeployqt\" \"${BINARY_FILE}\" WORKING_DIRECTORY \"${QT_BIN_DIRECTORY}\")") + else() + install(CODE "execute_process(COMMAND \"windeployqt\" \"${BINARY_FILE}\" )") + endif() +endif() From 36a927e4cb6d5c751cf612463d2e3e769ac7b9cf Mon Sep 17 00:00:00 2001 From: aliakseis Date: Thu, 17 Mar 2022 14:33:21 +0300 Subject: [PATCH 6/7] configuration fixed --- main.cpp | 5 +++++ mainwindow.cpp | 6 +++--- 2 files changed, 8 insertions(+), 3 deletions(-) diff --git a/main.cpp b/main.cpp index 9242c5e..5a52ecd 100644 --- a/main.cpp +++ b/main.cpp @@ -14,6 +14,11 @@ int main(int argc, char *argv[]) { QApplication a(argc, argv); + + a.setApplicationName("segmentation-depthmap-3d-opencv"); + a.setOrganizationName("AbsurdePhoton"); + a.setOrganizationDomain("www.absurdephoton.fr"); + MainWindow w; w.readPositionSettings(); w.show(); diff --git a/mainwindow.cpp b/mainwindow.cpp index f65392d..72bb22c 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -2030,7 +2030,7 @@ void MainWindow::ChangeLabelGradient() // update depthmap mask with gradient // https://stackoverflow.com/questions/74690/how-do-i-store-the-window-size-between-sessions-in-qt void MainWindow::writePositionSettings() { - QSettings qsettings("noname", "Tiles"); + QSettings qsettings; qsettings.beginGroup("mainwindow"); @@ -2047,7 +2047,7 @@ void MainWindow::writePositionSettings() void MainWindow::readPositionSettings() { - QSettings qsettings("noname", "Tiles"); + QSettings qsettings; qsettings.beginGroup("mainwindow"); @@ -2055,7 +2055,7 @@ void MainWindow::readPositionSettings() restoreState(qsettings.value("savestate", saveState()).toByteArray()); move(qsettings.value("pos", pos()).toPoint()); resize(qsettings.value("size", size()).toSize()); - if (qsettings.value("maximized", isMaximized()).toBool()) + if (qsettings.value("maximized", true).toBool()) showMaximized(); qsettings.endGroup(); From 7c0e0084fddbe27b6b22a4378cca677beab05f82 Mon Sep 17 00:00:00 2001 From: aliakseis Date: Thu, 24 Mar 2022 13:36:49 +0300 Subject: [PATCH 7/7] clang-tidy, formatting, clazy --- CMakeLists.txt | 2 +- main.cpp | 8 +- mainwindow.cpp | 790 ++++++++++++++++------------ mainwindow.h | 2 +- mat-image-tools.cpp | 547 ++++++++++--------- openglwidget.cpp | 247 +++++---- openglwidget.h | 4 +- segmentation-depthmap-3d-opencv.pro | 2 +- 8 files changed, 912 insertions(+), 690 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 2aada2c..60adcff 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -8,7 +8,7 @@ set(CMAKE_AUTOUIC ON) set(CMAKE_AUTOMOC ON) set(CMAKE_AUTORCC ON) -set(CMAKE_CXX_STANDARD 11) +set(CMAKE_CXX_STANDARD 17) set(CMAKE_CXX_STANDARD_REQUIRED ON) # QtCreator supports the following variables for Android, which are identical to qmake Android variables. diff --git a/main.cpp b/main.cpp index 5a52ecd..187977c 100644 --- a/main.cpp +++ b/main.cpp @@ -15,13 +15,13 @@ int main(int argc, char *argv[]) { QApplication a(argc, argv); - a.setApplicationName("segmentation-depthmap-3d-opencv"); - a.setOrganizationName("AbsurdePhoton"); - a.setOrganizationDomain("www.absurdephoton.fr"); + QApplication::setApplicationName("segmentation-depthmap-3d-opencv"); + QApplication::setOrganizationName("AbsurdePhoton"); + QApplication::setOrganizationDomain("www.absurdephoton.fr"); MainWindow w; w.readPositionSettings(); w.show(); - return a.exec(); + return QApplication::exec(); } diff --git a/mainwindow.cpp b/mainwindow.cpp index 72bb22c..315f8bd 100644 --- a/mainwindow.cpp +++ b/mainwindow.cpp @@ -32,7 +32,7 @@ using namespace std; void MainWindow::AddCurveItem(const QString &title, const QColor &color, const QString &tip) // used to add gray gradient curves to the list { - QListWidgetItem *item = new QListWidgetItem (); // create new label + auto *item = new QListWidgetItem(); // create new label item->setText(title); // set name to current label item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled); // item enabled and selectable item->setSelected(false); // but don't select it ! @@ -49,8 +49,8 @@ MainWindow::MainWindow(QWidget *parent) : // window setWindowFlags((((windowFlags() | Qt::CustomizeWindowHint) - & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); // don't show buttons in title bar - //this->setWindowState(Qt::WindowMaximized); // maximize window + & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); // don't show buttons in title bar +//this->setWindowState(Qt::WindowMaximized); // maximize window setFocusPolicy(Qt::StrongFocus); // catch keyboard and mouse in priority // add size grip to openGL widget @@ -63,17 +63,17 @@ MainWindow::MainWindow(QWidget *parent) : // populate gray gradient curve combobox ui->listWidget_gradient_curve->blockSignals(true); // don't trigger automatic actions for these widgets - AddCurveItem("LINEAR", QColor(0,128,0), "Gray levels go straight from beginning to end\n f(x) = x"); - AddCurveItem("COSINUS²", QColor(0,0,128), "Gray levels are spread a bit more at beginning and end\n f(x)=cos(pi/2-x*pi/2)²"); - AddCurveItem("SIGMOID", QColor(0,0,128), "S-shaped gray levels\n1/3 beginning 1/3 gradient 1/3 end\n f(x)=1/(1 + e(-5*(2x -1))"); - AddCurveItem("COSINUS", QColor(128,128,0), "Fast beginning and strong end grays\n f(x)=cos(pi/2-x*pi/2)"); - AddCurveItem("COS²SQRT", QColor(128,128,0), "Center grays are predominant\n f(x)=cos(pi/2−sqrt(x)*pi/2)²"); - AddCurveItem("POWER²", QColor(128,64,0), "1/3 of beginning level then gradient for the rest, fast ending\n f(x)=x²"); - AddCurveItem("COS²POWER2", QColor(128,64,0), "Very strong beginning then gradient\n f(x)=cos(pi/2−x²∙pi/2)²"); - AddCurveItem("POWER³", QColor(128,64,0), "Very strong beginning and fast ending\n f(x)=x³"); - AddCurveItem("UNDULATE", QColor(128,0,0), "Waves modulated by gray levels range\n f(x)=cos(x/4*pi)"); - AddCurveItem("UNDULATE²", QColor(128,0,0), "Increasing waves modulated by gray levels range\n f(x)=cos((x*2*pi)²)/2+0.5"); - AddCurveItem("UNDULATE+", QColor(128,0,0), "Increasing levels of gray stripes\n f(x)=f(x) = cos(pi²∙(x+2.085)²) / ((x+2.085)³+10) + (x+2.085) − 2.11"); + AddCurveItem("LINEAR", QColor(0, 128, 0), "Gray levels go straight from beginning to end\n f(x) = x"); + AddCurveItem("COSINUS²", QColor(0, 0, 128), "Gray levels are spread a bit more at beginning and end\n f(x)=cos(pi/2-x*pi/2)²"); + AddCurveItem("SIGMOID", QColor(0, 0, 128), "S-shaped gray levels\n1/3 beginning 1/3 gradient 1/3 end\n f(x)=1/(1 + e(-5*(2x -1))"); + AddCurveItem("COSINUS", QColor(128, 128, 0), "Fast beginning and strong end grays\n f(x)=cos(pi/2-x*pi/2)"); + AddCurveItem("COS²SQRT", QColor(128, 128, 0), "Center grays are predominant\n f(x)=cos(pi/2−sqrt(x)*pi/2)²"); + AddCurveItem("POWER²", QColor(128, 64, 0), "1/3 of beginning level then gradient for the rest, fast ending\n f(x)=x²"); + AddCurveItem("COS²POWER2", QColor(128, 64, 0), "Very strong beginning then gradient\n f(x)=cos(pi/2−x²∙pi/2)²"); + AddCurveItem("POWER³", QColor(128, 64, 0), "Very strong beginning and fast ending\n f(x)=x³"); + AddCurveItem("UNDULATE", QColor(128, 0, 0), "Waves modulated by gray levels range\n f(x)=cos(x/4*pi)"); + AddCurveItem("UNDULATE²", QColor(128, 0, 0), "Increasing waves modulated by gray levels range\n f(x)=cos((x*2*pi)²)/2+0.5"); + AddCurveItem("UNDULATE+", QColor(128, 0, 0), "Increasing levels of gray stripes\n f(x)=f(x) = cos(pi²∙(x+2.085)²) / ((x+2.085)³+10) + (x+2.085) − 2.11"); ui->listWidget_gradient_curve->blockSignals(false); // populate tints comboBox @@ -113,7 +113,9 @@ void MainWindow::InitializeValues() // Global variables init if (fs.isOpened()) { fs["BaseDir"] >> basedir; // load labels } - else basedir = "/home/"; // base path and file + else { + basedir = "/home/"; // base path and file + } basefile = "example"; nbLabels = 0; // no labels yet updateVertices3D = false; // not an update @@ -129,7 +131,7 @@ void MainWindow::InitializeValues() // Global variables init cvtColor(depthmap, depthmap, COLOR_BGR2GRAY); ui->openGLWidget_3d->image3D = image; // transfer data to widget ui->openGLWidget_3d->depthmap3D = depthmap; - ui->openGLWidget_3d->area3D = Rect(0, 0, image.cols,image.rows); + ui->openGLWidget_3d->area3D = Rect(0, 0, image.cols, image.rows); ui->openGLWidget_3d->mask3D = Mat::zeros(image.rows, image.cols, CV_8UC1); ui->openGLWidget_3d->depth3D = 1; // initial view ui->openGLWidget_3d->anaglyphShift = -1.5; @@ -141,9 +143,10 @@ void MainWindow::InitializeValues() // Global variables init void MainWindow::on_button_quit_clicked() { - int quit = QMessageBox::question(this, "Quit this wonderful program", "Are you sure you want to quit?", QMessageBox::Yes|QMessageBox::No); // quit, are you sure ? - if (quit == QMessageBox::No) // don't quit ! + int quit = QMessageBox::question(this, "Quit this wonderful program", "Are you sure you want to quit?", QMessageBox::Yes | QMessageBox::No); // quit, are you sure ? + if (quit == QMessageBox::No) { // don't quit ! return; + } writePositionSettings(); @@ -201,15 +204,17 @@ void MainWindow::on_listWidget_labels_currentItemChanged(QListWidgetItem *curren selection_rect.height += selection_rect.y - rect_temp.y; selection_rect.y = rect_temp.y; } - if (rect_temp.x + rect_temp.width > selection_rect.x + selection_rect.width) + if (rect_temp.x + rect_temp.width > selection_rect.x + selection_rect.width) { selection_rect.width = rect_temp.x + rect_temp.width - selection_rect.x; - if (rect_temp.y + rect_temp.height > selection_rect.y + selection_rect.height) + } + if (rect_temp.y + rect_temp.height > selection_rect.y + selection_rect.height) { selection_rect.height = rect_temp.y + rect_temp.height - selection_rect.y; + } } ui->openGLWidget_3d->area3D = selection_rect; // update opengl widget elements ui->openGLWidget_3d->mask3D = currentLabelMask; - drawContours(selection, contours, -1, Vec3b(0, 255, 255), 1, 8, hierarchy ); // draw contour of new cell in selection mask + drawContours(selection, contours, -1, Vec3b(0, 255, 255), 1, 8, hierarchy); // draw contour of new cell in selection mask /*cv::rectangle(selection, Rect(selection_rect.x, selection_rect.y, selection_rect.width, selection_rect.height), Vec3b(255, 255, 255), 2); // draw entire selection rectangle*/ @@ -218,22 +223,22 @@ void MainWindow::on_listWidget_labels_currentItemChanged(QListWidgetItem *curren int row = ui->listWidget_labels->currentRow(); // get current row of the list to access array indexed on it switch (gradients[row].gradient) { // gradient type ? - case gradient_flat: { - ui->radioButton_flat->setChecked(true); // set correspondng radio button - break; - } - case gradient_linear: { - ui->radioButton_linear->setChecked(true); - break; - } - case gradient_doubleLinear: { - ui->radioButton_double_linear->setChecked(true); - break; - } - case gradient_radial: { - ui->radioButton_radial->setChecked(true); - break; - } + case gradient_flat: { + ui->radioButton_flat->setChecked(true); // set correspondng radio button + break; + } + case gradient_linear: { + ui->radioButton_linear->setChecked(true); + break; + } + case gradient_doubleLinear: { + ui->radioButton_double_linear->setChecked(true); + break; + } + case gradient_radial: { + ui->radioButton_radial->setChecked(true); + break; + } } ui->horizontalSlider_begin->setValue(gradients[row].beginColor); // show begin and end colors @@ -257,20 +262,21 @@ void MainWindow::SaveDirBaseFile() fs.release(); // close file } -void MainWindow::ChangeBaseDir(QString filename) // Set base dir and file +void MainWindow::ChangeBaseDir(const QString& filename) // Set base dir and file { basefile = filename.toUtf8().constData(); // base file name and dir are used after to save other files // Remove extension if present size_t period_idx = basefile.rfind('.'); - if (std::string::npos != period_idx) + if (std::string::npos != period_idx) { basefile.erase(period_idx); + } basedir = basefile; size_t found = basefile.find_last_of("\\/"); // find last directory - std::string separator = basefile.substr(found,1); // copy path separator (Linux <> Windows) - basedir = basedir.substr(0,found) + separator; // define base path - basefile = basefile.substr(found+1); // delete path in base file + std::string separator = basefile.substr(found, 1); // copy path separator (Linux <> Windows) + basedir = basedir.substr(0, found) + separator; // define base path + basefile = basefile.substr(found + 1); // delete path in base file SaveDirBaseFile(); // Save current path to ini file } @@ -298,8 +304,9 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML { QString filename = QFileDialog::getOpenFileName(this, "Load segmentation from XML file...", QString::fromStdString(basedir + "*-segmentation-data.xml"), "*.xml *.XML"); // get file name - if (filename.isNull() || filename.isEmpty()) // cancel ? + if (filename.isNull() || filename.isEmpty()) { // cancel ? return; + } QApplication::setOverrideCursor(Qt::WaitCursor); // wait cursor qApp->processEvents(); @@ -315,11 +322,12 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML ChangeBaseDir(filename); size_t pos = basefile.find("-segmentation-data"); // the XML file must end with this - if (pos != std::string::npos) // yes ! + if (pos != std::string::npos) { // yes ! basefile.erase(pos, basefile.length()); + } else { // doesn't end with "-segmentation-data" QMessageBox::critical(this, "File name error", - "There was a problem reading the segmentation mask file:\nit must end with ''-segmentation-data.xml''"); + "There was a problem reading the segmentation mask file:\nit must end with ''-segmentation-data.xml''"); DisableGUI(); // problem : reset GUI elements and exit return; } @@ -328,12 +336,14 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML std::string filesession = filename.toUtf8().constData(); // base file name pos = filesession.find("-segmentation-data.xml"); // ends with "-segmentation-data.xml" - if (pos != std::string::npos) filesession.erase(pos, filesession.length()); // this is what will be used to name files hereafter + if (pos != std::string::npos) { + filesession.erase(pos, filesession.length()); // this is what will be used to name files hereafter + } depthmap = cv::imread(filesession + "-segmentation-mask.png", IMREAD_COLOR); // load segmentation mask if (depthmap.empty()) { // mask empty, not good ! QMessageBox::critical(this, "File error", - "There was a problem reading the segmentation mask file:\nit must end with ''-segmentation-mask.png''"); + "There was a problem reading the segmentation mask file:\nit must end with ''-segmentation-mask.png''"); DisableGUI(); // problem : reset GUI elements and exit return; } @@ -342,14 +352,14 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML if (image.empty()) { QMessageBox::critical(this, "File error", - "There was a problem reading the segmentation image file:\nit must end with ''-segmentation-image.png''"); + "There was a problem reading the segmentation image file:\nit must end with ''-segmentation-image.png''"); DisableGUI(); return; } if ((image.cols != depthmap.cols) | (image.rows != depthmap.rows)) { // image and mask sizes not the same -> not good ! QMessageBox::critical(this, "Image size error", - "The image and mask image size (width and height) differ"); + "The image and mask image size (width and height) differ"); DisableGUI(); return; } @@ -363,7 +373,7 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML if (!fs.isOpened()) { // file not found ? this error is not handled by the above instructions QMessageBox::critical(this, "File error", - "There was a problem reading the segmentation data file:\nit must end with ''-segmentation-data.xml''"); + "There was a problem reading the segmentation data file:\nit must end with ''-segmentation-data.xml''"); DisableGUI(); return; } @@ -372,12 +382,12 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML try { // try to load labels data fs["LabelsMask"] >> labels; // load labels mask } - catch( cv::Exception& e ) // problem ? + catch (cv::Exception& e) // problem ? { const char* err_msg = e.what(); QMessageBox::critical(this, "XML Segmentation file error", - "There was a problem reading the segmentation XML file\nThe \"LabelsMask\" data is wrong\nError:\n" - + QString(err_msg)); + "There was a problem reading the segmentation XML file\nThe \"LabelsMask\" data is wrong\nError:\n" + + QString(err_msg)); DisableGUI(); return; } @@ -387,7 +397,7 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML if (nbLabels <= 0) { // no labels ? QMessageBox::critical(this, "XML Segmentation file error", - "There was a problem reading the segmentation XML file\nThe data is wrong"); + "There was a problem reading the segmentation XML file\nThe data is wrong"); DisableGUI(); return; } @@ -398,15 +408,15 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML int num = -1; std::string name = "###Error###"; // errors are not handled here, better set rubbish values, the user will see it anyway - QListWidgetItem *item = new QListWidgetItem (); // create new label item + auto *item = new QListWidgetItem(); // create new label item std::string field; field = "LabelId" + std::to_string(i); // read label id - fs [field] >> num; + fs[field] >> num; item->setData(Qt::UserRole, num); // set it to current label field = "LabelName" + std::to_string(i); // read label name - fs [field] >> name; + fs[field] >> name; item->setText(QString::fromStdString(name)); // set name to current label item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled); // item enabled, editable and selectable @@ -416,7 +426,7 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML Mat1b mask_temp = labels == num; // extract label to temp depthmap Moments m = moments(mask_temp, false); // compute the barycenter of the label representation - cv::Point p(m.m10/m.m00, m.m01/m.m00); + cv::Point p(m.m10 / m.m00, m.m01 / m.m00); uchar col = int(round(double(p.y) / image.rows * 255)); // set an arbitray gray level based on the barycenter height in image @@ -426,10 +436,12 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML gradients[i].beginColor = col; // gradient begin and end color are barycenter color gradients[i].endColor = col; gradients[i].beginPoint = p; // begin point of gradient arrow is the barycenter - if (p.y - 50 > 0) // the arrow is 50 pixels long (vertical), the head should stay in the image rectangle + if (p.y - 50 > 0) { // the arrow is 50 pixels long (vertical), the head should stay in the image rectangle gradients[i].endPoint = cv::Point(p.x, p.y - 50); - else + } + else { gradients[i].endPoint = cv::Point(p.x, p.y + 50); + } gradients[i].gradient = gradient_flat; // gradient flat at first gradients[i].curve = curve_linear; // and gradient curve set to linear } @@ -458,13 +470,17 @@ void MainWindow::on_button_load_segmentation_clicked() // load segmentation XML double zoomX = double(ui->label_viewport->width()) / image.cols; // find the best fit for the viewport : try vertical and horizontal ratios double zoomY = double(ui->label_viewport->height()) / image.rows; - if (zoomX < zoomY) zoom = zoomX; // the lowest fits the view - else zoom = zoomY; + if (zoomX < zoomY) { + zoom = zoomX; // the lowest fits the view + } + else { + zoom = zoomY; + } oldZoom = zoom; // no zoom change ShowZoomValue(); // display current zoom value viewport = Rect(0, 0, image.cols, image.rows); // update viewport size - thumbnail = ResizeImageAspectRatio(image, cv::Size(ui->label_thumbnail->width(),ui->label_thumbnail->height())); // create thumbnail + thumbnail = ResizeImageAspectRatio(image, cv::Size(ui->label_thumbnail->width(), ui->label_thumbnail->height())); // create thumbnail ShowThumbnail(); ui->openGLWidget_3d->depthmap3D = depthmap; // initialize 3D view data @@ -486,14 +502,15 @@ void MainWindow::on_button_save_depthmap_clicked() // save XML and image depthma { if (!loaded) { // nothing loaded yet = get out QMessageBox::warning(this, "Nothing to save", - "Not now!\n\nBefore anything else, load a Segmentation or Depthmap project"); + "Not now!\n\nBefore anything else, load a Segmentation or Depthmap project"); return; } QString filename = QFileDialog::getSaveFileName(this, "Save depthmap to XML file...", "./" + QString::fromStdString(basedir + basefile + "-depthmap-data.xml"), "*.xml *.XML"); // filename - if (filename.isNull() || filename.isEmpty()) // cancel ? + if (filename.isNull() || filename.isEmpty()) { // cancel ? return; + } QApplication::setOverrideCursor(Qt::WaitCursor); // wait cursor qApp->processEvents(); @@ -508,13 +525,19 @@ void MainWindow::on_button_save_depthmap_clicked() // save XML and image depthma basefile = basefile.substr(found+1); // delete ending slash*/ ChangeBaseDir(filename); size_t pos = basefile.find("-depthmap-data"); - if (pos != std::string::npos) basefile.erase(pos, basefile.length()); + if (pos != std::string::npos) { + basefile.erase(pos, basefile.length()); + } std::string filesession = filename.toUtf8().constData(); pos = filesession.find("-depthmap-data.xml"); // use base file name - if (pos != std::string::npos) filesession.erase(pos, filesession.length()); + if (pos != std::string::npos) { + filesession.erase(pos, filesession.length()); + } pos = filesession.find(".xml"); // use base file name - if (pos != std::string::npos) filesession.erase(pos, filesession.length()); + if (pos != std::string::npos) { + filesession.erase(pos, filesession.length()); + } bool write; Mat depthmap_temp; @@ -523,7 +546,7 @@ void MainWindow::on_button_save_depthmap_clicked() // save XML and image depthma if (!write) { // problem ? QApplication::restoreOverrideCursor(); // Restore cursor QMessageBox::critical(this, "File error", - "There was a problem saving the depthmap mask image file"); + "There was a problem saving the depthmap mask image file"); return; } @@ -531,7 +554,7 @@ void MainWindow::on_button_save_depthmap_clicked() // save XML and image depthma if (!write) { // problem ? QApplication::restoreOverrideCursor(); // Restore cursor QMessageBox::critical(this, "File error", - "There was a problem saving the depthmap image file"); + "There was a problem saving the depthmap image file"); return; } @@ -539,7 +562,7 @@ void MainWindow::on_button_save_depthmap_clicked() // save XML and image depthma if (!fs.isOpened()) { // problem ? QApplication::restoreOverrideCursor(); // Restore cursor QMessageBox::critical(this, "File error", - "There was a problem writing the depthmap data file"); + "There was a problem writing the depthmap data file"); return; } @@ -590,8 +613,9 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file { QString filename = QFileDialog::getOpenFileName(this, "Load depthmap from XML file...", QString::fromStdString(basedir + "*-depthmap-data.xml"), "*.xml *.XML"); // file name - if (filename.isNull() || filename.isEmpty()) // cancel ? + if (filename.isNull() || filename.isEmpty()) { // cancel ? return; + } QApplication::setOverrideCursor(Qt::WaitCursor); // wait cursor qApp->processEvents(); @@ -607,25 +631,29 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file basefile = basefile.substr(found+1); // delete ending slash*/ ChangeBaseDir(filename); size_t pos = basefile.find("-depthmap-data"); - if (pos != std::string::npos) + if (pos != std::string::npos) { basefile.erase(pos, basefile.length()); + } else { // doesn't end with "-depthmap-data" QMessageBox::critical(this, "File name error", - "There was a problem reading the depthmap file:\nit must end with ''-depthmap-data.xml''"); + "There was a problem reading the depthmap file:\nit must end with ''-depthmap-data.xml''"); DisableGUI(); // problem : reset GUI elements and exit return; } std::string filesession = filename.toUtf8().constData(); // base file name pos = filesession.find("-depthmap-data.xml"); // ends with "depthmap-data.xml" - if (pos != std::string::npos) filesession.erase(pos, filesession.length()); + if (pos != std::string::npos) { + filesession.erase(pos, filesession.length()); + } depthmap = cv::imread(filesession + "-depthmap-mask.png", IMREAD_COLOR); // load depthmap mask - if (depthmap.channels() > 1) + if (depthmap.channels() > 1) { cvtColor(depthmap, depthmap, COLOR_BGR2GRAY); + } if (depthmap.empty()) { // problem ? QMessageBox::critical(this, "File error", - "There was a problem reading the depthmap mask file:\nit must end with ''-depthmap-mask.png''"); + "There was a problem reading the depthmap mask file:\nit must end with ''-depthmap-mask.png''"); DisableGUI(); return; } @@ -634,14 +662,14 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file if (image.empty()) { QMessageBox::critical(this, "File error", - "There was a problem reading the depthmap image file:\nit must end with ''-depthmap-image.png''"); + "There was a problem reading the depthmap image file:\nit must end with ''-depthmap-image.png''"); DisableGUI(); return; } if ((image.cols != depthmap.cols) | (image.rows != depthmap.rows)) { // image and mask sizes not the same -> not good ! QMessageBox::critical(this, "Image size error", - "The image and mask image size (width and height) differ"); + "The image and mask image size (width and height) differ"); DisableGUI(); return; } @@ -654,7 +682,7 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file if (!fs.isOpened()) { // file not opened, not handled by above instructions QMessageBox::critical(this, "File error", - "There was a problem reading the depthmap data file:\nit must end with ''-depthmap-data.xml''"); + "There was a problem reading the depthmap data file:\nit must end with ''-depthmap-data.xml''"); DisableGUI(); return; } @@ -663,12 +691,12 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file try { // try to read labels data fs["Labels"] >> labels_temp; // load labels } - catch( cv::Exception& e ) // problem ? + catch (cv::Exception& e) // problem ? { const char* err_msg = e.what(); // get error from openCV QMessageBox::critical(this, "XML Depthmap file error", - "There was a problem reading the depthmap XML file\nThe \"Labels\" data is wrong\nError:\n" - + QString(err_msg)); + "There was a problem reading the depthmap XML file\nThe \"Labels\" data is wrong\nError:\n" + + QString(err_msg)); DisableGUI(); return; } @@ -680,7 +708,7 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file if (nbLabels <= 0) { // no labels ? QMessageBox::critical(this, "XML Depthmap file error", - "There was a problem reading the depthmap XML file\nThe data is wrong"); + "There was a problem reading the depthmap XML file\nThe data is wrong"); DisableGUI(); return; } @@ -688,22 +716,22 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file ui->listWidget_labels->blockSignals(true); // don't trigger events when populating labels list for (int i = 0; i < nbLabels; i++) { // for each label to load - std::string name ="###Error###"; // init default values, no error handling this time, the user should see if there was a problem + std::string name = "###Error###"; // init default values, no error handling this time, the user should see if there was a problem int num = -1; - cv::Point point = cv::Point(0,0); + cv::Point point = cv::Point(0, 0); int color = 255; int gtype = 0; int gcurve = 0; - QListWidgetItem *item = new QListWidgetItem (); // create new label item + auto *item = new QListWidgetItem(); // create new label item std::string field; field = "LabelId" + std::to_string(i); // read label id - fs [field] >> num; + fs[field] >> num; item->setData(Qt::UserRole, num); // set it to current label field = "LabelName" + std::to_string(i); // read label name - fs [field] >> name; + fs[field] >> name; item->setText(QString::fromStdString(name)); // set name to current label item->setFlags(Qt::ItemIsSelectable | Qt::ItemIsEnabled); // item enabled and selectable @@ -712,27 +740,27 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file item->setSelected(false); // don't select it ! field = "GradientType" + std::to_string(i); - fs [field] >> gtype; // load gradient type + fs[field] >> gtype; // load gradient type gradients[i].gradient = gradientType(gtype); field = "GradientCurve" + std::to_string(i); - fs [field] >> gcurve; // load curve type + fs[field] >> gcurve; // load curve type gradients[i].curve = curveType(gcurve); field = "GradientBeginColor" + std::to_string(i); - fs [field] >> color; // load gradient begin color + fs[field] >> color; // load gradient begin color gradients[i].beginColor = color; field = "GradientEndColor" + std::to_string(i); - fs [field] >> color; // load gradient end color + fs[field] >> color; // load gradient end color gradients[i].endColor = color; field = "GradientBeginPoint" + std::to_string(i); - fs [field] >> point; // load gradient begin point + fs[field] >> point; // load gradient begin point gradients[i].beginPoint = point; field = "GradientEndPoint" + std::to_string(i); - fs [field] >> point; // load gradient end point + fs[field] >> point; // load gradient end point gradients[i].endPoint = point; } @@ -762,13 +790,17 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file double zoomX = double(ui->label_viewport->width()) / image.cols; // find best fit for the viewport, try vertical and horizontal ratios double zoomY = double(ui->label_viewport->height()) / image.rows; - if (zoomX < zoomY) zoom = zoomX; // the lowest fits the view - else zoom = zoomY; + if (zoomX < zoomY) { + zoom = zoomX; // the lowest fits the view + } + else { + zoom = zoomY; + } oldZoom = zoom; // zoom already good ShowZoomValue(); // display current zoom viewport = Rect(0, 0, image.cols, image.rows); // update viewport size - thumbnail = ResizeImageAspectRatio(image, cv::Size(ui->label_thumbnail->width(),ui->label_thumbnail->height())); // create thumbnail + thumbnail = ResizeImageAspectRatio(image, cv::Size(ui->label_thumbnail->width(), ui->label_thumbnail->height())); // create thumbnail ShowThumbnail(); // update thumbnail view @@ -790,10 +822,11 @@ void MainWindow::on_button_load_depthmap_clicked() // load depthmap XML file void MainWindow::on_button_load_rgbd_clicked() // load previous session { QString filename = QFileDialog::getOpenFileName(this, "Load RGB+D : image...", QString::fromStdString(basedir), - "Images (*.jpg *.JPG *.jpeg *.JPEG *.jp2 *.JP2 *.png *.PNG *.tif *.TIF *.tiff *.TIFF *.bmp *.BMP)"); // reference image file name + "Images (*.jpg *.JPG *.jpeg *.JPEG *.jp2 *.JP2 *.png *.PNG *.tif *.TIF *.tiff *.TIFF *.bmp *.BMP)"); // reference image file name - if (filename.isNull() || filename.isEmpty()) // cancel ? + if (filename.isNull() || filename.isEmpty()) { // cancel ? return; + } QApplication::setOverrideCursor(Qt::WaitCursor); // wait cursor qApp->processEvents(); @@ -818,19 +851,22 @@ void MainWindow::on_button_load_rgbd_clicked() // load previous session } filename = QFileDialog::getOpenFileName(this, "Load RGB+D : depthmap...", QString::fromStdString(basedir), - "Images (*.jpg *.JPG *.jpeg *.JPEG *.jp2 *.JP2 *.png *.PNG *.tif *.TIF *.tiff *.TIFF *.bmp *.BMP)"); // depthmap file name + "Images (*.jpg *.JPG *.jpeg *.JPEG *.jp2 *.JP2 *.png *.PNG *.tif *.TIF *.tiff *.TIFF *.bmp *.BMP)"); // depthmap file name ChangeBaseDir(filename); filesession = filename.toUtf8().constData(); // base file name depthmap = cv::imread(filesession, IMREAD_COLOR); // load depthmap - if (depthmap.channels() > 1) + if (depthmap.channels() > 1) { cvtColor(depthmap, depthmap, COLOR_BGR2GRAY); + } if ((depthmap.empty()) | (image.cols != depthmap.cols) | (image.rows != depthmap.rows)) { // if image and depthmap sizes differ or depthmap empty - if ((image.cols != depthmap.cols) | (image.rows != depthmap.rows)) // sizes differ + if ((image.cols != depthmap.cols) | (image.rows != depthmap.rows)) { // sizes differ QMessageBox::critical(this, "Image size error", "The image and depthmap size (width and height) must be the same."); - else + } + else { QMessageBox::critical(this, "File error", "There was a problem reading the depthmap file"); + } DisableGUI(); return; } @@ -865,13 +901,17 @@ void MainWindow::on_button_load_rgbd_clicked() // load previous session double zoomX = double(ui->label_viewport->width()) / image.cols; // find best fit for viewport, try vertical and horizontal ratios double zoomY = double(ui->label_viewport->height()) / image.rows; - if (zoomX < zoomY) zoom = zoomX; // the lowest fits the view - else zoom = zoomY; + if (zoomX < zoomY) { + zoom = zoomX; // the lowest fits the view + } + else { + zoom = zoomY; + } oldZoom = zoom; // zoom already set ShowZoomValue(); // display current zoom viewport = Rect(0, 0, image.cols, image.rows); // update viewport size - thumbnail = ResizeImageAspectRatio(image, cv::Size(ui->label_thumbnail->width(),ui->label_thumbnail->height())); // create thumbnail + thumbnail = ResizeImageAspectRatio(image, cv::Size(ui->label_thumbnail->width(), ui->label_thumbnail->height())); // create thumbnail CopyFromImage(image, viewport).copyTo(disp_color); // copy only the viewport part of image QPixmap D; @@ -897,14 +937,15 @@ void MainWindow::on_button_save_ply_clicked() // save XML and image depthmap fil { if (!loaded) { // nothing loaded yet = get out QMessageBox::warning(this, "Nothing to save", - "Not now!\n\nBefore anything else, load a Segmentation or Depthmap project"); + "Not now!\n\nBefore anything else, load a Segmentation or Depthmap project"); return; } QString filename = QFileDialog::getSaveFileName(this, "Save mesh to PLY file...", "./" + QString::fromStdString(basedir + basefile + ".ply"), "*.ply *.PLY"); // filename - if (filename.isNull() || filename.isEmpty()) // cancel ? + if (filename.isNull() || filename.isEmpty()) { // cancel ? return; + } QApplication::setOverrideCursor(Qt::WaitCursor); // wait cursor qApp->processEvents(); @@ -953,8 +994,12 @@ void MainWindow::on_verticalScrollBar_viewport_valueChanged() // update viewpor void MainWindow::ZoomPlus() // zoom in { int z = 0; - while (zoom >= zooms[z]) z++; // from lowest to highest value find the next one - if (z == num_zooms) z--; // maximum + while (zoom >= zooms[z]) { + z++; // from lowest to highest value find the next one + } + if (z == num_zooms) { + z--; // maximum + } if (zoom != zooms[z]) { // zoom changed ? QApplication::setOverrideCursor(Qt::SizeVerCursor); // zoom cursor @@ -970,8 +1015,12 @@ void MainWindow::ZoomPlus() // zoom in void MainWindow::ZoomMinus() // zoom out { int z = num_zooms; - while (zoom <= zooms[z]) z--; // from highest to lowest value find the next one - if (z == 0) z++; // minimum + while (zoom <= zooms[z]) { + z--; // from highest to lowest value find the next one + } + if (z == 0) { + z++; // minimum + } if (zoom != zooms[z]) { // zoom changed ? QApplication::setOverrideCursor(Qt::SizeVerCursor); // zoom cursor @@ -1002,8 +1051,12 @@ void MainWindow::on_pushButton_zoom_fit_clicked() // zoom fit from button oldZoom = zoom; double zoomX = double(ui->label_viewport->width()) / image.cols; // find the best value of zoom double zoomY = double(ui->label_viewport->height()) / image.rows; - if (zoomX < zoomY) zoom = zoomX; // less = fit view borders - else zoom = zoomY; + if (zoomX < zoomY) { + zoom = zoomX; // less = fit view borders + } + else { + zoom = zoomY; + } QApplication::setOverrideCursor(Qt::SizeVerCursor); // zoom cursor qApp->processEvents(); @@ -1024,48 +1077,50 @@ void MainWindow::on_pushButton_zoom_100_clicked() // zoom 100% from button void MainWindow::ShowZoomValue() // display zoom percentage in ui { - ui->label_zoom->setText(QString::number(int(zoom * 100)) +"%"); // show new zoom value in percent + ui->label_zoom->setText(QString::number(int(zoom * 100)) + "%"); // show new zoom value in percent } //// Gradients signals void MainWindow::ShowGradient() // show gradient example from current label { - if (!loaded) // no need for that if nothing is loaded ! + if (!loaded) { // no need for that if nothing is loaded ! return; + } - Mat gradient= cv::Mat::zeros(ui->label_gradient->height(), ui->label_gradient->width(), CV_8UC1); // final image to display + Mat gradient = cv::Mat::zeros(ui->label_gradient->height(), ui->label_gradient->width(), CV_8UC1); // final image to display Mat1b msk = cv::Mat::zeros(ui->label_gradient->height(), ui->label_gradient->width(), CV_8UC1); // the gradient needs a mask msk = 255; // fill the mask with non-zero values int row = ui->listWidget_labels->currentRow(); // get current label - cv::Point beginPoint, endPoint; // begin and end points will change with gradient example type + cv::Point beginPoint; + cv::Point endPoint; // begin and end points will change with gradient example type switch (gradients[row].gradient) { - case gradient_flat: { - break; - } - case gradient_linear: { - beginPoint = cv::Point(0, 0); - endPoint = cv::Point(ui->label_gradient->width(), ui->label_gradient->height()); - break; - } - case gradient_doubleLinear: { - beginPoint = cv::Point(ui->label_gradient->width() / 2, ui->label_gradient->height() / 2); - endPoint = cv::Point(0, 0); - break; - } - case gradient_radial: { - beginPoint = cv::Point(ui->label_gradient->width() / 2, ui->label_gradient->height() / 2); - endPoint = cv::Point(ui->label_gradient->width() / 2, -20); - break; - } + case gradient_flat: { + break; + } + case gradient_linear: { + beginPoint = cv::Point(0, 0); + endPoint = cv::Point(ui->label_gradient->width(), ui->label_gradient->height()); + break; + } + case gradient_doubleLinear: { + beginPoint = cv::Point(ui->label_gradient->width() / 2, ui->label_gradient->height() / 2); + endPoint = cv::Point(0, 0); + break; + } + case gradient_radial: { + beginPoint = cv::Point(ui->label_gradient->width() / 2, ui->label_gradient->height() / 2); + endPoint = cv::Point(ui->label_gradient->width() / 2, -20); + break; + } } GradientFillGray(gradients[row].gradient, gradient, msk, - beginPoint, endPoint, - gradients[row].beginColor, gradients[row].endColor, - ui->listWidget_gradient_curve->currentRow()); // fill shape with gray gradient + beginPoint, endPoint, + gradients[row].beginColor, gradients[row].endColor, + ui->listWidget_gradient_curve->currentRow()); // fill shape with gray gradient cvtColor(gradient, gradient, COLOR_GRAY2BGR); QPixmap D; @@ -1074,50 +1129,50 @@ void MainWindow::ShowGradient() // show gradient example from current label QString file; switch (ui->listWidget_gradient_curve->currentRow()) { // file in .qrc resource needed to display curve shape - case curve_linear: { - file = ":/icons/curve-linear.png"; - break; - } - case curve_cosinus: { - file = ":/icons/curve-cosinus.png"; - break; - } - case curve_cosinus2: { - file = ":/icons/curve-cosinus2.png"; - break; - } - case curve_power2: { - file = ":/icons/curve-power2.png"; - break; - } - case curve_power3: { - file = ":/icons/curve-power3.png"; - break; - } - case curve_sigmoid: { - file = ":/icons/curve-sigmoid.png"; - break; - } - case curve_cos2power2: { - file = ":/icons/curve-cos2power2.png"; - break; - } - case curve_cos2sqrt: { - file = ":/icons/curve-cos2sqrt.png"; - break; - } - case curve_undulate: { - file = ":/icons/curve-undulate.png"; - break; - } - case curve_undulate2: { - file = ":/icons/curve-undulate2.png"; - break; - } - case curve_undulate3: { - file = ":/icons/curve-undulate3.png"; - break; - } + case curve_linear: { + file = ":/icons/curve-linear.png"; + break; + } + case curve_cosinus: { + file = ":/icons/curve-cosinus.png"; + break; + } + case curve_cosinus2: { + file = ":/icons/curve-cosinus2.png"; + break; + } + case curve_power2: { + file = ":/icons/curve-power2.png"; + break; + } + case curve_power3: { + file = ":/icons/curve-power3.png"; + break; + } + case curve_sigmoid: { + file = ":/icons/curve-sigmoid.png"; + break; + } + case curve_cos2power2: { + file = ":/icons/curve-cos2power2.png"; + break; + } + case curve_cos2sqrt: { + file = ":/icons/curve-cos2sqrt.png"; + break; + } + case curve_undulate: { + file = ":/icons/curve-undulate.png"; + break; + } + case curve_undulate2: { + file = ":/icons/curve-undulate2.png"; + break; + } + case curve_undulate3: { + file = ":/icons/curve-undulate3.png"; + break; + } } ui->label_curve_image->setPixmap(QPixmap::fromImage(QImage(file), Qt::AutoColor)); // show curve shape @@ -1125,8 +1180,9 @@ void MainWindow::ShowGradient() // show gradient example from current label void MainWindow::on_radioButton_flat_clicked() // flat gradient selected { - if (!loaded) // no need for that if file not loaded ! + if (!loaded) { // no need for that if file not loaded ! return; + } if (ui->radioButton_flat->isChecked()) { BlockGradientsSignals(true); @@ -1134,13 +1190,14 @@ void MainWindow::on_radioButton_flat_clicked() // flat gradient selected BlockGradientsSignals(false); ShowGradient(); // show gradient example ChangeLabelGradient(); // apply effect - } + } } void MainWindow::on_radioButton_linear_clicked() // linear gradient selected { - if (!loaded) + if (!loaded) { return; + } if (ui->radioButton_linear->isChecked()) { BlockGradientsSignals(true); @@ -1153,8 +1210,9 @@ void MainWindow::on_radioButton_linear_clicked() // linear gradient selected void MainWindow::on_radioButton_double_linear_clicked() // bi-linear gradient selected { - if (!loaded) + if (!loaded) { return; + } if (ui->radioButton_double_linear->isChecked()) { BlockGradientsSignals(true); @@ -1167,8 +1225,9 @@ void MainWindow::on_radioButton_double_linear_clicked() // bi-linear gradient se void MainWindow::on_radioButton_radial_clicked() // radial gradient selected { - if (!loaded) + if (!loaded) { return; + } if (ui->radioButton_radial->isChecked()) { BlockGradientsSignals(true); @@ -1181,8 +1240,9 @@ void MainWindow::on_radioButton_radial_clicked() // radial gradient selected void MainWindow::on_horizontalSlider_begin_valueChanged(int value) // begin color for gradient changed { - if (!loaded) // not needed if file not loaded ! + if (!loaded) { // not needed if file not loaded ! return; + } BlockGradientsSignals(true); gradients[ui->listWidget_labels->currentRow()].beginColor = value; // change value in gradients array @@ -1195,8 +1255,9 @@ void MainWindow::on_horizontalSlider_begin_valueChanged(int value) // begin colo void MainWindow::on_horizontalSlider_end_valueChanged(int value) // end color for gradient changed { - if (!loaded) + if (!loaded) { return; + } BlockGradientsSignals(true); gradients[ui->listWidget_labels->currentRow()].endColor = value; @@ -1207,10 +1268,11 @@ void MainWindow::on_horizontalSlider_end_valueChanged(int value) // end color fo ChangeLabelGradient(); } -void MainWindow::on_listWidget_gradient_curve_currentItemChanged(QListWidgetItem *currentItem) // gradient curve shape changed +void MainWindow::on_listWidget_gradient_curve_currentItemChanged(QListWidgetItem * /*currentItem*/) // gradient curve shape changed { - if (!loaded) // not needed if file not loaded ! + if (!loaded) { // not needed if file not loaded ! return; + } BlockGradientsSignals(true); gradients[ui->listWidget_labels->currentRow()].curve = curveType(ui->listWidget_gradient_curve->currentRow()); // change value in gradients array @@ -1257,7 +1319,7 @@ void MainWindow::on_checkBox_3d_anaglyph_clicked() // activate or not anaglyph v ui->openGLWidget_3d->update(); // view 3D scene } -void MainWindow::on_comboBox_3d_tint_currentIndexChanged(int index) // change tint of image in 3D scene +void MainWindow::on_comboBox_3d_tint_currentIndexChanged(int /*index*/) // change tint of image in 3D scene { Mat image_temp = AnaglyphTint(image, ui->comboBox_3d_tint->currentIndex()); ui->openGLWidget_3d->image3D = GammaCorrection(image_temp, ui->doubleSpinBox_gamma->value()); @@ -1266,7 +1328,7 @@ void MainWindow::on_comboBox_3d_tint_currentIndexChanged(int index) // change ti ui->openGLWidget_3d->update(); // view 3D scene } -void MainWindow::on_doubleSpinBox_gamma_valueChanged(double value) +void MainWindow::on_doubleSpinBox_gamma_valueChanged(double /*value*/) { on_comboBox_3d_tint_currentIndexChanged(0); } @@ -1282,19 +1344,20 @@ void MainWindow::on_checkBox_3d_blur_clicked() // blur depthmap for 3D view if (ui->checkBox_3d_blur->isChecked()) { // blur activated Mat depthmap3D_temp; cv::GaussianBlur(depthmap, depthmap3D_temp, - Size(ui->horizontalSlider_blur_amount->value() * 2 + 1, - ui->horizontalSlider_blur_amount->value() * 2 + 1), 0, 0); // gaussian blur image + Size(ui->horizontalSlider_blur_amount->value() * 2 + 1, + ui->horizontalSlider_blur_amount->value() * 2 + 1), 0, 0); // gaussian blur image ui->openGLWidget_3d->depthmap3D = depthmap3D_temp; // copy blurred depthmap } - else // no blur + else { // no blur ui->openGLWidget_3d->depthmap3D = depthmap; // copy original depthmap + } ui->openGLWidget_3d->updateVertices3D = true; // recompute 3D scene ui->openGLWidget_3d->updateAllVertices3D = true; // for all image ui->openGLWidget_3d->update(); // view 3D scene } -void MainWindow::on_horizontalSlider_blur_amount_valueChanged(int value) // change blur amount for 3D scene +void MainWindow::on_horizontalSlider_blur_amount_valueChanged(int /*value*/) // change blur amount for 3D scene { on_checkBox_3d_blur_clicked(); // view 3D scene with blur } @@ -1354,8 +1417,9 @@ void MainWindow::on_spinBox_3d_frames_valueChanged(int value) // change number o ui->checkBox_3d_infinite_cycle->setChecked(false); // uncheck infinite button ui->checkBox_3d_double_cycle->setChecked(false); // uncheck double cycle button } - else // not unique view + else { // not unique view ui->checkBox_3d_capture_unique->setChecked(false); // uncheck unique button + } } void MainWindow::on_checkBox_3d_capture_unique_clicked() // only one frame for 3D animation @@ -1386,8 +1450,9 @@ void MainWindow::on_checkBox_3d_capture_fullscreen_clicked() // activate fullscr QRect screenSize = qApp->desktop()->availableGeometry(qApp->desktop()->primaryScreen()); // get screen size in which app is run ui->spinBox_3d_resolution->setValue(screenSize.width()); // set screen size width to 3D resolution } - else // not fullscreen + else { // not fullscreen ui->spinBox_3d_resolution->setValue(ui->openGLWidget_3d->width()); // width = widget width + } } void MainWindow::on_checkBox_3d_double_cycle_clicked() // activate to-and-from cycle for 3D animation @@ -1411,12 +1476,15 @@ void MainWindow::on_checkBox_3d_capture_clicked() // launch 3D animation and sav if (ui->checkBox_3d_save_files->isChecked()) { // save to image option activated ? QString filename = QFileDialog::getSaveFileName(this, "Save 3D capture to images...", "./" + QString::fromStdString(basedir + basefile + "-capture.png"), "*.png *.PNG"); // get filename - if (filename.isNull() || filename.isEmpty()) // cancel ? + if (filename.isNull() || filename.isEmpty()) { // cancel ? return; + } filesession = filename.toUtf8().constData(); // base file name size_t pos = filesession.find(".png"); - if (pos != std::string::npos) filesession.erase(pos, filesession.length()); // delete extension + if (pos != std::string::npos) { + filesession.erase(pos, filesession.length()); // delete extension + } } QApplication::setOverrideCursor(Qt::WaitCursor); // wait cursor @@ -1448,23 +1516,31 @@ void MainWindow::on_checkBox_3d_capture_clicked() // launch 3D animation and sav //bool circular = ui->checkBox_3d_circular->isChecked(); // circular ? double middleX = double(ui->spinBox_3d_x_angle->value()) / 2; // halves of angles double middleY = double(ui->spinBox_3d_y_angle->value()) / 2; - double xSector, ySector; - xSector = double(ui->spinBox_3d_x_angle->value()) / (nb_frames); // angle value of one sector - ySector = double(ui->spinBox_3d_y_angle->value()) / (nb_frames); + double xSector; + double ySector; + xSector = double(ui->spinBox_3d_x_angle->value()) / (nb_frames); // angle value of one sector + ySector = double(ui->spinBox_3d_y_angle->value()) / (nb_frames); int begin; // 1st frame value - if (ui->checkBox_3d_double_cycle->isChecked()) // 1st frame value depends on mode (circular, double cycle) + if (ui->checkBox_3d_double_cycle->isChecked()) { // 1st frame value depends on mode (circular, double cycle) begin = -nb_frames; - else begin = 0; + } + else { + begin = 0; + } abort_3d = false; // used to interrupt capture int counter = 1; // used to save images only once if infinite cycle set int progress = 1; // for progress bar // progress bar init - if (ui->checkBox_3d_double_cycle->isChecked()) // double cycle ? + if (ui->checkBox_3d_double_cycle->isChecked()) { // double cycle ? ui->progressBar_3d_capture->setMaximum(nb_frames * 2); // max of progress bar is twice the amount + } else // not double cycle - if (!ui->checkBox_3d_capture_unique->isChecked()) // more than 1 frame ? + if (!ui->checkBox_3d_capture_unique->isChecked()) { // more than 1 frame ? ui->progressBar_3d_capture->setMaximum(1); // max of progress bar is the number of frames - else ui->progressBar_3d_capture->setMaximum(1); // else max of progress bar is 1 (setting it to 0 makes the progress bar shuffle) + } + else { + ui->progressBar_3d_capture->setMaximum(1); // else max of progress bar is 1 (setting it to 0 makes the progress bar shuffle) + } ui->progressBar_3d_capture->setValue(0); // 0% done in progress bar // compute frames @@ -1484,61 +1560,80 @@ void MainWindow::on_checkBox_3d_capture_clicked() // launch 3D animation and sav qApp->processEvents(); // force refresh of GUI } - else + else { for (int frame = begin; frame <= nb_frames; frame++) { // for each frame if (abort_3d) { // abort ? //QMessageBox::critical(this, "3D capture", "Capture was interrupted"); ui->progressBar_3d_capture->setValue(0); // reset progress bar and exit break; } - double xAngle, yAngle; + double xAngle; + double yAngle; if (ui->checkBox_3d_circular_vertical->isChecked()) { int frm; - if (frame < 0) + if (frame < 0) { frm = -frame; - else + } + else { frm = frame; - if (frm == 0) // compute current angles of view on x and y axes for each interval + } + if (frm == 0) { // compute current angles of view on x and y axes for each interval xAngle = -middleX; - else if (frm < double(nb_frames) / 4) // 1st quarter + } + else if (frm < double(nb_frames) / 4) { // 1st quarter xAngle = xSector * double(frm) * 2 - middleX; - else if (frm < double(nb_frames) / 2) // 2nd quarter + } + else if (frm < double(nb_frames) / 2) { // 2nd quarter xAngle = xSector * double(frm) * 2 - middleX; - else if (frm < double(nb_frames) * 3 / 4) // 3rd quarter + } + else if (frm < double(nb_frames) * 3 / 4) { // 3rd quarter xAngle = middleX - xSector * (double(frm) - double(nb_frames) / 2) * 2; - else // 4th quarter + } + else { // 4th quarter xAngle = middleX - xSector * (double(frm) - double(nb_frames) / 2) * 2; + } } if (ui->checkBox_3d_circular_horizontal->isChecked()) { int frm; - if (frame < 0) + if (frame < 0) { frm = nb_frames + frame; - else + } + else { frm = frame; - if (frm == 0) // compute current angles of view on x and y axes for each interval + } + if (frm == 0) { // compute current angles of view on x and y axes for each interval yAngle = 0; - else if (frm < double(nb_frames) / 4) // 1st quarter + } + else if (frm < double(nb_frames) / 4) { // 1st quarter yAngle = ySector * double(frm) * 2; - else if (frm < double(nb_frames) / 2) // 2nd quarter + } + else if (frm < double(nb_frames) / 2) { // 2nd quarter yAngle = middleY - ySector * (double(frm) - double(nb_frames) / 4) * 2; - else if (frm < double(nb_frames) * 3 / 4) // 3rd quarter + } + else if (frm < double(nb_frames) * 3 / 4) { // 3rd quarter yAngle = middleY - ySector * (double(frm) - double(nb_frames) / 4) * 2; - else // 4th quarter + } + else { // 4th quarter yAngle = ySector * (double(frm) - double(nb_frames) * 3 / 4) * 2 - middleY; + } } if (!ui->checkBox_3d_circular_vertical->isChecked()) { // vertical wobble xAngle = xSector * double(frame) - middleX; // compute angles - if (frame < 0) // in double cycle mode angle is not the same for "negative" frames + if (frame < 0) { // in double cycle mode angle is not the same for "negative" frames xAngle = double(-ui->spinBox_3d_x_angle->value()) - xAngle; - if (ui->checkBox_3d_double_cycle->isChecked()) // in double cycle mode invert angles + } + if (ui->checkBox_3d_double_cycle->isChecked()) { // in double cycle mode invert angles xAngle = -xAngle; + } } if (!ui->checkBox_3d_circular_horizontal->isChecked()) { // horizontal wobble yAngle = ySector * double(frame) - middleY; - if (frame < 0) // in double cycle mode angle is not the same for "negative" frames + if (frame < 0) { // in double cycle mode angle is not the same for "negative" frames yAngle = double(-ui->spinBox_3d_y_angle->value()) - yAngle; - if (ui->checkBox_3d_double_cycle->isChecked()) // in double cycle mode invert angles + } + if (ui->checkBox_3d_double_cycle->isChecked()) { // in double cycle mode invert angles yAngle = -yAngle; + } } ui->openGLWidget_3d->xRot = xAngle; // set angles in 3D scene @@ -1562,35 +1657,41 @@ void MainWindow::on_checkBox_3d_capture_clicked() // launch 3D animation and sav qApp->processEvents(); // force refresh of GUI if ((ui->checkBox_3d_infinite_cycle->isChecked()) & (frame == nb_frames)) { // infinite cycle and end of animation reached ? - if (ui->checkBox_3d_double_cycle->isChecked()) + if (ui->checkBox_3d_double_cycle->isChecked()) { frame = -frame; // return to 1st frame - else + } + else { frame = 0; + } counter++; // end of animation reached = no more writing image to file } progress++; // increase value for progress bar - the 1st time progress is already equal to 1 that's why it is increased after updating the progress bar - if (progress > 10000) // if infinite loop don't go too far + if (progress > 10000) { // if infinite loop don't go too far progress = nb_frames + 1; // reset progress value to just over 100% + } } + } ui->openGLWidget_3d->xRot = rX; // restore initial 3D scene values ui->openGLWidget_3d->yRot = rY; ui->openGLWidget_3d->zRot = rZ; - if (ui->checkBox_3d_capture_fullscreen->isChecked()) // full screen ? + if (ui->checkBox_3d_capture_fullscreen->isChecked()) { // full screen ? this->setWindowFlags((((windowFlags() | Qt::CustomizeWindowHint) - & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); - show(); - ui->openGLWidget_3d->move(QPoint(saveXOpenGL, saveYOpenGL)); // move back openGL widget - ui->spinBox_3D_rotate_x->raise(); // bring back controls over 3D view - ui->spinBox_3D_rotate_y->raise(); - ui->horizontalSlider_3D_rotate_y->raise(); - ui->verticalSlider_3D_rotate_x->raise(); - ui->button_3d_reset->raise(); + & ~Qt::WindowCloseButtonHint) | Qt::WindowMinMaxButtonsHint)); + } + show(); + ui->openGLWidget_3d->move(QPoint(saveXOpenGL, saveYOpenGL)); // move back openGL widget + ui->spinBox_3D_rotate_x->raise(); // bring back controls over 3D view + ui->spinBox_3D_rotate_y->raise(); + ui->horizontalSlider_3D_rotate_y->raise(); + ui->verticalSlider_3D_rotate_x->raise(); + ui->button_3d_reset->raise(); ui->openGLWidget_3d->resize(saveWidthOpenGL, saveHeightOpenGL); // and resize it - if ((ui->spinBox_3d_resolution->value() == ui->openGLWidget_3d->width()) & (!ui->checkBox_3d_capture_fullscreen->isChecked())) - ui->openGLWidget_3d->update(); // force update of 3D scene if width of widget hasn't changed + if ((ui->spinBox_3d_resolution->value() == ui->openGLWidget_3d->width()) & (!ui->checkBox_3d_capture_fullscreen->isChecked())) { + ui->openGLWidget_3d->update(); // force update of 3D scene if width of widget hasn't changed + } ui->frame_3D_capture->setEnabled(true); // activate capture panel ui->checkBox_3d_capture->setChecked(false); // set capture button to initial state @@ -1615,22 +1716,30 @@ void MainWindow::keyReleaseEvent(QKeyEvent *keyEvent) // spacebar = move view in QApplication::restoreOverrideCursor(); // Restore cursor } else { - if (keyEvent->key() == Qt::Key_Left) + if (keyEvent->key() == Qt::Key_Left) { ui->openGLWidget_3d->SetShiftLeft(); - if (keyEvent->key() == Qt::Key_Right) + } + if (keyEvent->key() == Qt::Key_Right) { ui->openGLWidget_3d->SetShiftRight(); - if (keyEvent->key() == Qt::Key_Up) + } + if (keyEvent->key() == Qt::Key_Up) { ui->openGLWidget_3d->SetShiftUp(); - if (keyEvent->key() == Qt::Key_Down) + } + if (keyEvent->key() == Qt::Key_Down) { ui->openGLWidget_3d->SetShiftDown(); - if (keyEvent->key() == Qt::Key_PageUp) + } + if (keyEvent->key() == Qt::Key_PageUp) { ui->openGLWidget_3d->SetAngleUp(); - if (keyEvent->key() == Qt::Key_PageDown) + } + if (keyEvent->key() == Qt::Key_PageDown) { ui->openGLWidget_3d->SetAngleDown(); - if (keyEvent->key() == Qt::Key_Home) + } + if (keyEvent->key() == Qt::Key_Home) { ui->openGLWidget_3d->SetAngleLeft(); - if (keyEvent->key() == Qt::Key_End) + } + if (keyEvent->key() == Qt::Key_End) { ui->openGLWidget_3d->SetAngleRight(); + } } } @@ -1666,7 +1775,7 @@ void MainWindow::keyPressEvent(QKeyEvent *keyEvent) // special keys /////////////////// Mouse events ////////////////////// -void MainWindow::mouseReleaseEvent(QMouseEvent *eventRelease) // event triggered by releasing mouse button +void MainWindow::mouseReleaseEvent(QMouseEvent * /*eventRelease*/) // event triggered by releasing mouse button { QApplication::restoreOverrideCursor(); // Restore cursor @@ -1684,11 +1793,13 @@ void MainWindow::mouseReleaseEvent(QMouseEvent *eventRelease) // event triggered void MainWindow::mousePressEvent(QMouseEvent *eventPress) // event triggered by a mouse click { - if (!loaded) return; // no image = get out + if (!loaded) { + return; // no image = get out + } if (ui->label_viewport->underMouse()) { // mouse over viewport ? - bool key_control = QGuiApplication::queryKeyboardModifiers().testFlag(Qt::ControlModifier); // modifier keys pressed ? (shift control etc) - bool key_alt = QGuiApplication::queryKeyboardModifiers().testFlag(Qt::AltModifier); + //bool key_control = QGuiApplication::queryKeyboardModifiers().testFlag(Qt::ControlModifier); // modifier keys pressed ? (shift control etc) + //bool key_alt = QGuiApplication::queryKeyboardModifiers().testFlag(Qt::AltModifier); mouse_origin = ui->label_viewport->mapFromGlobal(QCursor::pos()); // current mouse position, save it cv::Point pos = Viewport2Image(cv::Point(mouse_origin.x(), mouse_origin.y())); // convert from viewport to image coordinates @@ -1700,30 +1811,26 @@ void MainWindow::mousePressEvent(QMouseEvent *eventPress) // event triggered by return; } if ((mouseButton == Qt::MiddleButton) & (pos.x >= 0) & (pos.x < image.cols) - & (pos.y >= 0) & (pos.y < image.rows)) { // right button = get gray value + & (pos.y >= 0) & (pos.y < image.rows)) { // right button = get gray value uchar color = depthmap.at(pos.y, pos.x); // get gray "color" under mouse cursor ui->label_gray_level->setText(QString::number(color)); // show it return; } - else if (loaded) { + if (loaded) { int row = ui->listWidget_labels->currentRow(); // get current label number in list if (mouseButton == Qt::LeftButton) { // left mouse button ? - int size; - if (zoom < 1) - size = 1.0 / zoom * 6; - else - size = zoom * 6; + int size = (zoom < 1) ? 1.0 / zoom * 6 : zoom * 6; if ((pos.x >= gradients[row].endPoint.x - size - 1) & (pos.y >= gradients[row].endPoint.y - size - 1) - & (pos.x <= gradients[row].endPoint.x + size + 1) & (pos.y <= gradients[row].endPoint.y + size + 1)) { // mouse cursor over end point - moveEnd = true; // begin moving it + & (pos.x <= gradients[row].endPoint.x + size + 1) & (pos.y <= gradients[row].endPoint.y + size + 1)) { // mouse cursor over end point + moveEnd = true; // begin moving it } else if ((pos.x >= gradients[row].beginPoint.x - size - 1) & (pos.y >= gradients[row].beginPoint.y - size - 1) - & (pos.x <= gradients[row].beginPoint.x + size + 1) & (pos.y <= gradients[row].beginPoint.y + size + 1)) { // mouse cursor over begin point + & (pos.x <= gradients[row].beginPoint.x + size + 1) & (pos.y <= gradients[row].beginPoint.y + size + 1)) { // mouse cursor over begin point moveBegin = true; // begin moving it } else if ((pos.x >= 0) & (pos.x < image.cols) - & (pos.y >= 0) & (pos.y < image.rows)) { // select label in viewport + & (pos.y >= 0) & (pos.y < image.rows)) { // select label in viewport int value = labels.at(pos.y, pos.x); // get label mask value under mouse cursor for (int i = 0; i < ui->listWidget_labels->count(); i++) { // find clicked label id QListWidgetItem *item = ui->listWidget_labels->item(i); // pointer on item @@ -1733,7 +1840,7 @@ void MainWindow::mousePressEvent(QMouseEvent *eventPress) // event triggered by } } } - } + } } } else if (ui->label_thumbnail->underMouse()) { // if mouse over thumbnail @@ -1744,7 +1851,7 @@ void MainWindow::mousePressEvent(QMouseEvent *eventPress) // event triggered by // convert middle of viewport from thumbnail to image coordinates int middleX = double(mouse_pos.x() - (ui->label_thumbnail->width() - ui->label_thumbnail->pixmap()->width()) / 2) / ui->label_thumbnail->pixmap()->width() * image.cols; int middleY = double(mouse_pos.y() - (ui->label_thumbnail->height() - ui->label_thumbnail->pixmap()->height()) / 2) / ui->label_thumbnail->pixmap()->height() * image.rows; - SetViewportXY(middleX - viewport.width /2, middleY - viewport.height /2); // set viewport to new middle position + SetViewportXY(middleX - viewport.width / 2, middleY - viewport.height / 2); // set viewport to new middle position ShowThumbnail(); // show thumbnail Render(); // show result @@ -1752,9 +1859,11 @@ void MainWindow::mousePressEvent(QMouseEvent *eventPress) // event triggered by } } -void MainWindow::mouseMoveEvent(QMouseEvent *eventMove) // first mouse click has already been treated and is holded down +void MainWindow::mouseMoveEvent(QMouseEvent * /*eventMove*/) // first mouse click has already been treated and is holded down { - if (!loaded) return;// no image = get out + if (!loaded) { + return;// no image = get out + } if (ui->label_thumbnail->underMouse()) { // mouse over thumbnail ? QPoint mouse_pos = ui->label_thumbnail->mapFromGlobal(QCursor::pos()); // current mouse position @@ -1763,7 +1872,7 @@ void MainWindow::mouseMoveEvent(QMouseEvent *eventMove) // first mouse click has // convert middle of viewport from thumbnail to image coordinates int middleX = double(mouse_pos.x() - (ui->label_thumbnail->width() - ui->label_thumbnail->pixmap()->width()) / 2) / ui->label_thumbnail->pixmap()->width() * image.cols; int middleY = double(mouse_pos.y() - (ui->label_thumbnail->height() - ui->label_thumbnail->pixmap()->height()) / 2) / ui->label_thumbnail->pixmap()->height() * image.rows; - SetViewportXY(middleX - viewport.width /2, middleY - viewport.height /2); // set viewport to new middle position + SetViewportXY(middleX - viewport.width / 2, middleY - viewport.height / 2); // set viewport to new middle position ShowThumbnail(); // show thumbnail Render(); // display result @@ -1774,10 +1883,18 @@ void MainWindow::mouseMoveEvent(QMouseEvent *eventMove) // first mouse click has QPoint mouse_pos = ui->label_viewport->mapFromGlobal(QCursor::pos()); // current mouse position cv::Point pos = Viewport2Image(cv::Point(mouse_pos.x(), mouse_pos.y())); // convert from viewport to image coordinates - if (pos.x < 1) pos.x = 1; // mouse coordinates can't be outside image - if (pos.y < 1) pos.y = 1; - if (pos.x > image.cols-2) pos.x = image.cols - 2; - if (pos.y > image.rows-2) pos.y = image.rows - 2; + if (pos.x < 1) { + pos.x = 1; // mouse coordinates can't be outside image + } + if (pos.y < 1) { + pos.y = 1; + } + if (pos.x > image.cols - 2) { + pos.x = image.cols - 2; + } + if (pos.y > image.rows - 2) { + pos.y = image.rows - 2; + } if (moveBegin) { // move base of label vector int row = ui->listWidget_labels->currentRow(); // get current label @@ -1790,7 +1907,7 @@ void MainWindow::mouseMoveEvent(QMouseEvent *eventMove) // first mouse click has ChangeLabelGradient(); } else if ((mouseButton == Qt::MiddleButton) & (pos.x >= 0) & (pos.x < image.cols) - & (pos.y >= 0) & (pos.y < image.rows)) { // show gray "color" under mouse cursor + & (pos.y >= 0) & (pos.y < image.rows)) { // show gray "color" under mouse cursor uchar color = depthmap.at(pos.y, pos.x); // get color in labels mask ui->label_gray_level->setText(QString::number(color)); // show it return; @@ -1811,8 +1928,9 @@ void MainWindow::mouseMoveEvent(QMouseEvent *eventMove) // first mouse click has void MainWindow::wheelEvent(QWheelEvent *wheelEvent) // mouse wheel turned { - if (!loaded) + if (!loaded) { return;// no image = get out + } if (ui->label_viewport->underMouse()) { // if mouse over viewport mouse_origin = ui->label_viewport->mapFromGlobal(QCursor::pos()); // mouse position @@ -1836,7 +1954,7 @@ cv::Point MainWindow::Viewport2Image(const cv::Point &p) // convert viewport to { int posX = double(p.x - (ui->label_viewport->width() - ui->label_viewport->pixmap()->width()) / 2) / zoom + viewport.x; int posY = double(p.y - (ui->label_viewport->height() - ui->label_viewport->pixmap()->height()) / 2) / zoom + viewport.y; - return cv::Point(posX, posY); + return { posX, posY }; } void MainWindow::ShowThumbnail() // image thumnail with transparent viewport indicator @@ -1846,7 +1964,10 @@ void MainWindow::ShowThumbnail() // image thumnail with transparent viewport ind return; } - double p1x, p1y, p2x, p2y; // rectangle position + double p1x; + double p1y; + double p2x; + double p2y; // rectangle position if (viewport.width == image.cols) { // entire horizontal view ? p1x = 0; p2x = ui->label_thumbnail->pixmap()->width(); @@ -1860,7 +1981,7 @@ void MainWindow::ShowThumbnail() // image thumnail with transparent viewport ind if (viewport.height == image.rows) { // entire vertical view ? p1y = 0; - p2y= ui->label_thumbnail->pixmap()->height(); + p2y = ui->label_thumbnail->pixmap()->height(); } else { // partial view p1y = double(viewport.y) / image.rows; @@ -1870,8 +1991,8 @@ void MainWindow::ShowThumbnail() // image thumnail with transparent viewport ind } Mat thumbnail_temp = Mat::zeros(thumbnail.rows, thumbnail.cols, CV_8UC3); // create thumbnail mask - rectangle(thumbnail_temp, cv::Point(p1x, p1y), cv::Point(p2x-1, p2y-1), Scalar(92,92,92), -1); // draw filled rectangle representing the viewport position - rectangle(thumbnail_temp, cv::Point(p1x, p1y), cv::Point(p2x-1, p2y-1), Scalar(255,255,255), 2); // draw rectangle with thick border + rectangle(thumbnail_temp, cv::Point(p1x, p1y), cv::Point(p2x - 1, p2y - 1), Scalar(92, 92, 92), -1); // draw filled rectangle representing the viewport position + rectangle(thumbnail_temp, cv::Point(p1x, p1y), cv::Point(p2x - 1, p2y - 1), Scalar(255, 255, 255), 2); // draw rectangle with thick border cv::addWeighted(thumbnail, 1, thumbnail_temp, 0.25, 0, thumbnail_temp, -1); // add to thumbnail with transparency ui->label_thumbnail->setPixmap(Mat2QPixmap(thumbnail_temp)); // update thumbnail view } @@ -1888,18 +2009,30 @@ void MainWindow::UpdateViewportDimensions() // recompute viewport width and heig viewport.height = image.rows; viewport.y = 0; } - if (viewport.x > (image.cols - viewport.width)) viewport.x = image.cols - viewport.width; // can't be out of image at right-bottom - if (viewport.y > (image.rows - viewport.height)) viewport.y = image.rows - viewport.height; + if (viewport.x > (image.cols - viewport.width)) { + viewport.x = image.cols - viewport.width; // can't be out of image at right-bottom + } + if (viewport.y > (image.rows - viewport.height)) { + viewport.y = image.rows - viewport.height; + } } void MainWindow::SetViewportXY(const int &x, const int &y) // set viewport top-left to (x,y) new coordinates { viewport.x = x; viewport.y = y; - if (viewport.x < 0) viewport.x = 0; // can't be out of image - if (viewport.x > image.cols - viewport.width) viewport.x = image.cols - viewport.width; - if (viewport.y < 0) viewport.y = 0; - if (viewport.y > image.rows - viewport.height) viewport.y = image.rows - viewport.height; + if (viewport.x < 0) { + viewport.x = 0; // can't be out of image + } + if (viewport.x > image.cols - viewport.width) { + viewport.x = image.cols - viewport.width; + } + if (viewport.y < 0) { + viewport.y = 0; + } + if (viewport.y > image.rows - viewport.height) { + viewport.y = image.rows - viewport.height; + } ui->horizontalScrollBar_viewport->blockSignals(true); // horizontal scrollbars must not trigger any action ui->horizontalScrollBar_viewport->setValue(viewport.x); // new x value @@ -1911,10 +2044,12 @@ void MainWindow::SetViewportXY(const int &x, const int &y) // set viewport top-l void MainWindow::Render() // show masks for image + depthmap + selection { - if (!loaded) // no image to display + if (!loaded) { // no image to display return; + } - int oldMiddleX, oldMiddleY; + int oldMiddleX; + int oldMiddleY; if (oldZoom != zoom) { // zoom changed ? oldMiddleX = viewport.x + viewport.width / 2; // current middle of viewport for zooming to center of image oldMiddleY = viewport.y + viewport.height / 2; @@ -1926,8 +2061,8 @@ void MainWindow::Render() // show masks for image + depthmap + selection UpdateViewportDimensions(); // recompute viewport width and height double newPosX = oldMiddleX - viewport.width / 2; // compute new middle of viewport double newPosY = oldMiddleY - viewport.height / 2; - ui->horizontalScrollBar_viewport->setMaximum(image.cols-viewport.width); // update scrollbars range - ui->verticalScrollBar_viewport->setMaximum(image.rows-viewport.height); + ui->horizontalScrollBar_viewport->setMaximum(image.cols - viewport.width); // update scrollbars range + ui->verticalScrollBar_viewport->setMaximum(image.rows - viewport.height); SetViewportXY(newPosX, newPosY); // set top-left coordinates of viewport to new value ShowZoomValue(); // display new zoom value oldZoom = zoom; // backup zoom value @@ -1935,14 +2070,16 @@ void MainWindow::Render() // show masks for image + depthmap + selection Mat image_temp = CopyFromImage(image, viewport); // copy only zoomed part of image disp_color = Mat::zeros(image_temp.rows, image_temp.cols, CV_8UC3); // empty new view - if (ui->checkBox_image->isChecked()) // image mask with transparency - cv::addWeighted(disp_color, 1-double(ui->horizontalSlider_blend_image->value()) / 100, - image_temp, double(ui->horizontalSlider_blend_image->value()) / 100, - 0, disp_color, -1); - if (ui->checkBox_depthmap->isChecked() & (!depthmap.empty())) // depthmap with transparency + if (ui->checkBox_image->isChecked()) { // image mask with transparency + cv::addWeighted(disp_color, 1 - double(ui->horizontalSlider_blend_image->value()) / 100, + image_temp, double(ui->horizontalSlider_blend_image->value()) / 100, + 0, disp_color, -1); + } + if (ui->checkBox_depthmap->isChecked() & (!depthmap.empty())) { // depthmap with transparency cv::addWeighted(disp_color, 1, - CopyFromImage(depthmap, viewport), double(ui->horizontalSlider_blend_depthmap->value()) / 100, - 0, disp_color, -1); + CopyFromImage(depthmap, viewport), double(ui->horizontalSlider_blend_depthmap->value()) / 100, + 0, disp_color, -1); + } if ((!selection.empty()) & (loaded)) { // something selected ? Mat selection_temp; selection.copyTo(selection_temp); // make a copy of selection mask @@ -1951,37 +2088,45 @@ void MainWindow::Render() // show masks for image + depthmap + selection // draw end point : a blue tiny rectangle crossed by diagonals int size; - if (zoom < 1) + if (zoom < 1) { size = 1.0 / zoom * 6; - else + } + else { size = zoom * 6; + } cv::rectangle(selection_temp, Rect(gradients[row].endPoint.x - size, gradients[row].endPoint.y - size, size * 2 + 1, size * 2 + 1), - Vec3b(255, 0, 0), 2); + Vec3b(255, 0, 0), 2); cv::line(selection_temp, cv::Point(gradients[row].endPoint.x - size, gradients[row].endPoint.y - size), - cv::Point(gradients[row].endPoint.x + size, gradients[row].endPoint.y + size), - Vec3b(255, 0, 0), 2); + cv::Point(gradients[row].endPoint.x + size, gradients[row].endPoint.y + size), + Vec3b(255, 0, 0), 2); cv::line(selection_temp, cv::Point(gradients[row].endPoint.x - size, gradients[row].endPoint.y + size), - cv::Point(gradients[row].endPoint.x + size, gradients[row].endPoint.y - size), - Vec3b(255, 0, 0), 2); + cv::Point(gradients[row].endPoint.x + size, gradients[row].endPoint.y - size), + Vec3b(255, 0, 0), 2); // draw begin point : a red tiny rectangle crossed by diagonals cv::rectangle(selection_temp, Rect(gradients[row].beginPoint.x - size, gradients[row].beginPoint.y - size, size * 2 + 1, size * 2 + 1), - Vec3b(0, 0, 255), 2); + Vec3b(0, 0, 255), 2); cv::line(selection_temp, cv::Point(gradients[row].beginPoint.x - size, gradients[row].beginPoint.y - size), - cv::Point(gradients[row].beginPoint.x + size, gradients[row].beginPoint.y + size), - Vec3b(0, 0, 255), 2); + cv::Point(gradients[row].beginPoint.x + size, gradients[row].beginPoint.y + size), + Vec3b(0, 0, 255), 2); cv::line(selection_temp, cv::Point(gradients[row].beginPoint.x - size, gradients[row].beginPoint.y + size), - cv::Point(gradients[row].beginPoint.x + size, gradients[row].beginPoint.y - size), - Vec3b(0, 0, 255), 2); + cv::Point(gradients[row].beginPoint.x + size, gradients[row].beginPoint.y - size), + Vec3b(0, 0, 255), 2); Vec3b CO; // color of vector - if ((moveBegin) | (moveEnd)) // is it being moved ? + if ((moveBegin) | (moveEnd)) { // is it being moved ? CO = Vec3b(0, 255, 0); // yes = green - else + } + else { CO = Vec3b(255, 255, 255); // no = white + } cv::arrowedLine(selection_temp, gradients[row].beginPoint, gradients[row].endPoint, CO, 1, 8, 0, 0.2); // draw vector selection_temp = CopyFromImage(selection_temp, viewport); // only keep the view part of selection mask - if (zoom <= 1) selection_temp = DilatePixels(selection_temp, int(1/zoom)); // dilation depends of zoom - else selection_temp = DilatePixels(selection_temp, 3-zoom); + if (zoom <= 1) { + selection_temp = DilatePixels(selection_temp, int(1 / zoom)); // dilation depends of zoom + } + else { + selection_temp = DilatePixels(selection_temp, 3 - zoom); + } /*cv::addWeighted(disp_color, 1, selection_temp, 0.25, 0, disp_color, -1);*/ @@ -2018,9 +2163,9 @@ void MainWindow::ChangeLabelGradient() // update depthmap mask with gradient int row = ui->listWidget_labels->currentRow(); // get current label row in list GradientFillGray(gradients[row].gradient, depthmap, currentLabelMask, - gradients[row].beginPoint, gradients[row].endPoint, - gradients[row].beginColor, gradients[row].endColor, - gradients[row].curve, selection_rect); // fill depthmap mask with gradient + gradients[row].beginPoint, gradients[row].endPoint, + gradients[row].beginColor, gradients[row].endColor, + gradients[row].curve, selection_rect); // fill depthmap mask with gradient ui->openGLWidget_3d->depthmap3D = depthmap; updateVertices3D = true; @@ -2055,13 +2200,14 @@ void MainWindow::readPositionSettings() restoreState(qsettings.value("savestate", saveState()).toByteArray()); move(qsettings.value("pos", pos()).toPoint()); resize(qsettings.value("size", size()).toSize()); - if (qsettings.value("maximized", true).toBool()) + if (qsettings.value("maximized", true).toBool()) { showMaximized(); + } qsettings.endGroup(); } -void MainWindow::moveEvent(QMoveEvent*) +void MainWindow::moveEvent(QMoveEvent* /*event*/) { writePositionSettings(); } diff --git a/mainwindow.h b/mainwindow.h index 37f1976..2f9f4a2 100644 --- a/mainwindow.h +++ b/mainwindow.h @@ -55,7 +55,7 @@ private slots: void on_listWidget_labels_currentItemChanged(QListWidgetItem *currentItem); // show current label color when item change //// load & save - void ChangeBaseDir(QString filename); // Set base dir and file + void ChangeBaseDir(const QString& filename); // Set base dir and file void SaveDirBaseFile(); // just to keep the last open dir void DisableGUI(); diff --git a/mat-image-tools.cpp b/mat-image-tools.cpp index a44f2ca..3cbdccf 100644 --- a/mat-image-tools.cpp +++ b/mat-image-tools.cpp @@ -14,6 +14,8 @@ #include "mat-image-tools.h" +#include + using namespace std; using namespace cv; @@ -23,12 +25,12 @@ using namespace cv; bool IsRGBColorDark(int red, int green, int blue) // tell if a RGB color is dark or not { - double brightness; - brightness = (red * 299) + (green * 587) + (blue * 114); - brightness = brightness / 255000; + double brightness; + brightness = (red * 299) + (green * 587) + (blue * 114); + brightness = brightness / 255000; - // values range from 0 to 1 : anything greater than 0.5 should be bright enough for dark text - return (brightness <= 0.5); + // values range from 0 to 1 : anything greater than 0.5 should be bright enough for dark text + return (brightness <= 0.5); } /////////////////////////////////////////////////////////// @@ -39,42 +41,42 @@ Mat QImage2Mat(const QImage &source) // Convert QImage to Mat, every number of c { switch (source.format()) { // which format ? - case QImage::Format_ARGB32: // 8-bit, 4 channel - case QImage::Format_ARGB32_Premultiplied: { + case QImage::Format_ARGB32: // 8-bit, 4 channel + case QImage::Format_ARGB32_Premultiplied: { Mat temp(source.height(), source.width(), CV_8UC4, - const_cast(source.bits()), - static_cast(source.bytesPerLine())); + const_cast(source.bits()), + static_cast(source.bytesPerLine())); return temp; - } + } - case QImage::Format_RGB32: { // 8-bit, 3 channel + case QImage::Format_RGB32: { // 8-bit, 3 channel Mat temp(source.height(), source.width(), CV_8UC4, - const_cast(source.bits()), - static_cast(source.bytesPerLine())); + const_cast(source.bits()), + static_cast(source.bytesPerLine())); Mat tempNoAlpha; cvtColor(temp, tempNoAlpha, COLOR_BGRA2BGR); // drop the all-white alpha channel return tempNoAlpha; - } + } - case QImage::Format_RGB888: { // 8-bit, 3 channel + case QImage::Format_RGB888: { // 8-bit, 3 channel QImage swapped = source.rgbSwapped(); return Mat(swapped.height(), swapped.width(), CV_8UC3, - const_cast(swapped.bits()), - static_cast(swapped.bytesPerLine())).clone(); - } + const_cast(swapped.bits()), + static_cast(swapped.bytesPerLine())).clone(); + } - case QImage::Format_Indexed8: { // 8-bit, 1 channel + case QImage::Format_Indexed8: { // 8-bit, 1 channel Mat temp(source.height(), source.width(), CV_8UC1, - const_cast(source.bits()), - static_cast(source.bytesPerLine())); + const_cast(source.bits()), + static_cast(source.bytesPerLine())); return temp; - } + } - default: - break; + default: + break; } - return Mat(); // return empty Mat if type not found + return {}; // return empty Mat if type not found } Mat QPixmap2Mat(const QPixmap &source) // Convert Pixmap to Mat @@ -88,12 +90,12 @@ Mat QPixmap2Mat(const QPixmap &source) // Convert Pixmap to Mat QImage Mat2QImage(const Mat &source) // convert BGR Mat to RGB QImage { - Mat temp; - cvtColor(source, temp, COLOR_BGR2RGB); // convert Mat BGR to QImage RGB - QImage dest((const uchar *) temp.data, temp.cols, temp.rows, temp.step, QImage::Format_RGB888); // conversion - dest.bits(); // enforce deep copy of QImage::QImage (const uchar * data, int width, int height, Format format) + Mat temp; + cvtColor(source, temp, COLOR_BGR2RGB); // convert Mat BGR to QImage RGB + QImage dest((const uchar *)temp.data, temp.cols, temp.rows, temp.step, QImage::Format_RGB888); // conversion + dest.bits(); // enforce deep copy of QImage::QImage (const uchar * data, int width, int height, Format format) - return dest; + return dest; } QPixmap Mat2QPixmap(const Mat &source) // convert Mat to QPixmap @@ -107,8 +109,12 @@ QPixmap Mat2QPixmap(const Mat &source) // convert Mat to QPixmap QPixmap Mat2QPixmapResized(const Mat &source, const int &width, const int &height, const bool &smooth) // convert Mat to resized QPixmap { Qt::TransformationMode quality; // quality - if (smooth) quality = Qt::SmoothTransformation; - else quality = Qt::FastTransformation; + if (smooth) { + quality = Qt::SmoothTransformation; + } + else { + quality = Qt::FastTransformation; + } QImage i = Mat2QImage(source); // first step: convert to QImage QPixmap p = QPixmap::fromImage(i, Qt::AutoColor); // then convert QImage to QPixmap @@ -120,31 +126,32 @@ QImage cvMatToQImage(const Mat &source) // another implementation BGR to RGB, ev { switch (source.type()) { // which type ? - case CV_8UC4: { // 8-bit, 4 channel - QImage image( source.data, - source.cols, source.rows, - static_cast(source.step), - QImage::Format_ARGB32); - return image; - } + case CV_8UC4: { // 8-bit, 4 channel + QImage image(source.data, + source.cols, source.rows, + static_cast(source.step), + QImage::Format_ARGB32); + return image; + } - case CV_8UC3: { // 8-bit, 3 channel - QImage image( source.data, - source.cols, source.rows, - static_cast(source.step), - QImage::Format_RGB888); - return image.rgbSwapped(); - } + case CV_8UC3: { // 8-bit, 3 channel + QImage image(source.data, + source.cols, source.rows, + static_cast(source.step), + QImage::Format_RGB888); + return image.rgbSwapped(); + } - case CV_8UC1: { // 8-bit, 1 channel - QImage image(source.data, source.cols, source.rows, - static_cast(source.step), - QImage::Format_Grayscale8); - } + case CV_8UC1: { // 8-bit, 1 channel + QImage image(source.data, source.cols, source.rows, + static_cast(source.step), + QImage::Format_Grayscale8); + return image; + } } - return QImage(); // return empty Mat if type not found + return {}; // return empty Mat if type not found } /////////////////////////////////////////////////////////// @@ -156,10 +163,13 @@ Mat BrightnessContrast(const Mat &source, const double &alpha, const int &beta) Mat image = Mat::zeros(source.size(), source.type()); // Do the operation new_image(i,j) = alpha*image(i,j) + beta - for (int y = 0; y < source.rows; y++) // scan entire Mat - for (int x = 0; x < source.cols; x++) - for (int c = 0; c < 3; c++) - image.at(y, x)[c] = saturate_cast(alpha*(source.at(y, x)[c]) + beta); // change individual values + for (int y = 0; y < source.rows; y++) { // scan entire Mat + for (int x = 0; x < source.cols; x++) { + for (int c = 0; c < 3; c++) { + image.at(y, x)[c] = saturate_cast(alpha*(source.at(y, x)[c]) + beta); // change individual values + } + } + } return image; } @@ -169,8 +179,9 @@ Mat GammaCorrection(const Mat &source, const double gamma) Mat lookUpTable(1, 256, CV_8U); uchar* p = lookUpTable.ptr(); - for( int i = 0; i < 256; ++i) + for (int i = 0; i < 256; ++i) { p[i] = saturate_cast(pow(i / 255.0, gamma) * 255.0); + } Mat res = source.clone(); LUT(source, lookUpTable, res); @@ -202,22 +213,22 @@ Mat EqualizeHistogram(const Mat &source) // histogram equalization Mat SimplestColorBalance(const Mat &source, const float &percent) // color balance with histograms { Mat dest; - float half_percent = percent / 200.0f; + float half_percent = percent / 200.0F; vector tmpsplit; split(source, tmpsplit); // split channels - for (int i=0; i<3; i++) { // just the 3 first channels + for (int i = 0; i < 3; i++) { // just the 3 first channels // find low and high precentile values (based on input percentile) Mat flat; - tmpsplit[i].reshape(1,1).copyTo(flat); - cv::sort(flat,flat,SORT_EVERY_ROW + SORT_ASCENDING); + tmpsplit[i].reshape(1, 1).copyTo(flat); + cv::sort(flat, flat, SORT_EVERY_ROW + SORT_ASCENDING); int lowval = flat.at(cvFloor(((float)flat.cols) * half_percent)); int highval = flat.at(cvCeil(((float)flat.cols) * (1.0 - half_percent))); // saturate below the low percentile and above the high percentile - tmpsplit[i].setTo(lowval,tmpsplit[i] < lowval); - tmpsplit[i].setTo(highval,tmpsplit[i] > highval); + tmpsplit[i].setTo(lowval, tmpsplit[i] < lowval); + tmpsplit[i].setTo(highval, tmpsplit[i] > highval); // scale channel normalize(tmpsplit[i], tmpsplit[i], 0, 255, NORM_MINMAX); @@ -234,13 +245,14 @@ Mat SimplestColorBalance(const Mat &source, const float &percent) // color balan Mat DilatePixels(const Mat &source, const int &dilation_size) // dilate pixels { - if (dilation_size <= 0) + if (dilation_size <= 0) { return source; + } // Create structuring element Mat element = getStructuringElement(MORPH_RECT, // MORPH_CROSS MORPH_RECT & MORPH_ELLIPSE possible - Size(2 * dilation_size + 1, 2 * dilation_size + 1), - Point(-1,-1)); + Size(2 * dilation_size + 1, 2 * dilation_size + 1), + Point(-1, -1)); Mat dest; dilate(source, dest, element); // Apply dilation on the image @@ -249,13 +261,14 @@ Mat DilatePixels(const Mat &source, const int &dilation_size) // dilate pixels Mat ErodePixels(const Mat &source, const int &erosion_size) // erode pixels { - if (erosion_size <= 0) + if (erosion_size <= 0) { return source; + } // Create structuring element - Mat element = getStructuringElement( MORPH_RECT, // MORPH_CROSS MORPH_RECT & MORPH_ELLIPSE possible - Size(2 * erosion_size + 1, 2 * erosion_size + 1), - Point(-1,-1)); + Mat element = getStructuringElement(MORPH_RECT, // MORPH_CROSS MORPH_RECT & MORPH_ELLIPSE possible + Size(2 * erosion_size + 1, 2 * erosion_size + 1), + Point(-1, -1)); Mat dest; erode(source, dest, element); // Apply erosion on the image @@ -271,18 +284,18 @@ Mat ShiftFrame(const Mat &source, const int &nb_pixels, const shift_direction &d Mat dest = Mat::zeros(source.size(), source.type()); switch (direction) { // just make a copy with Rect - case(shift_up) : - source(Rect(0, nb_pixels, source.cols, source.rows - nb_pixels)).copyTo(dest(Rect(0, 0, dest.cols, dest.rows - nb_pixels))); - break; - case(shift_right) : - source(Rect(0, 0, source.cols - nb_pixels, source.rows)).copyTo(dest(Rect(nb_pixels, 0, source.cols - nb_pixels, source.rows))); - break; - case(shift_down) : - source(Rect(0, 0, source.cols, source.rows - nb_pixels)).copyTo(dest(Rect(0, nb_pixels, source.cols, source.rows - nb_pixels))); - break; - case(shift_left) : - source(Rect(nb_pixels, 0, source.cols - nb_pixels, source.rows)).copyTo(dest(Rect(0, 0, source.cols - nb_pixels, source.rows))); - break; + case(shift_up): + source(Rect(0, nb_pixels, source.cols, source.rows - nb_pixels)).copyTo(dest(Rect(0, 0, dest.cols, dest.rows - nb_pixels))); + break; + case(shift_right): + source(Rect(0, 0, source.cols - nb_pixels, source.rows)).copyTo(dest(Rect(nb_pixels, 0, source.cols - nb_pixels, source.rows))); + break; + case(shift_down): + source(Rect(0, 0, source.cols, source.rows - nb_pixels)).copyTo(dest(Rect(0, nb_pixels, source.cols, source.rows - nb_pixels))); + break; + case(shift_left): + source(Rect(nb_pixels, 0, source.cols - nb_pixels, source.rows)).copyTo(dest(Rect(0, 0, source.cols - nb_pixels, source.rows))); + break; } return dest; @@ -292,10 +305,11 @@ Mat ShiftFrame(const Mat &source, const int &nb_pixels, const shift_direction &d //// Clipping & Resize /////////////////////////////////////////////////////////// -Mat CopyFromImage (Mat source, const Rect &frame) // copy part of an image +Mat CopyFromImage(Mat source, const Rect &frame) // copy part of an image { - if (source.channels() == 1) + if (source.channels() == 1) { cvtColor(source, source, COLOR_GRAY2BGR); + } return source(frame); // just use the Rect area } @@ -304,8 +318,12 @@ Mat ResizeImageAspectRatio(const Mat &source, const Size &frame) // Resize image double zoomX = double(frame.width) / source.cols; // try vertical and horizontal ratios double zoomY = double(frame.height) / source.rows; double zoom; - if (zoomX < zoomY) zoom = zoomX; // the lowest fit the view - else zoom = zoomY; + if (zoomX < zoomY) { + zoom = zoomX; // the lowest fit the view + } + else { + zoom = zoomY; + } Mat dest; resize(source, dest, Size(int(source.cols*zoom), int(source.rows*zoom)), 0, 0, INTER_AREA); // resize @@ -319,7 +337,7 @@ Mat ResizeImageAspectRatio(const Mat &source, const Size &frame) // Resize image double MedianMat(Mat source) // median, used by DrawColoredContours { - source = source.reshape(0,1);// spread source to single row + source = source.reshape(0, 1);// spread source to single row std::vector vecFromMat; source.copyTo(vecFromMat); // copy source to vector vecFromMat @@ -331,23 +349,25 @@ double MedianMat(Mat source) // median, used by DrawColoredContours Mat DrawColoredContours(const Mat &source, const double &sigma, const int &aperture_size, const int &thickness) // Canny edge detection { RNG rng(123); // controled randomization - Mat canny_output, gray, blur; + Mat canny_output; + Mat gray; + Mat blur; - GaussianBlur(source, blur, Size(3,3), 0, 0); // better with a gaussian blur first - cvtColor(blur, gray, COLOR_BGR2GRAY ); // convert to gray + GaussianBlur(source, blur, Size(3, 3), 0, 0); // better with a gaussian blur first + cvtColor(blur, gray, COLOR_BGR2GRAY); // convert to gray double median = MedianMat(gray); // get median value of matrix - int lower = (int)std::max(0.0, (1.0-sigma) * median); // generate thresholds - int upper = (int)std::min(255.0, (1.0+sigma) * median); + int lower = (int)std::max(0.0, (1.0 - sigma) * median); // generate thresholds + int upper = (int)std::min(255.0, (1.0 + sigma) * median); vector> contours; // create structure to compute vector hierarchy; - Canny(gray, canny_output, lower, upper, aperture_size, true ); // apply Canny + Canny(gray, canny_output, lower, upper, aperture_size, true); // apply Canny findContours(canny_output, contours, hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0)); // find the edges Mat drawing = Mat::zeros(canny_output.size(), CV_8UC3); - for (size_t i = 0; i< contours.size(); i++) { // scan the image - Scalar color = Scalar(rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255)); // apply colors + for (size_t i = 0; i < contours.size(); i++) { // scan the image + Scalar color = Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255)); // apply colors drawContours(drawing, contours, (int)i, color, thickness, 8, hierarchy, 0, Point()); // draw the lines } @@ -369,15 +389,15 @@ double PSNR(const Mat &source1, const Mat &source2) // noise difference between double sse = s.val[0] + s.val[1] + s.val[2]; // sum channels - if( sse <= 1e-10) // for small values return zero + if (sse <= 1e-10) { // for small values return zero return 0; - else // compute PSNR - { - double mse =sse /(double)(source1.channels() * source1.total()); - double psnr = 10.0*log10((255*255)/mse); + } // compute PSNR + + double mse = sse / (double)(source1.channels() * source1.total()); + double psnr = 10.0*log10((255 * 255) / mse); + + return psnr; - return psnr; - } } /////////////////////////////////////////////////////////// @@ -388,135 +408,165 @@ double GrayCurve(const int &color, const int &type, const int &begin, const int { // all functions must have continuous results in [0..1] range and have this shape : f(x) * range + begin - double x = double(color-begin) / range; // x of f(x) + double x = double(color - begin) / range; // x of f(x) - if (range == 0) return color; // faster this way + if (range == 0) { + return color; // faster this way + } switch (type) { // good spread - case curve_linear: return color; // linear -> the same color ! - // S-shaped - case curve_cosinus2: return pow(cos(Pi / 2 - x * Pi/2), 2) * range + begin; // cosinus² (better color gradient): f(x)=cos(pi/2-x*pi/2)² - case curve_sigmoid: return 1.0 / (1 + exp(-5 * (2* (x) - 1))) * range + begin; // sigmoid (S-shaped): f(x)=1/(1 + e(-5*(2x -1)) - // fast beginning - case curve_cosinus: return cos(Pi / 2 - x * Pi/2) * range + begin; // cosinus (more of end color): f(x)=cos(pi/2-x*pi/2) - case curve_cos2sqrt: return pow(cos(Pi/2 - sqrt(x) * Pi/2), 2) * range + begin; // cos²sqrt: f(x)=cos(pi/2−sqrt(x)*pi/2)² - // fast ending - case curve_power2: return pow(x, 2) * range + begin; // power2 (more of begin color): f(x)=x² - case curve_cos2power2: return pow(cos(Pi/2 - pow(x, 2) * Pi/2), 2) * range + begin; // cos²power2: f(x)=cos(pi/2−x²∙pi/2)² - case curve_power3: return pow(x, 3) * range + begin; // power3 (even more of begin color): f(x)=x³ - // undulate - case curve_undulate: return cos(double(color-begin) / 4 * Pi) * range + begin; // undulate: f(x)=cos(x/4*pi) - case curve_undulate2: return cos(pow(double(color-begin) * 2 * Pi/2 + 0.5, 2)) * range + begin; // undulate²: f(x)=cos((x*2*pi)²) / 2 + 0.5 - case curve_undulate3: return (cos(Pi*Pi*pow(x+2.085,2))/(pow(x+2.085,3)+8)+(x+2.085)-2.11) * range + begin; // undulate3: f(x) = cos(pi²∙(x+2.085)²) / ((x+2.085)³+10) + (x+2.085) − 2.11 - - //case 5: return sqrt(double(color-begin) / range) * range + begin; + case curve_linear: return color; // linear -> the same color ! + // S-shaped + case curve_cosinus2: return pow(cos(Pi / 2 - x * Pi / 2), 2) * range + begin; // cosinus² (better color gradient): f(x)=cos(pi/2-x*pi/2)² + case curve_sigmoid: return 1.0 / (1 + exp(-5 * (2 * (x)-1))) * range + begin; // sigmoid (S-shaped): f(x)=1/(1 + e(-5*(2x -1)) + // fast beginning + case curve_cosinus: return cos(Pi / 2 - x * Pi / 2) * range + begin; // cosinus (more of end color): f(x)=cos(pi/2-x*pi/2) + case curve_cos2sqrt: return pow(cos(Pi / 2 - sqrt(x) * Pi / 2), 2) * range + begin; // cos²sqrt: f(x)=cos(pi/2−sqrt(x)*pi/2)² + // fast ending + case curve_power2: return pow(x, 2) * range + begin; // power2 (more of begin color): f(x)=x² + case curve_cos2power2: return pow(cos(Pi / 2 - pow(x, 2) * Pi / 2), 2) * range + begin; // cos²power2: f(x)=cos(pi/2−x²∙pi/2)² + case curve_power3: return pow(x, 3) * range + begin; // power3 (even more of begin color): f(x)=x³ + // undulate + case curve_undulate: return cos(double(color - begin) / 4 * Pi) * range + begin; // undulate: f(x)=cos(x/4*pi) + case curve_undulate2: return cos(pow(double(color - begin) * 2 * Pi / 2 + 0.5, 2)) * range + begin; // undulate²: f(x)=cos((x*2*pi)²) / 2 + 0.5 + case curve_undulate3: return (cos(Pi*Pi*pow(x + 2.085, 2)) / (pow(x + 2.085, 3) + 8) + (x + 2.085) - 2.11) * range + begin; // undulate3: f(x) = cos(pi²∙(x+2.085)²) / ((x+2.085)³+10) + (x+2.085) − 2.11 + + //case 5: return sqrt(double(color-begin) / range) * range + begin; } return color; } -float EuclideanDistance(Point center, Point point, int radius){ // return distance between 2 points +float EuclideanDistance(const Point& center, const Point& point, int radius) { // return distance between 2 points float distance = sqrt(std::pow(center.x - point.x, 2) + std::pow(center.y - point.y, 2)); - if (distance > radius) return radius; // no value beyond radius - else return distance; + if (distance > radius) { + return radius; // no value beyond radius + } return distance; } void GradientFillGray(const int &gradient_type, Mat &img, const Mat &msk, const Point &beginPoint, - const Point &endPoint, const int &beginColor, const int &endColor, - const int &curve, Rect area) // fill a 1-channel image with the mask converted to gray gradients - // img must be 1-channel and msk 1-channel - // img is input-output - // area = rectangle containing the mask (speeds up the image scan) - no area given = entire image scanned + const Point &endPoint, const int &beginColor, const int &endColor, + const int &curve, Rect area) // fill a 1-channel image with the mask converted to gray gradients +// img must be 1-channel and msk 1-channel +// img is input-output +// area = rectangle containing the mask (speeds up the image scan) - no area given = entire image scanned { - if (area == Rect(0, 0, 0, 0)) // default area = 0 - area = Rect(0,0, img.cols, img.rows); // set it to image dimensions + if (area == Rect(0, 0, 0, 0)) { // default area = 0 + area = Rect(0, 0, img.cols, img.rows); // set it to image dimensions + } switch (gradient_type) { - case (gradient_flat): { // flat = same color everywhere - img.setTo(beginColor, msk); // fill the mask with this color - return; - } - case (gradient_linear): { // grayscale is spread along the line - int A = (endPoint.x - beginPoint.x); // horizontal difference - int B = (endPoint.y - beginPoint.y); // vertical difference - int C1 = A * beginPoint.x + B * beginPoint.y; // sort of euclidian distance, but no need to compute sqrt values - int C2 = A * endPoint.x + B * endPoint.y; - - float CO; // will contain color values - - for (int row = area.y; row < area.y + area.height; row++) // scan mask - for (int col = area.x; col < area.x + area.width; col++) - if (msk.at(row, col) != 0) { // non-zero pixel in mask - int C = A * col + B * row; // "distance" for this pixel - - if (C <= C1) CO = beginColor; // before begin point : begin color - else if (C >= C2) CO = endColor; // after end point : end color - else CO = round(GrayCurve(float(beginColor * (C2 - C) + endColor * (C - C1))/(C2 - C1), - curve, beginColor, endColor - beginColor)); // C0 = percentage between begin and end colors, "shaped" by gray curve - img.at(row, col) = CO; // set grayscale to image + case (gradient_flat): { // flat = same color everywhere + img.setTo(beginColor, msk); // fill the mask with this color + return; + } + case (gradient_linear): { // grayscale is spread along the line + int A = (endPoint.x - beginPoint.x); // horizontal difference + int B = (endPoint.y - beginPoint.y); // vertical difference + int C1 = A * beginPoint.x + B * beginPoint.y; // sort of euclidian distance, but no need to compute sqrt values + int C2 = A * endPoint.x + B * endPoint.y; + + float CO; // will contain color values + + for (int row = area.y; row < area.y + area.height; row++) { // scan mask + for (int col = area.x; col < area.x + area.width; col++) { + if (msk.at(row, col) != 0) { // non-zero pixel in mask + int C = A * col + B * row; // "distance" for this pixel + + if (C <= C1) { + CO = beginColor; // before begin point : begin color + } + else if (C >= C2) { + CO = endColor; // after end point : end color + } + else { + CO = round(GrayCurve(float(beginColor * (C2 - C) + endColor * (C - C1)) / (C2 - C1), + curve, beginColor, endColor - beginColor)); // C0 = percentage between begin and end colors, "shaped" by gray curve } - return; // done ! -> exit + img.at(row, col) = CO; // set grayscale to image + } + } } - case (gradient_doubleLinear): { // double linear = 2 times linear, just invert the vector the second time - int A = (endPoint.x - beginPoint.x); // same comments as linear - int B = (endPoint.y - beginPoint.y); - int C1 = A * beginPoint.x + B * beginPoint.y; - int C2 = A * endPoint.x + B * endPoint.y; - - float CO; - - for (int row = area.y; row < area.y + area.height; row++) - for (int col = area.x; col < area.x + area.width; col++) - if (msk.at(row, col) != 0) { - int C = A * col + B * row; - if (((C > C1) & (C < C2)) | (C >= C2) | (C == C1)) { // the only difference is we don't fill "before" the begin point - if (C == C1) CO = beginColor; - else if (C >= C2) CO = endColor; - else CO = round(GrayCurve(float(beginColor * (C2 - C) + endColor * (C - C1))/(C2 - C1), - curve, beginColor, endColor - beginColor)); - img.at(row, col) = CO; + return; // done ! -> exit + } + case (gradient_doubleLinear): { // double linear = 2 times linear, just invert the vector the second time + int A = (endPoint.x - beginPoint.x); // same comments as linear + int B = (endPoint.y - beginPoint.y); + int C1 = A * beginPoint.x + B * beginPoint.y; + int C2 = A * endPoint.x + B * endPoint.y; + + float CO; + + for (int row = area.y; row < area.y + area.height; row++) { + for (int col = area.x; col < area.x + area.width; col++) { + if (msk.at(row, col) != 0) { + int C = A * col + B * row; + if (((C > C1) & (C < C2)) | (C >= C2) | (C == C1)) { // the only difference is we don't fill "before" the begin point + if (C == C1) { + CO = beginColor; + } + else if (C >= C2) { + CO = endColor; } + else { + CO = round(GrayCurve(float(beginColor * (C2 - C) + endColor * (C - C1)) / (C2 - C1), + curve, beginColor, endColor - beginColor)); + } + img.at(row, col) = CO; } + } + } + } - Point newEndPoint; // invert the vector - newEndPoint.x = 2* beginPoint.x - endPoint.x; - newEndPoint.y = 2* beginPoint.y - endPoint.y; - - A = (newEndPoint.x - beginPoint.x); // same as before, but with new inverted vector - B = (newEndPoint.y - beginPoint.y); - C1 = A * beginPoint.x + B * beginPoint.y; - C2 = A * newEndPoint.x + B * newEndPoint.y; - - for (int row = area.y; row < area.y + area.height; row++) - for (int col = area.x; col < area.x + area.width; col++) - if (msk.at(row, col) != 0) { - int C = A * col + B * row; - if (((C > C1) & (C < C2)) | (C >= C2) | (C == C1)) { // once again don't fill "before" begin point - if (C == C1) CO = beginColor; - else if (C >= C2) CO = endColor; - else CO = round(GrayCurve(float(beginColor * (C2 - C) + endColor * (C - C1))/(C2 - C1), - curve, beginColor, endColor - beginColor)); - img.at(row, col) = CO; + Point newEndPoint; // invert the vector + newEndPoint.x = 2 * beginPoint.x - endPoint.x; + newEndPoint.y = 2 * beginPoint.y - endPoint.y; + + A = (newEndPoint.x - beginPoint.x); // same as before, but with new inverted vector + B = (newEndPoint.y - beginPoint.y); + C1 = A * beginPoint.x + B * beginPoint.y; + C2 = A * newEndPoint.x + B * newEndPoint.y; + + for (int row = area.y; row < area.y + area.height; row++) { + for (int col = area.x; col < area.x + area.width; col++) { + if (msk.at(row, col) != 0) { + int C = A * col + B * row; + if (((C > C1) & (C < C2)) | (C >= C2) | (C == C1)) { // once again don't fill "before" begin point + if (C == C1) { + CO = beginColor; + } + else if (C >= C2) { + CO = endColor; + } + else { + CO = round(GrayCurve(float(beginColor * (C2 - C) + endColor * (C - C1)) / (C2 - C1), + curve, beginColor, endColor - beginColor)); } + img.at(row, col) = CO; } - return; + } + } } - case (gradient_radial): { // radial = concentric circles - float radius = std::sqrt(std::pow(beginPoint.x - endPoint.x, 2) + std::pow(beginPoint.y - endPoint.y, 2)); // maximum euclidian distance = vector length + return; + } + case (gradient_radial): { // radial = concentric circles + float radius = std::sqrt(std::pow(beginPoint.x - endPoint.x, 2) + std::pow(beginPoint.y - endPoint.y, 2)); // maximum euclidian distance = vector length - int CO; // will contain color values + int CO; // will contain color values - for (int row = area.y; row < area.y + area.height; row++) // scan entire mask - for (int col = area.x; col < area.x + area.width; col++) - if (msk.at(row, col) != 0) { // non-zero pixel in mask - CO = round(GrayCurve(beginColor + EuclideanDistance(beginPoint, Point(col, row), radius) / radius * (endColor - beginColor), - curve, beginColor, endColor - beginColor)); // pixel in temp gradient mask = distance percentage, "shaped" by gray curve - img.at(row, col) = CO; - } - return; + for (int row = area.y; row < area.y + area.height; row++) { // scan entire mask + for (int col = area.x; col < area.x + area.width; col++) { + if (msk.at(row, col) != 0) { // non-zero pixel in mask + CO = round(GrayCurve(beginColor + EuclideanDistance(beginPoint, Point(col, row), radius) / radius * (endColor - beginColor), + curve, beginColor, endColor - beginColor)); // pixel in temp gradient mask = distance percentage, "shaped" by gray curve + img.at(row, col) = CO; + } + } } + return; + } } } @@ -528,60 +578,61 @@ Mat AnaglyphTint(const Mat & source, const int &tint) // change tint of image to Mat splt[3]; //destination array split(source, splt); // split source to 3 channels (RGB) - for (int row = 0; row < source.rows; row++) // scan entire image + for (int row = 0; row < source.rows; row++) { // scan entire image for (int col = 0; col < source.cols; col++) { - int R, G, B; + int R; + int G; + int B; Vec3b color = source.at(row, col); // get pixel color switch (tint) { // apply tint formula - case tint_color: - default: { // original colors - R = round( 1.000 * color[2] + 0.000 * color[1] + 0.000 * color[0]); - G = round( 0.000 * color[2] + 1.000 * color[1] + 0.000 * color[0]); - B = round( 0.000 * color[2] + 0.000 * color[1] + 1.000 * color[0]); - break; - } - case tint_gray: { // grayscale - R = round( 0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); - G = round( 0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); - B = round( 0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); - break; - } - case tint_true: { // only red and cyan values - R = round( 0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); - G = round( 0.000 * color[2] + 0.000 * color[1] + 0.000 * color[0]); - B = round( 0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); - break; - } - case tint_half: { // half-colors - R = round( 0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); - G = round( 0.000 * color[2] + 1.000 * color[1] + 0.000 * color[0]); - B = round( 0.000 * color[2] + 0.000 * color[1] + 1.000 * color[0]); - break; - } - case tint_optimized: { // optimized colors - R = round( 0.000 * color[2] + 0.700 * color[1] + 0.300 * color[0]); - G = round( 0.000 * color[2] + 1.000 * color[1] + 0.000 * color[0]); - B = round( 0.000 * color[2] + 0.000 * color[1] + 1.000 * color[0]); - break; - } - case tint_dubois: { // Dubois algorithm - R = round( 0.4045 * color[2] + 0.4346 * color[1] + 0.1609 * color[0]); - G = round( 0.3298 * color[2] + 0.6849 * color[1] - 0.0146 * color[0]); - B = round(-0.1162 * color[2] - 0.1902 * color[1] + 1.3099 * color[0]); - break; - } + case tint_color: + default: { // original colors + R = round(1.000 * color[2] + 0.000 * color[1] + 0.000 * color[0]); + G = round(0.000 * color[2] + 1.000 * color[1] + 0.000 * color[0]); + B = round(0.000 * color[2] + 0.000 * color[1] + 1.000 * color[0]); + break; + } + case tint_gray: { // grayscale + R = round(0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); + G = round(0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); + B = round(0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); + break; + } + case tint_true: { // only red and cyan values + R = round(0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); + G = round(0.000 * color[2] + 0.000 * color[1] + 0.000 * color[0]); + B = round(0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); + break; + } + case tint_half: { // half-colors + R = round(0.299 * color[2] + 0.587 * color[1] + 0.114 * color[0]); + G = round(0.000 * color[2] + 1.000 * color[1] + 0.000 * color[0]); + B = round(0.000 * color[2] + 0.000 * color[1] + 1.000 * color[0]); + break; + } + case tint_optimized: { // optimized colors + R = round(0.000 * color[2] + 0.700 * color[1] + 0.300 * color[0]); + G = round(0.000 * color[2] + 1.000 * color[1] + 0.000 * color[0]); + B = round(0.000 * color[2] + 0.000 * color[1] + 1.000 * color[0]); + break; + } + case tint_dubois: { // Dubois algorithm + R = round(0.4045 * color[2] + 0.4346 * color[1] + 0.1609 * color[0]); + G = round(0.3298 * color[2] + 0.6849 * color[1] - 0.0146 * color[0]); + B = round(-0.1162 * color[2] - 0.1902 * color[1] + 1.3099 * color[0]); + break; + } } - if (R < 0) R = 0; // check if values are out of range - if (G < 0) G = 0; - if (B < 0) B = 0; - if (R > 255) R = 255; - if (G > 255) G = 255; - if (B > 255) B = 255; + // check if values are out of range + R = std::clamp(R, 0, 255); + G = std::clamp(G, 0, 255); + B = std::clamp(B, 0, 255); dest.at(row, col) = Vec3b(B, G, R); // new pixel RGB values } + } return dest; } diff --git a/openglwidget.cpp b/openglwidget.cpp index dd6c989..3b958ff 100644 --- a/openglwidget.cpp +++ b/openglwidget.cpp @@ -40,23 +40,38 @@ using namespace cv; +namespace { + + int NumberOfIndexes(const int &rows, const int &cols) // number of expected indexes for an image + { + if (rows <= 0) { // special case + return 0; + } // a bit complicated, see ComputeIndexes() + return double(double(cols) * double(rows)) + double((double(cols) * 2 - 1) * (double(rows) / 2 - 1)); + } + + int NumberOfVertices(const int &rows, const int &cols) // number of expected vertices for an image + { + return rows * cols; + } + +} // namespace + /////////////////////////////////////////////// //// Widget /////////////////////////////////////////////// openGLWidget::openGLWidget(QWidget *parent) : QOpenGLWidget(parent), - vertexbuffer(QOpenGLBuffer::VertexBuffer), - indexbuffer(QOpenGLBuffer::IndexBuffer), - colorbuffer(QOpenGLBuffer::VertexBuffer) + vertexbuffer(QOpenGLBuffer::VertexBuffer), + indexbuffer(QOpenGLBuffer::IndexBuffer), + colorbuffer(QOpenGLBuffer::VertexBuffer) { } openGLWidget::~openGLWidget() -{ - -} += default; /////////////////////////////////////////////// //// Redefine widget main functions : @@ -95,29 +110,16 @@ void openGLWidget::initializeGL() // launched when the widget is initialized //// Lights GLfloat light_position[] = { 5000, 0, 5000, 1.0 }; - GLfloat light_ambient[] = { 0.1, 0.1, 0.1, 1.0}; - GLfloat light_diffuse[] = { 3, 3, 3, 0.5}; + GLfloat light_ambient[] = { 0.1, 0.1, 0.1, 1.0 }; + GLfloat light_diffuse[] = { 3, 3, 3, 0.5 }; glEnable(GL_LIGHTING); // enable lighting glEnable(GL_LIGHT0); // define the first light glLightfv(GL_LIGHT0, GL_POSITION, light_position); - glLightfv(GL_LIGHT0, GL_AMBIENT, light_ambient); // ambient - glLightfv(GL_LIGHT0, GL_DIFFUSE, light_diffuse); // diffuse -} - -int openGLWidget::NumberOfIndexes(const int &rows, const int &cols) // number of expected indexes for an image -{ - if (rows <= 0) // special case - return 0; - else // a bit complicated, see ComputeIndexes() - return double(double(cols) * double(rows)) + double((double(cols) * 2 - 1) * (double(rows) / 2 - 1)); + glLightfv(GL_LIGHT0, GL_AMBIENT, light_ambient); // ambient + glLightfv(GL_LIGHT0, GL_DIFFUSE, light_diffuse); // diffuse } -int openGLWidget::NumberOfVertices(const int &rows, const int &cols) // number of expected vertices for an image -{ - return rows * cols; -} - -GLuint openGLWidget::VertexIndex(const int &y, const int &x) // pixel's index in a image +GLuint openGLWidget::VertexIndex(const int &y, const int &x) const // pixel's index in a image { return y * image3D.cols + x; // just like an array } @@ -131,17 +133,17 @@ void openGLWidget::ComputeVertices() // (re)create vertices array and buffer int halfX = -depthmap3D.cols / 2; // to center the axes in middle of image int halfY = depthmap3D.rows / 2; - for (float row = 0; row < depthmap3D.rows; row++) { // for each row of the image - for (float col = 0; col < depthmap3D.cols; col++) { // for each pixel in the row from left to right - vertexarray.push_back(QVector3D(col+halfX, -row+halfY, - (depthmap3D.at(row, col) - 127) * depth3D)); // vertex in buffer + for (int row = 0; row < depthmap3D.rows; row++) { // for each row of the image + for (int col = 0; col < depthmap3D.cols; col++) { // for each pixel in the row from left to right + vertexarray.push_back(QVector3D(col + halfX, -row + halfY, + (depthmap3D.at(row, col) - 127) * depth3D)); // vertex in buffer } } vertexbuffer.create(); // create VBO vertices buffer vertexbuffer.bind(); // bind it vertexbuffer.setUsagePattern(QOpenGLBuffer::DynamicDraw); // vertex buffer will be often modified - vertexbuffer.allocate(vertexarray.constData(), vertexarray.size()*sizeof(QVector3D)); // allocate and populate in GPU RAM + vertexbuffer.allocate(vertexarray.constData(), vertexarray.size() * sizeof(QVector3D)); // allocate and populate in GPU RAM vertexbuffer.release(); // done computeVertices3D = false; // done recomputing @@ -155,9 +157,9 @@ void openGLWidget::UpdateVertices() // update vertices z } vertexbuffer.bind(); // use current VBO - GLfloat* posBuffer = (GLfloat*) (vertexbuffer.map(QOpenGLBuffer::WriteOnly)); // map a pointer on it + auto* posBuffer = (GLfloat*)(vertexbuffer.map(QOpenGLBuffer::WriteOnly)); // map a pointer on it - if (posBuffer != (GLfloat*) NULL) { // pointer not valid ? + if (posBuffer != (GLfloat*) nullptr) { // pointer not valid ? int index; // index of current vertex @@ -190,27 +192,27 @@ void openGLWidget::ComputeIndexes() // (re)create index array and buffer int arraySize = NumberOfIndexes(image3D.rows, image3D.cols); // how many indexes ? //indexarray.reserve(arraySize); // reserve memory space in advance - for (int row = 0; row < depthmap3D.rows -1; row++) { // for each row of the images - if (int(row)%2 == 0) { // row number is even + for (int row = 0; row < depthmap3D.rows - 1; row++) { // for each row of the images + if (int(row) % 2 == 0) { // row number is even for (int col = 0; col < depthmap3D.cols; col++) { // for each pixel in the row from left to right indexarray.push_back(VertexIndex(row, col)); // index in buffer - indexarray.push_back(VertexIndex(row+1, col)); + indexarray.push_back(VertexIndex(row + 1, col)); } } else { // row number is odd - for (int col = depthmap3D.cols - 1; col > 0 ; col--) { // for each pixel in the row from right to left except the first one - indexarray.push_back(VertexIndex(row+1, col)); // the first time it creates a "degenerate triangle" - indexarray.push_back(VertexIndex(row, col-1)); + for (int col = depthmap3D.cols - 1; col > 0; col--) { // for each pixel in the row from right to left except the first one + indexarray.push_back(VertexIndex(row + 1, col)); // the first time it creates a "degenerate triangle" + indexarray.push_back(VertexIndex(row, col - 1)); } int col = 1; - indexarray.push_back(VertexIndex(row+1, col-1)); // add one more vertex to finish the column + indexarray.push_back(VertexIndex(row + 1, col - 1)); // add one more vertex to finish the column } } indexbuffer.create(); // create VBO vertices buffer indexbuffer.bind(); // bind it indexbuffer.setUsagePattern(QOpenGLBuffer::StaticDraw); // vertex buffer will be often modified - indexbuffer.allocate(indexarray.constData(), indexarray.size()*sizeof(GLuint)); // allocate and populate in GPU RAM + indexbuffer.allocate(indexarray.constData(), indexarray.size() * sizeof(GLuint)); // allocate and populate in GPU RAM indexbuffer.release(); // done computeIndexes3D = false; // done recomputing @@ -223,11 +225,10 @@ void openGLWidget::ComputeColors() // (re)create colors array and buffer colorarray.clear(); // destroy buffers and arrays colorbuffer.destroy(); colorarray.reserve(NumberOfVertices(image3D.rows, image3D.cols)); // reserve memory space in advance - Vec3b CO; // pixel color of reference image - for (float row = 0; row < image3D.rows; row++) { // for each row of the images - for (float col = 0; col < image3D.cols; col++) { // for each pixel in the row from left to right - CO = Vec3b(image3D.at(row, col)); // // pixel color of reference image + for (int row = 0; row < image3D.rows; row++) { // for each row of the images + for (int col = 0; col < image3D.cols; col++) { // for each pixel in the row from left to right + auto CO = Vec3b(image3D.at(row, col)); // // pixel color of reference image colorarray.push_back(QVector3D(CO[2] / 255.0, CO[1] / 255.0, CO[0] / 255.0)); // color in buffer, RGB values between 0 and 1 } } @@ -236,7 +237,7 @@ void openGLWidget::ComputeColors() // (re)create colors array and buffer colorbuffer.create(); // create colors buffer colorbuffer.bind(); // bind it - colorbuffer.allocate(colorarray.constData(), colorarray.size()*sizeof(QVector3D)); // copy data to VBO + colorbuffer.allocate(colorarray.constData(), colorarray.size() * sizeof(QVector3D)); // copy data to VBO colorbuffer.release(); // release VBO } @@ -251,22 +252,23 @@ void openGLWidget::SaveToObj(const QString &filename) // Save current 3D arrays //file.close(); // save vertices - QVector3D vertex, color; + QVector3D vertex; + QVector3D color; int max = NumberOfVertices(image3D.rows, image3D.cols); for (int index = 0; index < max; index++) { vertex = vertexarray[index]; color = colorarray[index]; stream << "v " << vertex.x() << " " << vertex.y() << " " << vertex.z() - << " " << color[0] << " " << color[1] << " " << color[2] - << "\n"; + << " " << color[0] << " " << color[1] << " " << color[2] + << "\n"; } // save indexes max = NumberOfIndexes(image3D.rows, image3D.cols) - 2; for (int index = 0; index < max; index++) { - stream << "f " << indexarray[index]+1 << " " << indexarray[index+1]+1 << " " << indexarray[index+2]+1 << "\n"; + stream << "f " << indexarray[index] + 1 << " " << indexarray[index + 1] + 1 << " " << indexarray[index + 2] + 1 << "\n"; } file.close(); @@ -301,31 +303,32 @@ void openGLWidget::SaveToPly(const QString &filename) // Save current 3D arrays stream << "property uchar green" << "\n"; stream << "property uchar blue" << "\n"; - // triangles + // triangles int maxTriangles = NumberOfIndexes(image3D.rows, image3D.cols) - 2; stream << "element face " << maxTriangles << "\n"; stream << "property list uchar int vertex_index" << "\n"; - // end of header + // end of header stream << "end_header" << "\n"; //// save vertices - QVector3D vertex, color; + QVector3D vertex; + QVector3D color; int index; // index of current vertex std::string st; // used to get a comma separator for floats, streams use locales ! for (int row = 0; row < image3D.rows; row++) { // for each row of area for (int col = 0; col < image3D.cols; col++) { // for each pixel in the row from left to right - index = VertexIndex(row, col); // use index of this pixel - color = colorarray[index]; - /*stream << col << " " << -row << " " << qSetRealNumberPrecision(5) << (depthmap3D.at(row, col) - 127) * depth3D - << " " << int(round(color[0]*255)) << " " << int(round(color[1]*255)) << " " << int(round(color[2]*255)) - << "\n";*/ - st = std::to_string(col) + " " + std::to_string(-row) + " " + std::to_string(float((depthmap3D.at(row, col) - 127) * depth3D)) - + " " + std::to_string(int(round(color[0]*255))) + " " + std::to_string(int(round(color[1]*255))) + " " + std::to_string(int(round(color[2]*255))) - + "\n"; - stream << QString::fromStdString(st); + index = VertexIndex(row, col); // use index of this pixel + color = colorarray[index]; + /*stream << col << " " << -row << " " << qSetRealNumberPrecision(5) << (depthmap3D.at(row, col) - 127) * depth3D + << " " << int(round(color[0]*255)) << " " << int(round(color[1]*255)) << " " << int(round(color[2]*255)) + << "\n";*/ + st = std::to_string(col) + " " + std::to_string(-row) + " " + std::to_string(float((depthmap3D.at(row, col) - 127) * depth3D)) + + " " + std::to_string(int(round(color[0] * 255))) + " " + std::to_string(int(round(color[1] * 255))) + " " + std::to_string(int(round(color[2] * 255))) + + "\n"; + stream << QString::fromStdString(st); } } @@ -333,7 +336,7 @@ void openGLWidget::SaveToPly(const QString &filename) // Save current 3D arrays int maxIndexes = NumberOfIndexes(image3D.rows, image3D.cols) - 2; for (int index = 0; index < maxIndexes; index++) { - stream << "3 " << indexarray[index] << " " << indexarray[index+1] << " " << indexarray[index+2] << "\n"; + stream << "3 " << indexarray[index] << " " << indexarray[index + 1] << " " << indexarray[index + 2] << "\n"; } // Close the file @@ -346,8 +349,10 @@ void openGLWidget::paintGL() // 3D rendering { if (lightEnabled) { glEnable(GL_LIGHTING); // turn on the lights... - } else + } + else { glDisable(GL_LIGHTING); // ... or not + } if (qualityEnabled) { // antialiasing @@ -371,7 +376,9 @@ void openGLWidget::paintGL() // 3D rendering glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); // clear color and depth buffers - if (anaglyphEnabled) glRotatef(-anaglyphShift, 0.0, 1.0, 0.0); // rotate a bit the first image on y axis + if (anaglyphEnabled) { + glRotatef(-anaglyphShift, 0.0, 1.0, 0.0); // rotate a bit the first image on y axis + } glLoadIdentity(); // replace current matrix with identity matrix (reset) @@ -388,60 +395,75 @@ void openGLWidget::paintGL() // 3D rendering if (axesEnabled) { // yes draw origin axes glLineWidth(2); // bigger width of the lines to really see them - if (anaglyphEnabled) glColor3d(1,1,1); - else glColor3d(1,0,0); // x axis color : red + if (anaglyphEnabled) { + glColor3d(1, 1, 1); + } + else { + glColor3d(1, 0, 0); // x axis color : red + } glBegin(GL_LINES); // draw several lines - glVertex3f(0.0f, 0.0f, 0.0f); - glVertex3f(1000.0f, 0.0f, 0.0f); + glVertex3f(0.0F, 0.0F, 0.0F); + glVertex3f(1000.0F, 0.0F, 0.0F); glEnd(); glBegin(GL_TRIANGLES); // triangle at the end of the line = arrow - glVertex3f( 925.0f, -50.0f, 0.0f ); - glVertex3f( 925.0f, 50.0f, 0.0f ); - glVertex3f( 1000.0f, 0.0f, 0.0f ); + glVertex3f(925.0F, -50.0F, 0.0F); + glVertex3f(925.0F, 50.0F, 0.0F); + glVertex3f(1000.0F, 0.0F, 0.0F); glEnd(); - if (anaglyphEnabled) glColor3d(0.85,0.85,0.85); - else glColor3d(0,0,1); // y axis color : blue + if (anaglyphEnabled) { + glColor3d(0.85, 0.85, 0.85); + } + else { + glColor3d(0, 0, 1); // y axis color : blue + } glBegin(GL_LINES); - glVertex3f( 0.0f, 0.0f, 0.0f ); - glVertex3f( 0.0f, -1000.0f, 0.0f ); + glVertex3f(0.0F, 0.0F, 0.0F); + glVertex3f(0.0F, -1000.0F, 0.0F); glEnd(); glBegin(GL_TRIANGLES); - glVertex3f( -50.0f, -925.0f, 0.0f ); - glVertex3f( 50.0f, -925.0f, 0.0f ); - glVertex3f( 0.0f, -1000.0f, 0.0f ); + glVertex3f(-50.0F, -925.0F, 0.0F); + glVertex3f(50.0F, -925.0F, 0.0F); + glVertex3f(0.0F, -1000.0F, 0.0F); glEnd(); - if (anaglyphEnabled) glColor3d(0.75,0.75,0.75); - else glColor3d(0,1,0); // z axis color : green + if (anaglyphEnabled) { + glColor3d(0.75, 0.75, 0.75); + } + else { + glColor3d(0, 1, 0); // z axis color : green + } glBegin(GL_LINES); - glVertex3f( 0.0f, 0.0f, 0.0f); - glVertex3f( 0.0f, 0.0f, 1000.0f); + glVertex3f(0.0F, 0.0F, 0.0F); + glVertex3f(0.0F, 0.0F, 1000.0F); glEnd(); glBegin(GL_TRIANGLES); - glVertex3f(0.0f, -50.0f, 925.0f); - glVertex3f(0.0f, 50.0f, 925.0f); - glVertex3f(0.0f, 0.0f, 1000.0f); + glVertex3f(0.0F, -50.0F, 925.0F); + glVertex3f(0.0F, 50.0F, 925.0F); + glVertex3f(0.0F, 0.0F, 1000.0F); glEnd(); } - if ((depthmap3D.empty()) | (image3D.empty())) // nothing more to render => exit + if ((depthmap3D.empty()) | (image3D.empty())) { // nothing more to render => exit return; + } if (computeVertices3D) { // totally recompute vertices ComputeVertices(); updateVertices3D = false; } - if (updateVertices3D) // partially recompute vertices + if (updateVertices3D) { // partially recompute vertices UpdateVertices(); + } if (computeIndexes3D) { // totally recompute vertices ComputeIndexes(); } - if (computeColors3D) // totally recompute vertices colors + if (computeColors3D) { // totally recompute vertices colors ComputeColors(); + } if (anaglyphEnabled) { glRotatef(-anaglyphShift, 0.0, 1.0, 0.0); @@ -450,15 +472,15 @@ void openGLWidget::paintGL() // 3D rendering glEnableClientState(GL_VERTEX_ARRAY); // vertices drawing mode glEnableClientState(GL_COLOR_ARRAY); // with colors - colorbuffer.bind(); // VBO color buffer used - glColorPointer(3, GL_FLOAT, 0, NULL); // use color buffer - colorbuffer.release(); // done for colors - vertexbuffer.bind(); // the same for vertices - glVertexPointer(3, GL_FLOAT, 0, NULL); - vertexbuffer.release(); - indexbuffer.bind(); // the same for indexes - glDrawElements(GL_TRIANGLE_STRIP, indexarray.size(), GL_UNSIGNED_INT, NULL); // draw triangles - indexbuffer.release(); + colorbuffer.bind(); // VBO color buffer used + glColorPointer(3, GL_FLOAT, 0, nullptr); // use color buffer + colorbuffer.release(); // done for colors + vertexbuffer.bind(); // the same for vertices + glVertexPointer(3, GL_FLOAT, 0, nullptr); + vertexbuffer.release(); + indexbuffer.bind(); // the same for indexes + glDrawElements(GL_TRIANGLE_STRIP, indexarray.size(), GL_UNSIGNED_INT, nullptr); // draw triangles + indexbuffer.release(); glDisableClientState(GL_VERTEX_ARRAY); // finished defining vertices and colors glDisableClientState(GL_COLOR_ARRAY); @@ -467,24 +489,24 @@ void openGLWidget::paintGL() // 3D rendering glClear(GL_DEPTH_BUFFER_BIT); // only reset the depth not the colors this time //glTranslatef(anaglyphShift * 2048, 0, 0); // bad method ! glRotatef(anaglyphShift, 0.0, 1.0, 0.0); // rotate a bit the second image on y axis - glBlendFunc(GL_SRC_ALPHA,GL_ONE_MINUS_SRC_ALPHA); // blend the right image with the already drawn left image + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); // blend the right image with the already drawn left image glColorMask(GL_FALSE, GL_TRUE, GL_TRUE, GL_TRUE); // draw only in the cyan channels (blue + green) glEnableClientState(GL_VERTEX_ARRAY); // exactly as the left image, using the same buffers glEnableClientState(GL_COLOR_ARRAY); colorbuffer.bind(); - glColorPointer(3, GL_FLOAT, 0, NULL); + glColorPointer(3, GL_FLOAT, 0, nullptr); colorbuffer.release(); vertexbuffer.bind(); - glVertexPointer(3, GL_FLOAT, 0, NULL); + glVertexPointer(3, GL_FLOAT, 0, nullptr); vertexbuffer.release(); indexbuffer.bind(); - glDrawElements(GL_TRIANGLE_STRIP, indexarray.size(), GL_UNSIGNED_INT, NULL); + glDrawElements(GL_TRIANGLE_STRIP, indexarray.size(), GL_UNSIGNED_INT, nullptr); indexbuffer.release(); glDisableClientState(GL_VERTEX_ARRAY); glDisableClientState(GL_COLOR_ARRAY); - glColorMask(GL_TRUE,GL_TRUE,GL_TRUE,GL_TRUE); // red, green and blue channels enabled again + glColorMask(GL_TRUE, GL_TRUE, GL_TRUE, GL_TRUE); // red, green and blue channels enabled again } } @@ -496,9 +518,9 @@ void openGLWidget::resizeGL(int width, int height) // called when the widget is glMatrixMode(GL_PROJECTION); // openGL projection mode glLoadIdentity(); #ifdef QT_OPENGL_ES_1 // older versions of openGL - glOrthof(-4 * 2048, +4 * 2048, -4 * 2048 / ratio, +4 * 2048 / ratio, -5000*2048, 5000*2048); // define view rectangle and clipping + glOrthof(-4 * 2048, +4 * 2048, -4 * 2048 / ratio, +4 * 2048 / ratio, -5000 * 2048, 5000 * 2048); // define view rectangle and clipping #else - glOrtho(-4 * 2048, +4 * 2048, -4 * 2048 / ratio, +4 * 2048 / ratio, -5000*2048, 5000*2048); + glOrtho(-4 * 2048, +4 * 2048, -4 * 2048 / ratio, +4 * 2048 / ratio, -5000 * 2048, 5000 * 2048); #endif glMatrixMode(GL_MODELVIEW); // now openGL model mode } @@ -510,7 +532,9 @@ void openGLWidget::resizeGL(int width, int height) // called when the widget is static void NormalizeAngle(int &angle) // angle must be between 0 and 359° { - while (angle < 0) angle = 360 + angle; // no negative values + while (angle < 0) { + angle = 360 + angle; // no negative values + } angle = angle % 360; // 0 -> 359° } @@ -611,9 +635,10 @@ void openGLWidget::mouseMoveEvent(QMouseEvent *event) // move and rotate view wi if (event->buttons() & Qt::LeftButton) { // left button = rotate on x and y axes SetXRotation(xRot + dy); // x = vertical SetYRotation(yRot + dx); // y = horizontal - } else if (event->buttons() & Qt::RightButton) { // right button = move the view on x and y axes - //setXRotation(xRot + 8 * dy); - //setZRotation(zRot + 8 * dx); + } + else if (event->buttons() & Qt::RightButton) { // right button = move the view on x and y axes + //setXRotation(xRot + 8 * dy); + //setZRotation(zRot + 8 * dx); SetXShift(xShift + dx * 48); SetYShift(yShift - dy * 48); @@ -628,10 +653,12 @@ void openGLWidget::wheelEvent(QWheelEvent *event) // zoom int n = event->delta(); // amount of wheel turn //zoom3D += n / 120 / 2; // should work with this (standard) value - if (n < 0) // zoom out + if (n < 0) { // zoom out zoom3D = zoom3D / 1.25; - else // zoom in + } + else { // zoom in zoom3D = zoom3D * 1.25; + } emit zoomChanged(zoom3D); // emit signal @@ -664,4 +691,4 @@ void openGLWidget::keyPressEvent(QKeyEvent *keyEvent) // special keys break; } } -} +} diff --git a/openglwidget.h b/openglwidget.h index 72bcfc2..75044db 100644 --- a/openglwidget.h +++ b/openglwidget.h @@ -95,9 +95,7 @@ class openGLWidget : public QOpenGLWidget void ComputeIndexes(); // create indexes void UpdateVertices(); // update vertices z void ComputeColors(); // recompute colors - int NumberOfVertices(const int &rows, const int &cols); // number of expected vertices for an image - int NumberOfIndexes(const int &rows, const int &cols); // number of expected indexes for an image - GLuint VertexIndex(const int &y, const int &x); // pixel's index in a image + GLuint VertexIndex(const int &y, const int &x) const; // pixel's index in a image public slots: diff --git a/segmentation-depthmap-3d-opencv.pro b/segmentation-depthmap-3d-opencv.pro index b4daaf8..49c8b3b 100644 --- a/segmentation-depthmap-3d-opencv.pro +++ b/segmentation-depthmap-3d-opencv.pro @@ -34,7 +34,7 @@ FORMS += mainwindow.ui CONFIG += link_pkgconfig PKGCONFIG += opencv4 -QMAKE_CXXFLAGS += -std=c++11 +QMAKE_CXXFLAGS += -std=c++17 # icons RESOURCES += resources.qrc