58#include <qpushbutton.h>
61#include <qimagewriter.h>
64#include <qtreewidget.h>
65#include <qapplication.h>
66#include <qmessagebox.h>
67#include <qfiledialog.h>
69#if (QT_VERSION >= QT_VERSION_CHECK(5, 14, 0))
70 #include <qelapsedtimer.h>
72#if (QT_VERSION < QT_VERSION_CHECK(5, 10, 0))
73 #include "qdesktopwidget.h"
77#include <qcolordialog.h>
84#include <qmainwindow.h>
85#include <qtablewidget.h>
86#include <qheaderview.h>
87#include <qscrollarea.h>
93#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
96#include <QOpenGLContext>
100#include "moc_G4OpenGLQtViewer.cpp"
116#if QT_VERSION < 0x060000
119 ResizeWindow(glWidget->devicePixelRatio()*
fVP.GetWindowSizeHintX(),glWidget->devicePixelRatio()*
fVP.GetWindowSizeHintY());
132 if (UI == NULL)
return;
141 bool isTabbedView =
false;
151 QObject::connect(
fUiQt->GetViewerTabWidget(),
152 SIGNAL(currentChanged(
int)),
154 SLOT(currentTabActivated(
int)));
156#if QT_VERSION < 0x060000
158 createViewerPropertiesWidget();
162 createSceneTreeWidget();
169 QWidget *glDialogWidget = getParentWidget();
170 if (glDialogWidget == NULL) {
173 glWidget->setParent(glDialogWidget);
174 QHBoxLayout *mainLayout =
new QHBoxLayout();
176 mainLayout->setContentsMargins(0,0,0,0);
177 mainLayout->setSpacing(0);
179 if (
fGLWidget->inherits(
"QMainWindow")) {
182 glDialogWidget->setLayout(mainLayout);
186#if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0))
187 G4int offset = QGuiApplication::primaryScreen()->availableGeometry().height()
188 - QGuiApplication::screenAt(QPoint(20,20))->availableGeometry().height();
191 - QApplication::desktop()->availableGeometry().height();
194 G4int YPos=
fVP.GetWindowAbsoluteLocationHintY(QGuiApplication::primaryScreen()->availableGeometry().height());
195 if (
fVP.GetWindowAbsoluteLocationHintY(QGuiApplication::primaryScreen()->availableGeometry().height())<
offset) {
199 glDialogWidget->move(
fVP.GetWindowAbsoluteLocationHintX(QGuiApplication::primaryScreen()->availableGeometry().width()),YPos);
200 glDialogWidget->show();
222 ,fLastPickPoint(-1,-1)
225 ,fHoldKeyEvent(false)
226 ,fHoldMoveEvent(false)
227 ,fHoldRotateEvent(false)
231 ,fMovieTempFolderPath(
"")
233 ,fParameterFileName(
"ppmtompeg_encode_parameter_file.par")
234 ,fMovieParametersDialog(NULL)
235 ,fRecordingStep(WAIT)
237 ,fNbMaxFramesPerSec(100)
238 ,fNbMaxAnglePerSec(360)
239 ,fLaunchSpinDelay(100)
240 ,fUISceneTreeWidget(NULL)
241 ,fUIViewerPropertiesWidget(NULL)
242 ,fUIPickInfosWidget(NULL)
245 ,fControlKeyPress(false)
246 ,fShiftKeyPress(false)
248 ,fCheckSceneTreeComponentSignalLock(false)
249 ,fViewerPropertiesTableWidgetIsInit(false)
250 ,fSceneTreeComponentTreeWidget(NULL)
251 ,fSceneTreeWidget(NULL)
252 ,fPVRootNodeCreate(false)
256 ,fTouchableVolumes(
"Touchables")
257 ,fShortcutsDialog(NULL)
258 ,fViewerPropertiesTableWidget(NULL)
259 ,fPickInfosWidget(NULL)
260 ,fPickInfosScrollArea(NULL)
261 ,fTreeWidgetInfosIgnoredCommands(0)
262 ,fSceneTreeDepthSlider(NULL)
264 ,fModelShortNameItem(NULL)
265 ,fMaxPOindexInserted(-1)
267 ,fTreeIconClosed(NULL)
268 ,fLastExportSliderValue(80)
269 ,fLastHighlightColor(
G4Color(0,0,0,0))
270 ,fLastHighlightName(0)
274 if (QCoreApplication::instance () == NULL) {
279 fLastPos3 = QPoint(-1,-1);
280 fLastPos2 = QPoint(-1,-1);
281 fLastPos1 = QPoint(-1,-1);
283 initMovieParameters();
285#if (QT_VERSION < QT_VERSION_CHECK(5, 14, 0))
286 fLastEventTime =
new QTime();
288 fLastEventTime =
new QElapsedTimer();
291 fFileSavePath = QDir::currentPath();
294 QList<QByteArray> formats = QImageWriter::supportedImageFormats ();
295 for (
int i = 0; i < formats.size(); ++i) {
299 const char *
const icon1[]={
357 const char *
const icon2[]={
450 const char *
const search[] = {
462 "OOOOOOOOOOOOOOOOOOO",
463 "OOOOOOOOOOOOOOOOOOO",
464 "OOOOOOOo. .oOOOOOO",
467 "OOOOO. XOOOOX .OOOO",
470 "OOOOO. XOOOOo .OOOO",
474 "OOOOOOOOOOOOO. XOO",
475 "OOOOOOOOOOOOOO. XOO",
476 "OOOOOOOOOOOOOOOoOOO",
477 "OOOOOOOOOOOOOOOOOOO",
478 "OOOOOOOOOOOOOOOOOOO",
479 "OOOOOOOOOOOOOOOOOOO",
480 "OOOOOOOOOOOOOOOOOOO"
483 fSearchIcon =
new QPixmap(search);
484 fTreeIconOpen =
new QPixmap(icon1);
485 fTreeIconClosed =
new QPixmap(icon2);
500 if (fSceneTreeWidget != NULL) {
501 if (fSceneTreeWidget->layout() != NULL) {
502 while ((wItem = fSceneTreeWidget->layout()->takeAt(0)) != 0) {
503 delete wItem->widget();
510 delete fTreeIconOpen;
511 delete fTreeIconClosed;
513#if QT_VERSION < 0x060000
514 G4cout <<removeTempFolder().toStdString().c_str() <<
G4endl;
520#if QT_VERSION < 0x060000
525 if (!qGLW)
return false;
526 return qGLW->isValid();
533void G4OpenGLQtViewer::createPopupMenu() {
535 fContextMenu =
new QMenu(
"All");
537 QMenu *mMouseAction = fContextMenu->addMenu(
"&Mouse actions");
538 fMouseRotateAction = mMouseAction->addAction(
"Rotate",
this, [
this](){ this->toggleMouseAction(1); });
539 fMouseMoveAction = mMouseAction->addAction(
"Move",
this, [
this](){ this->toggleMouseAction(2); });
540 fMousePickAction = mMouseAction->addAction(
"Pick",
this, [
this](){ this->toggleMouseAction(3); });
541 fMouseZoomOutAction = mMouseAction->addAction(
"Zoom out",
this, [
this](){ this->toggleMouseAction(4); });
542 fMouseZoomInAction = mMouseAction->addAction(
"Zoom in",
this, [
this](){ this->toggleMouseAction(5); });
543 QAction *shortcutsAction = mMouseAction->addAction(
"Show shortcuts");
545 fMouseRotateAction->setCheckable(
true);
546 fMouseMoveAction->setCheckable(
true);
547 fMousePickAction->setCheckable(
true);
548 fMouseZoomOutAction->setCheckable(
true);
549 fMouseZoomInAction->setCheckable(
true);
550 shortcutsAction->setCheckable(
false);
552 QObject::connect(shortcutsAction,
553 SIGNAL(triggered(
bool)),
555 SLOT(showShortcuts()));
558 QMenu *mStyle = fContextMenu->addMenu(
"&Style");
560 QMenu *mProjection = mStyle->addMenu(
"&Projection");
563 fProjectionOrtho = mProjection->addAction(
"Orthographic",
this, [
this](){ this->toggleProjection(
true); });
564 fProjectionPerspective = mProjection->addAction(
"Perspective",
this, [
this](){ this->toggleProjection(
false); });
566 QMenu *mDrawing = mStyle->addMenu(
"&Drawing");
567 fDrawingWireframe = mDrawing->addAction(
"Wireframe",
this, [
this](){ this->toggleSurfaceAction(1); });
568 fDrawingLineRemoval = mDrawing->addAction(
"Hidden line removal",
this, [
this](){ this->toggleSurfaceAction(2); });
569 fDrawingSurfaceRemoval = mDrawing->addAction(
"Hidden Surface removal",
this, [
this](){ this->toggleSurfaceAction(3); });
570 fDrawingLineSurfaceRemoval = mDrawing->addAction(
"Hidden line and surface removal",
this, [
this](){ this->toggleSurfaceAction(4); });
572 fDrawingWireframe->setCheckable(
true);
573 fDrawingLineRemoval->setCheckable(
true);
574 fDrawingSurfaceRemoval->setCheckable(
true);
575 fDrawingLineSurfaceRemoval->setCheckable(
true);
579 QAction *backgroundColorChooser ;
581 backgroundColorChooser = mStyle->addAction(
"Background color");
582 QObject ::connect(backgroundColorChooser,
585 SLOT(actionChangeBackgroundColor()));
589 QAction *textColorChooser ;
591 textColorChooser = mStyle->addAction(
"Text color");
592 QObject ::connect(textColorChooser,
595 SLOT(actionChangeTextColor()));
599 QAction *defaultColorChooser ;
601 defaultColorChooser = mStyle->addAction(
"Default color");
602 QObject ::connect(defaultColorChooser,
605 SLOT(actionChangeDefaultColor()));
609 QMenu *mActions = fContextMenu->addMenu(
"&Actions");
610 QAction *createEPS = mActions->addAction(
"Save as ...");
611 QObject ::connect(createEPS,
614 SLOT(actionSaveImage()));
617 QAction *movieParameters = mActions->addAction(
"Save as movie...");
618 QObject ::connect(movieParameters,
621 SLOT(actionMovieParameters()));
627 QMenu *mSpecial = fContextMenu->addMenu(
"S&pecial");
628 QMenu *mTransparency = mSpecial->addMenu(
"Transparency");
629 QAction *transparencyOn = mTransparency->addAction(
"On");
630 QAction *transparencyOff = mTransparency->addAction(
"Off");
633 createRadioAction(transparencyOn,transparencyOff,SLOT(toggleTransparency(
bool)),2);
635 createRadioAction(transparencyOn,transparencyOff,SLOT(toggleTransparency(
bool)),1);
641 QMenu *mAntialiasing = mSpecial->addMenu(
"Antialiasing");
642 QAction *antialiasingOn = mAntialiasing->addAction(
"On");
643 QAction *antialiasingOff = mAntialiasing->addAction(
"Off");
646 createRadioAction(antialiasingOn,antialiasingOff,SLOT(toggleAntialiasing(
bool)),2);
648 createRadioAction(antialiasingOn,antialiasingOff,SLOT(toggleAntialiasing(
bool)),1);
650 mAntialiasing->clear();
653 QMenu *mHaloing = mSpecial->addMenu(
"Haloing");
654 QAction *haloingOn = mHaloing->addAction(
"On");
655 QAction *haloingOff = mHaloing->addAction(
"Off");
657 createRadioAction(haloingOn,haloingOff,SLOT(toggleHaloing(
bool)),2);
659 createRadioAction(haloingOn,haloingOff,SLOT(toggleHaloing(
bool)),1);
664 QMenu *mAux = mSpecial->addMenu(
"Auxiliary edges");
665 QAction *auxOn = mAux->addAction(
"On");
666 QAction *auxOff = mAux->addAction(
"Off");
667 if (!
fVP.IsAuxEdgeVisible()) {
668 createRadioAction(auxOn,auxOff,SLOT(toggleAux(
bool)),2);
670 createRadioAction(auxOn,auxOff,SLOT(toggleAux(
bool)),1);
674 QMenu *mHiddenMarkers = mSpecial->addMenu(
"Hidden markers");
675 QAction *hiddenMarkersOn = mHiddenMarkers->addAction(
"On");
676 QAction *hiddenMarkersOff = mHiddenMarkers->addAction(
"Off");
677 if (
fVP.IsMarkerNotHidden()) {
678 createRadioAction(hiddenMarkersOn,hiddenMarkersOff,SLOT(toggleHiddenMarkers(
bool)),2);
680 createRadioAction(hiddenMarkersOn,hiddenMarkersOff,SLOT(toggleHiddenMarkers(
bool)),1);
685 QMenu *mFullScreen = mSpecial->addMenu(
"&Full screen");
686 fFullScreenOn = mFullScreen->addAction(
"On");
687 fFullScreenOff = mFullScreen->addAction(
"Off");
688 createRadioAction(fFullScreenOn,fFullScreenOff,SLOT(toggleFullScreen(
bool)),2);
697 G4cerr <<
"Visualization window not defined, please choose one before" <<
G4endl;
704 if ( fContextMenu ) {
705 fContextMenu->exec( e->globalPos() );
721void G4OpenGLQtViewer::createRadioAction(QAction *action1,QAction *action2,
const std::string& method,
unsigned int nCheck) {
723 action1->setCheckable(
true);
724 action2->setCheckable(
true);
727 action1->setChecked (
true);
729 action2->setChecked (
true);
731 QObject ::connect(action1, SIGNAL(triggered(
bool)),action2, SLOT(toggle()));
732 QObject ::connect(action2, SIGNAL(triggered(
bool)),action1, SLOT(toggle()));
734 QObject ::connect(action1, SIGNAL(toggled(
bool)),
this, method.c_str());
743void G4OpenGLQtViewer::showShortcuts() {
746 text =
"========= Mouse Shortcuts =========\n";
748 if (
fUiQt->IsIconRotateSelected()) {
749 text +=
"Click and move mouse to rotate volume \n";
750 text +=
"ALT + Click and move mouse to rotate volume (Toggle View/Theta-Phi Direction) \n";
751 text +=
"CTRL + Click and move mouse to zoom in/out \n";
752 text +=
"SHIFT + Click and move mouse to change camera point of view \n";
753 }
else if (
fUiQt->IsIconMoveSelected()) {
754 text +=
"Move camera point of view with mouse \n";
755 }
else if (
fUiQt->IsIconPickSelected()) {
756 text +=
"Click and pick \n";
759 text +=
"Click and move mouse to rotate volume \n";
760 text +=
"ALT + Click and move mouse to rotate volume (Toggle View/Theta-Phi Direction) \n";
761 text +=
"CTRL + Click and zoom mouse to zoom in/out \n";
762 text +=
"SHIFT + Click and zoommove camera point of view \n";
764 text +=
"========= Move Shortcuts ========= \n";
765 text +=
"Press left/right arrows to move volume left/right \n";
766 text +=
"Press up/down arrows to move volume up/down \n";
767 text +=
"Press '+'/'-' to move volume toward/forward \n";
769 text +=
"========= Rotation (Theta/Phi) Shortcuts ========= \n";
770 text +=
"Press SHIFT + left/right arrows to rotate volume left/right \n";
771 text +=
"Press SHIFT + up/down arrows to rotate volume up/down \n";
773 text +=
"========= Rotation (View Direction) Shortcuts ========= \n";
774 text +=
"Press ALT + left/right to rotate volume around vertical direction \n";
775 text +=
"Press ALT + up/down to rotate volume around horizontal direction \n";
777 text +=
"========= Zoom View ========= \n";
778 text +=
"Press CTRL + '+'/'-' to zoom into volume \n";
780 text +=
"========= Misc ========= \n";
781 text +=
"Press ALT +/- to slow/speed rotation/move \n";
782 text +=
"Press H to reset view \n";
783 text +=
"Press Esc to exit FullScreen \n";
785 text +=
"========= Video ========= \n";
786 text +=
"In video mode : \n";
787 text +=
" Press SPACE to Start/Pause video recording \n";
788 text +=
" Press RETURN to Stop video recording \n";
793 if ( fShortcutsDialog == NULL) {
794 fShortcutsDialog =
new QDialog();
795 fShortcutsDialogInfos =
new QTextEdit() ;
796 QVBoxLayout *mainLayout =
new QVBoxLayout;
797 mainLayout->addWidget(fShortcutsDialogInfos);
798 fShortcutsDialog->setLayout(mainLayout);
799 fShortcutsDialog->setWindowTitle(tr(
"Shortcuts"));
802 fShortcutsDialogInfos->setPlainText(text.data());
803 fShortcutsDialog->show();
814void G4OpenGLQtViewer::toggleMouseAction(
int aAction) {
817 fUiQt->SetIconRotateSelected();
818 }
else if (aAction == 2) {
819 fUiQt->SetIconMoveSelected();
820 }
else if (aAction == 3) {
822 }
else if (aAction == 4) {
823 fUiQt->SetIconZoomOutSelected();
824 }
else if (aAction == 5) {
825 fUiQt->SetIconZoomInSelected();
843void G4OpenGLQtViewer::toggleSurfaceAction(
int aAction) {
850 }
else if (aAction ==2) {
853 }
else if (aAction ==3) {
856 }
else if (aAction ==4) {
859 fVP.SetDrawingStyle(d_style);
876void G4OpenGLQtViewer::toggleProjection(
bool check) {
879 fVP.SetOrthogonalProjection ();
881 fVP.SetPerspectiveProjection();
892void G4OpenGLQtViewer::toggleTransparency(
bool check) {
908void G4OpenGLQtViewer::toggleAntialiasing(
bool check) {
912 glDisable (GL_LINE_SMOOTH);
913 glDisable (GL_POLYGON_SMOOTH);
916 glEnable (GL_LINE_SMOOTH);
917 glHint (GL_LINE_SMOOTH_HINT, GL_NICEST);
918 glEnable (GL_POLYGON_SMOOTH);
919 glHint (GL_POLYGON_SMOOTH_HINT, GL_NICEST);
931void G4OpenGLQtViewer::toggleHaloing(
bool check) {
947void G4OpenGLQtViewer::toggleAux(
bool check) {
949 fVP.SetAuxEdgeVisible(
true);
951 fVP.SetAuxEdgeVisible(
false);
959void G4OpenGLQtViewer::togglePicking() {
962 if (!
fVP.IsPicking()) {
963 fUiQt->SetIconPickSelected();
965 fUiQt->SetIconRotateSelected();
971 if (!
fVP.IsPicking()) {
972 UI->
ApplyCommand(std::string(
"/vis/viewer/set/picking true"));
974 UI->
ApplyCommand(std::string(
"/vis/viewer/set/picking false"));
985void G4OpenGLQtViewer::toggleHiddenMarkers(
bool check) {
987 fVP.SetMarkerHidden();
989 fVP.SetMarkerNotHidden();
999void G4OpenGLQtViewer::toggleFullScreen(
bool check) {
1000 if (check !=
fGLWidget->isFullScreen()) {
1007 if (fMovieTempFolderPath ==
"") {
1015 QString filePath =fMovieTempFolderPath+fileName;
1018#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
1019 image = qGLW->grabFrameBuffer();
1021 image = qGLW->grabFramebuffer();
1025 res = image.save(filePath,0);
1028 setRecordingInfos(
"Can't save tmp file "+filePath);
1032 setRecordingInfos(
"File "+fileName+
" saved");
1038void G4OpenGLQtViewer::actionSaveImage() {
1046 qFilename = QFileDialog::getSaveFileName (
fGLWidget,
1053 std::string
name = qFilename.toStdString().c_str();
1060 fFileSavePath = QFileInfo(qFilename).path();
1062 std::string format = selectedFormat->toLower().toStdString().c_str();
1067 std::string filename =
name;
1068 std::string extension =
"";
1069 if (
name.find_last_of(
".") != std::string::npos) {
1070 filename =
name.substr(0,
name.find_last_of(
".") + 1);
1071 extension =
name.substr(
name.find_last_of(
".") + 1);
1076 filename+=
"."+ extension;
1082 G4OpenGLQtExportDialog* exportDialog=
new G4OpenGLQtExportDialog(
fGLWidget,format.c_str(),
fGLWidget->height(),
fGLWidget->width());
1083 if( exportDialog->exec()) {
1108void G4OpenGLQtViewer::actionChangeBackgroundColor() {
1115 const QColor color =
1116 QColorDialog::getColor(Qt::black,
1118 " Get background color and transparency",
1119 QColorDialog::ShowAlphaChannel);
1120 if (color.isValid()) {
1121 G4Colour colour(((
G4double)color.red())/255,
1125 fVP.SetBackgroundColour(colour);
1132void G4OpenGLQtViewer::actionChangeTextColor() {
1133 const QColor& color =
1134 QColorDialog::getColor(Qt::yellow,
1136 " Get text color and transparency",
1137 QColorDialog::ShowAlphaChannel);
1138 if (color.isValid()) {
1139 G4Colour colour(((
G4double)color.red())/255,
1144 fVP.SetDefaultTextColour(colour);
1151void G4OpenGLQtViewer::actionChangeDefaultColor() {
1152 const QColor& color =
1153 QColorDialog::getColor(Qt::white,
1155 " Get default color and transparency",
1156 QColorDialog::ShowAlphaChannel);
1157 if (color.isValid()) {
1158 G4Colour colour(((
G4double)color.red())/255,
1163 fVP.SetDefaultColour(colour);
1171void G4OpenGLQtViewer::actionMovieParameters() {
1172 showMovieParametersDialog();
1176void G4OpenGLQtViewer::showMovieParametersDialog() {
1177 if (!fMovieParametersDialog) {
1178 fMovieParametersDialog=
new G4OpenGLQtMovieDialog(
this,
fGLWidget);
1180 fMovieParametersDialog->checkEncoderSwParameters();
1181 fMovieParametersDialog->checkSaveFileNameParameters();
1182 fMovieParametersDialog->checkTempFolderParameters();
1184 setRecordingInfos(
"ppmtompeg is needed to encode in video format. It is available here: http://netpbm.sourceforge.net ");
1187 fMovieParametersDialog->show();
1210 if (evnt->button() == Qt::RightButton) {
1213 if ((evnt->button() & Qt::LeftButton) && (! (evnt->modifiers() & Qt::ControlModifier ))){
1216 fLastPos1 = evnt->pos();
1217 fLastPos2 = fLastPos1;
1218 fLastPos3 = fLastPos2;
1219 fLastEventTime->start();
1220 if (
fUiQt != NULL) {
1222 if (
fUiQt->IsIconZoomInSelected()) {
1225 float deltaX = ((float)
getWinWidth()/2-evnt->pos().x());
1226 float deltaY = ((float)
getWinHeight()/2-evnt->pos().y());
1233 fVP.IncrementPan(-deltaX*coefTrans,deltaY*coefTrans,0);
1234 fVP.SetZoomFactor(1.5 *
fVP.GetZoomFactor());
1238 }
else if (
fUiQt->IsIconZoomOutSelected()) {
1242 fVP.SetZoomFactor(0.75 *
fVP.GetZoomFactor());
1245 }
else if (
fUiQt->IsIconRotateSelected() ) {
1247 if (fShiftKeyPress) {
1248 fGLWidget->setCursor(QCursor(Qt::SizeAllCursor));
1251 fGLWidget->setCursor(QCursor(Qt::ClosedHandCursor));
1253 }
else if (
fUiQt->IsIconMoveSelected()) {
1254 fGLWidget->setCursor(QCursor(Qt::SizeAllCursor));
1255 }
else if (
fUiQt->IsIconPickSelected()) {
1256 fGLWidget->setCursor(QCursor(Qt::PointingHandCursor));
1266#if QT_VERSION < 0x060000
1269 if (qGLW) qGLW->makeCurrent();}
1273 glGetIntegerv(GL_VIEWPORT, viewport);
1276 double factorX = ((double)viewport[2]/
fGLWidget->width());
1277 double factorY = ((double)viewport[3]/
fGLWidget->height());
1278 fSpinningDelay = (int)fLastEventTime->elapsed();
1279 QPoint delta = (fLastPos3-fLastPos1)*factorX;
1282 fGLWidget->setCursor(QCursor(Qt::ArrowCursor));
1284 if (
fVP.IsPicking()){
1285 if ((delta.x() != 0) || (delta.y() != 0)) {
1290 }
else if (fSpinningDelay < fLaunchSpinDelay ) {
1291 if ((delta.x() == 0) && (delta.y() == 0)) {
1296#if (QT_VERSION < QT_VERSION_CHECK(5, 14, 0))
1299 QElapsedTimer lastMoveTime;
1301 lastMoveTime.start();
1303 float correctionFactor = 5;
1305 if ( lastMoveTime.elapsed() >= (
int)(1000/fNbMaxFramesPerSec)) {
1306 float lTime = 1000.0f/lastMoveTime.elapsed();
1307 if (((((
float)delta.x())/correctionFactor)*lTime > fNbMaxAnglePerSec) ||
1308 ((((
float)delta.x())/correctionFactor)*lTime < -fNbMaxAnglePerSec) ) {
1309 correctionFactor = (float)delta.x()*(lTime/fNbMaxAnglePerSec);
1310 if (delta.x() <0 ) {
1311 correctionFactor = -correctionFactor;
1314 if (((((
float)delta.y())/correctionFactor)*lTime > fNbMaxAnglePerSec) ||
1315 ((((
float)delta.y())/correctionFactor)*lTime < -fNbMaxAnglePerSec) ) {
1316 correctionFactor = (float)delta.y()*(lTime/fNbMaxAnglePerSec);
1317 if (delta.y() <0 ) {
1318 correctionFactor = -correctionFactor;
1329 lastMoveTime.start();
1331 bool rotate =
false;
1334 if (
fUiQt != NULL) {
1335 if (
fUiQt->IsIconRotateSelected()) {
1337 }
else if (
fUiQt->IsIconMoveSelected()) {
1350 rotateQtScene(((
float)delta.x())/correctionFactor,((
float)delta.y())/correctionFactor);
1351 }
else if (fAltKeyPress) {
1352 rotateQtSceneToggle(((
float)delta.x())/correctionFactor,((
float)delta.y())/correctionFactor);
1356 moveScene(-((
float)delta.x())/correctionFactor,-((
float)delta.y())/correctionFactor,0,
true);
1383 Qt::MouseButtons mButtons = evnt->buttons();
1391 fLastPos3 = fLastPos2;
1392 fLastPos2 = fLastPos1;
1394#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
1395 fLastPos1 = QPoint(evnt->x(), evnt->y());
1397 fLastPos1 = QPoint(evnt->position().x(), evnt->position().y());
1400 int deltaX = fLastPos2.x()-fLastPos1.x();
1401 int deltaY = fLastPos2.y()-fLastPos1.y();
1404 if (
fUiQt != NULL) {
1405 if (
fUiQt->IsIconMoveSelected()) {
1410 if (mButtons & Qt::LeftButton) {
1413 }
else if (fAltKeyPress) {
1415 }
else if (fShiftKeyPress) {
1416 unsigned int sizeWin;
1423 float factor = ((float)100/(float)sizeWin) ;
1424 moveScene(-(
float)deltaX*factor,-(
float)deltaY*factor,0,
false);
1425 }
else if (fControlKeyPress) {
1426 fVP.SetZoomFactor(
fVP.GetZoomFactor()*(1+((
float)deltaY)));
1430 if (mButtons & Qt::LeftButton) {
1431 moveScene(-(
float)deltaX,-(
float)deltaY,0,
true);
1435 fLastEventTime->start();
1450 fHoldMoveEvent =
true;
1453 GLdouble coefDepth = 0;
1463 fVP.IncrementPan(-dx*coefTrans,dy*coefTrans,dz*coefDepth);
1469 fHoldMoveEvent =
false;
1480 if (fHoldRotateEvent)
1482 fHoldRotateEvent =
true;
1488 fHoldRotateEvent =
false;
1498 if (fHoldRotateEvent)
1500 fHoldRotateEvent =
true;
1506 fHoldRotateEvent =
false;
1517void G4OpenGLQtViewer::rescaleImage(
1539#if (QT_VERSION < QT_VERSION_CHECK(5, 15, 0))
1540 fVP.SetZoomFactor(
fVP.GetZoomFactor()+(
fVP.GetZoomFactor()*(evnt->delta())/1200));
1542 fVP.SetZoomFactor(
fVP.GetZoomFactor()+(
fVP.GetZoomFactor()*(evnt->angleDelta().y())/1200));
1553 fHoldKeyEvent =
true;
1558 if ((fNoKeyPress) || (evnt->modifiers() == Qt::KeypadModifier )) {
1559 if (evnt->key() == Qt::Key_Down) {
1562 else if (evnt->key() == Qt::Key_Up) {
1565 if (evnt->key() == Qt::Key_Left) {
1568 else if (evnt->key() == Qt::Key_Right) {
1571 if (evnt->key() == Qt::Key_Minus) {
1574 else if (evnt->key() == Qt::Key_Plus) {
1578 if (evnt->key() == Qt::Key_Escape) {
1579 toggleFullScreen(
false);
1588 if ((evnt->key() == Qt::Key_Return) || (evnt->key() == Qt::Key_Enter)){
1591 if (evnt->key() == Qt::Key_Space){
1596 if (evnt->key() == Qt::Key_H){
1603 if (fShiftKeyPress) {
1604 fGLWidget->setCursor(QCursor(Qt::SizeAllCursor));
1606 if (evnt->key() == Qt::Key_Down) {
1609 else if (evnt->key() == Qt::Key_Up) {
1612 if (evnt->key() == Qt::Key_Left) {
1615 else if (evnt->key() == Qt::Key_Right) {
1618 if (evnt->key() == Qt::Key_Plus) {
1625 if ((fAltKeyPress)) {
1626 fGLWidget->setCursor(QCursor(Qt::ClosedHandCursor));
1628 if (evnt->key() == Qt::Key_Down) {
1631 else if (evnt->key() == Qt::Key_Up) {
1634 if (evnt->key() == Qt::Key_Left) {
1637 else if (evnt->key() == Qt::Key_Right) {
1642 if (evnt->key() == Qt::Key_Plus) {
1646 else if (evnt->key() == Qt::Key_Minus) {
1653 if ((fControlKeyPress)) {
1654 if (evnt->key() == Qt::Key_Plus) {
1655 fVP.SetZoomFactor(
fVP.GetZoomFactor()*(1+fDeltaZoom));
1658 else if (evnt->key() == Qt::Key_Minus) {
1659 fVP.SetZoomFactor(
fVP.GetZoomFactor()*(1-fDeltaZoom));
1664 fHoldKeyEvent =
false;
1670 fGLWidget->setCursor(QCursor(Qt::ArrowCursor));
1678 fAltKeyPress =
false;
1679 fShiftKeyPress =
false;
1680 fControlKeyPress =
false;
1682 if (modifier & Qt::AltModifier ) {
1683 fAltKeyPress =
true;
1684 fNoKeyPress =
false;
1686 if (modifier & Qt::ShiftModifier ) {
1687 fShiftKeyPress =
true;
1688 fNoKeyPress =
false;
1690 if (modifier & Qt::ControlModifier ) {
1691 fControlKeyPress =
true;
1692 fNoKeyPress =
false;
1702 if (!fMovieParametersDialog) {
1703 showMovieParametersDialog();
1705 setRecordingStatus(STOP);
1709 if (!(fMovieParametersDialog->checkEncoderSwParameters())) {
1710 setRecordingStatus(BAD_ENCODER);
1711 }
else if (!(fMovieParametersDialog->checkSaveFileNameParameters())) {
1712 setRecordingStatus(BAD_OUTPUT);
1716 setRecordingInfos(
"No frame to encode.");
1725 if (!fMovieParametersDialog) {
1726 showMovieParametersDialog();
1729 fMovieParametersDialog->checkEncoderSwParameters();
1730 fMovieParametersDialog->checkSaveFileNameParameters();
1732 if (fRecordingStep == STOP) {
1733 setRecordingStatus(SAVE);
1746 if ( fRecordingStep == WAIT) {
1749 showMovieParametersDialog();
1750 setRecordingInfos(
"You should specified the temp folder in order to make movie");
1754 QString tmp = removeTempFolder();
1756 setRecordingInfos(tmp);
1759 tmp = createTempFolder();
1761 setRecordingInfos(
"Can't create temp folder."+tmp);
1767 if (fRecordingStep == WAIT) {
1768 setRecordingStatus(START);
1769 }
else if (fRecordingStep == START) {
1770 setRecordingStatus(PAUSE);
1771 }
else if (fRecordingStep == PAUSE) {
1772 setRecordingStatus(CONTINUE);
1773 }
else if (fRecordingStep == CONTINUE) {
1774 setRecordingStatus(PAUSE);
1778void G4OpenGLQtViewer::setRecordingStatus(RECORDING_STEP step) {
1780 fRecordingStep = step;
1787 QString txtStatus =
"";
1788 if (fRecordingStep == WAIT) {
1789 txtStatus =
"Waiting to start...";
1791 }
else if (fRecordingStep == START) {
1792 txtStatus =
"Start Recording...";
1793 }
else if (fRecordingStep == PAUSE) {
1794 txtStatus =
"Pause Recording...";
1795 }
else if (fRecordingStep == CONTINUE) {
1796 txtStatus =
"Continue Recording...";
1797 }
else if (fRecordingStep == STOP) {
1798 txtStatus =
"Stop Recording...";
1799 }
else if (fRecordingStep == READY_TO_ENCODE) {
1800 txtStatus =
"Ready to Encode...";
1801 }
else if (fRecordingStep == ENCODING) {
1802 txtStatus =
"Encoding...";
1803 }
else if (fRecordingStep == FAILED) {
1804 txtStatus =
"Failed to encode...";
1805 }
else if ((fRecordingStep == BAD_ENCODER)
1806 || (fRecordingStep == BAD_OUTPUT)
1807 || (fRecordingStep == BAD_TMP)) {
1808 txtStatus =
"Correct above errors first";
1809 }
else if (fRecordingStep == SUCCESS) {
1810 txtStatus =
"File encoded successfully";
1814 if (fMovieParametersDialog) {
1815 fMovieParametersDialog->setRecordingStatus(txtStatus);
1819 setRecordingInfos(
"");
1823void G4OpenGLQtViewer::setRecordingInfos(
const QString& txt) {
1824 if (fMovieParametersDialog) {
1833void G4OpenGLQtViewer::initMovieParameters() {
1837 fProcess =
new QProcess();
1839 QObject ::connect(fProcess,SIGNAL(finished (
int)),
1840 this,SLOT(processLookForFinished()));
1841 fProcess->setProcessChannelMode(QProcess::MergedChannels);
1842#if (QT_VERSION < QT_VERSION_CHECK(5, 15, 0))
1843 fProcess->start (
"which ppmtompeg");
1845 fProcess->start (
"which ppmtompeg", QStringList());
1852 return fEncoderPath;
1862 return "ppmtompeg is needed to encode in video format. It is available here: http://netpbm.sourceforge.net ";
1865 path = QDir::cleanPath(path);
1866 QFileInfo *f =
new QFileInfo(path);
1868 return "File does not exist";
1869 }
else if (f->isDir()) {
1870 return "This is a directory";
1871 }
else if (!f->isExecutable()) {
1872 return "File exist but is not executable";
1873 }
else if (!f->isFile()) {
1874 return "This is not a file";
1876 fEncoderPath = path;
1878 if (fRecordingStep == BAD_ENCODER) {
1879 setRecordingStatus(STOP);
1886 if ((fRecordingStep == START) || (fRecordingStep == CONTINUE)) {
1893 if (fRecordingStep == PAUSE) {
1900 if (fRecordingStep == ENCODING) {
1907 if (fRecordingStep == WAIT) {
1914 if (fRecordingStep == STOP) {
1921 if (fRecordingStep == FAILED) {
1928 if (fRecordingStep == SUCCESS) {
1935 if (fRecordingStep == BAD_ENCODER) {
1941 if (fRecordingStep == BAD_TMP) {
1947 if (fRecordingStep == BAD_OUTPUT) {
1954 fRecordingStep = BAD_ENCODER;
1958 fRecordingStep = BAD_TMP;
1962 fRecordingStep = BAD_OUTPUT;
1967 fRecordingStep = WAIT;
1973 if (fRecordingStep == READY_TO_ENCODE) {
1980 setRecordingStatus(WAIT);
1990 return "Path does not exist";
1992 path = QDir::cleanPath(path);
1993 QFileInfo *d =
new QFileInfo(path);
1995 return "Path does not exist";
1996 }
else if (!d->isDir()) {
1997 return "This is not a directory";
1998 }
else if (!d->isReadable()) {
1999 return path +
" is read protected";
2000 }
else if (!d->isWritable()) {
2001 return path +
" is write protected";
2004 if (fRecordingStep == BAD_TMP) {
2005 setRecordingStatus(WAIT);
2007 fTempFolderPath = path;
2014 return fTempFolderPath;
2024 return "Path does not exist";
2027 QFileInfo *file =
new QFileInfo(path);
2028 QDir dir = file->dir();
2029 path = QDir::cleanPath(path);
2030 if (file->exists()) {
2031 return "File already exist, please choose a new one";
2032 }
else if (!dir.exists()) {
2033 return "Dir does not exist";
2034 }
else if (!dir.isReadable()) {
2035 return path +
" is read protected";
2038 if (fRecordingStep == BAD_OUTPUT) {
2039 setRecordingStatus(STOP);
2041 fSaveFileName = path;
2048 return fSaveFileName ;
2055QString G4OpenGLQtViewer::createTempFolder() {
2056 fMovieTempFolderPath =
"";
2062 QString sep = QString(QDir::separator());
2063 QString path = sep+
"QtMovie_"+QDateTime::currentDateTime ().toString(
"dd-MM-yyyy_hh-mm-ss")+sep;
2064 QDir *d =
new QDir(QDir::cleanPath(fTempFolderPath));
2066 if (d->exists(path)) {
2067 return "Folder "+path+
" already exists.Please remove it first";
2069 if (d->mkdir(fTempFolderPath+path)) {
2070 fMovieTempFolderPath = fTempFolderPath+path;
2073 return "Can't create "+fTempFolderPath+path;
2078QString G4OpenGLQtViewer::removeTempFolder() {
2080 if (fMovieTempFolderPath ==
"") {
2083 QDir *d =
new QDir(QDir::cleanPath(fMovieTempFolderPath));
2088 d->setFilter( QDir::Files );
2089 QStringList subDirList = d->entryList();
2092 for (QStringList::ConstIterator it = subDirList.begin() ;(it != subDirList.end()) ; it++) {
2093 const QString currentFile = *it;
2094 if (!d->remove(currentFile)) {
2096 QString file = fMovieTempFolderPath+currentFile;
2097 error +=
"Removing file failed : "+file;
2102 if (d->rmdir(fMovieTempFolderPath)) {
2103 fMovieTempFolderPath =
"";
2106 return "Dir "+fMovieTempFolderPath+
" should be empty, but could not remove it";
2110 return "Could not remove "+fMovieTempFolderPath+
" because of the following errors :"+error;
2127 bool increaseFileNumber =
true;
2129 if (name.size() != name.substr(name.find_last_of(
".") + 1).size()) {
2130 increaseFileNumber =
false;
2135 if ((width !=-1) && (height != -1)) {
2145#if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
2146 image = qGLW->grabFrameBuffer();
2148 image = qGLW->grabFramebuffer();
2168 fp = fopen (QString(fMovieTempFolderPath+fParameterFileName).toStdString().c_str(),
"w");
2171 setRecordingInfos(
"Generation of parameter file failed");
2175 fprintf (fp,
"# Pattern affects speed, quality and compression. See the User's Guide\n");
2176 fprintf (fp,
"# for more info.\n");
2178 fprintf (fp,
"PATTERN I\n");
2181 fprintf (fp,
"# You must specify the type of the input files. The choices are:\n");
2182 fprintf (fp,
"# YUV, PPM, JMOVIE, Y, JPEG, PNM\n");
2183 fprintf (fp,
"# (must be upper case)\n");
2185 fprintf (fp,
"BASE_FILE_FORMAT PPM\n");
2188 fprintf (fp,
"# If you are using YUV, there are different supported file formats.\n");
2189 fprintf (fp,
"# EYUV or UCB are the same as previous versions of this encoder.\n");
2190 fprintf (fp,
"# (All the Y's, then U's then V's, in 4:2:0 subsampling.)\n");
2191 fprintf (fp,
"# Other formats, such as Abekas, Phillips, or a general format are\n");
2192 fprintf (fp,
"# permissible, the general format is a string of Y's, U's, and V's\n");
2193 fprintf (fp,
"# to specify the file order.\n");
2195 fprintf (fp,
"INPUT_FORMAT UCB\n");
2197 fprintf (fp,
"# the conversion statement\n");
2199 fprintf (fp,
"# Each occurrence of '*' will be replaced by the input file\n");
2201 fprintf (fp,
"# e.g., if you have a bunch of GIF files, then this might be:\n");
2202 fprintf (fp,
"# INPUT_CONVERT giftoppm *\n");
2204 fprintf (fp,
"# e.g., if you have a bunch of files like a.Y a.U a.V, etc., then:\n");
2205 fprintf (fp,
"# INPUT_CONVERT cat *.Y *.U *.V\n");
2207 fprintf (fp,
"# e.g., if you are grabbing from laser disc you might have something like\n");
2208 fprintf (fp,
"# INPUT_CONVERT goto frame *; grabppm\n");
2209 fprintf (fp,
"# 'INPUT_CONVERT *' means the files are already in the base file format\n");
2211 fprintf (fp,
"INPUT_CONVERT * \n");
2213 fprintf (fp,
"# number of frames in a GOP.\n");
2215 fprintf (fp,
"# since each GOP must have at least one I-frame, the encoder will find the\n");
2216 fprintf (fp,
"# the first I-frame after GOP_SIZE frames to start the next GOP\n");
2218 fprintf (fp,
"# later, will add more flexible GOP signalling\n");
2220 fprintf (fp,
"GOP_SIZE 1\n");
2222 fprintf (fp,
"# number of slices in a frame\n");
2224 fprintf (fp,
"# 1 is a good number. another possibility is the number of macroblock rows\n");
2225 fprintf (fp,
"# (which is the height divided by 16)\n");
2227 fprintf (fp,
"SLICES_PER_FRAME 1\n");
2228 fprintf (fp,
"PIXEL HALF");
2230 fprintf (fp,
"# directory to get all input files from (makes this file easier to read)\n");
2231 fprintf (fp,
"INPUT_DIR %s\n",fMovieTempFolderPath.toStdString().c_str());
2233 fprintf (fp,
"# There are a bunch of ways to specify the input files.\n");
2234 fprintf (fp,
"# from a simple one-per-line listing, to the following \n");
2235 fprintf (fp,
"# way of numbering them. See the manual for more information.\n");
2236 fprintf (fp,
"INPUT\n");
2237 fprintf (fp,
"# '*' is replaced by the numbers 01, 02, 03, 04\n");
2238 fprintf (fp,
"# if I instead do [01-11], it would be 01, 02, ..., 09, 10, 11\n");
2239 fprintf (fp,
"# if I instead do [1-11], it would be 1, 2, 3, ..., 9, 10, 11\n");
2240 fprintf (fp,
"# if I instead do [1-11+3], it would be 1, 4, 7, 10\n");
2241 fprintf (fp,
"# the program assumes none of your input files has a name ending in ']'\n");
2242 fprintf (fp,
"# if you do, too bad!!!\n");
2246 fprintf (fp,
"# can have more files here if you want...there is no limit on the number\n");
2247 fprintf (fp,
"# of files\n");
2248 fprintf (fp,
"END_INPUT\n");
2252 fprintf (fp,
"# Many of the remaining options have to do with the motion search and qscale\n");
2254 fprintf (fp,
"# FULL or HALF -- must be upper case\n");
2255 fprintf (fp,
"# Should be FULL for computer generated images\n");
2256 fprintf (fp,
"PIXEL FULL\n");
2258 fprintf (fp,
"# means +/- this many pixels for both P and B frame searches\n");
2259 fprintf (fp,
"# specify two numbers if you wish to serc different ranges in the two.\n");
2260 fprintf (fp,
"RANGE 10\n");
2262 fprintf (fp,
"# The two search algorithm parameters below mostly affect speed,\n");
2263 fprintf (fp,
"# with some affect on compression and almost none on quality.\n");
2265 fprintf (fp,
"# this must be one of {EXHAUSTIVE, SUBSAMPLE, LOGARITHMIC}\n");
2266 fprintf (fp,
"PSEARCH_ALG LOGARITHMIC\n");
2268 fprintf (fp,
"# this must be one of {SIMPLE, CROSS2, EXHAUSTIVE}\n");
2270 fprintf (fp,
"# note that EXHAUSTIVE is really, really, really slow\n");
2272 fprintf (fp,
"BSEARCH_ALG SIMPLE\n");
2275 fprintf (fp,
"# these specify the q-scale for I, P, and B frames\n");
2276 fprintf (fp,
"# (values must be between 1 and 31)\n");
2277 fprintf (fp,
"# These are the Qscale values for the entire frame in variable bit-rate\n");
2278 fprintf (fp,
"# mode, and starting points (but not important) for constant bit rate\n");
2281 fprintf (fp,
"# Qscale (Quantization scale) affects quality and compression,\n");
2282 fprintf (fp,
"# but has very little effect on speed.\n");
2284 fprintf (fp,
"IQSCALE 4\n");
2285 fprintf (fp,
"PQSCALE 5\n");
2286 fprintf (fp,
"BQSCALE 12\n");
2288 fprintf (fp,
"# this must be ORIGINAL or DECODED\n");
2289 fprintf (fp,
"REFERENCE_FRAME ORIGINAL\n");
2291 fprintf (fp,
"# for parallel parameters see parallel.param in the examples subdirectory\n");
2293 fprintf (fp,
"# if you want constant bit-rate mode, specify it as follows (number is bits/sec):\n");
2294 fprintf (fp,
"#BIT_RATE 1000000\n");
2296 fprintf (fp,
"# To specify the buffer size (327680 is default, measused in bits, for 16bit words)\n");
2297 fprintf (fp,
"BUFFER_SIZE 327680\n");
2299 fprintf (fp,
"# The frame rate is the number of frames/second (legal values:\n");
2300 fprintf (fp,
"# 23.976, 24, 25, 29.97, 30, 50 ,59.94, 60\n");
2301 fprintf (fp,
"FRAME_RATE 30\n");
2303 fprintf (fp,
"# There are many more options, see the users manual for examples....\n");
2304 fprintf (fp,
"# ASPECT_RATIO, USER_DATA, GAMMA, IQTABLE, etc.\n");
2309 setRecordingInfos(
"Parameter file "+fParameterFileName+
" generated in "+fMovieTempFolderPath);
2310 setRecordingStatus(READY_TO_ENCODE);
2317 setRecordingStatus(ENCODING);
2319 fProcess =
new QProcess();
2320 QObject ::connect(fProcess,SIGNAL(finished (
int,QProcess::ExitStatus)),
2321 this,SLOT(processEncodeFinished()));
2322 QObject ::connect(fProcess,SIGNAL(readyReadStandardOutput ()),
2323 this,SLOT(processEncodeStdout()));
2324#if (QT_VERSION < QT_VERSION_CHECK(5, 10, 0))
2325 fProcess->setReadChannelMode(QProcess::MergedChannels);
2327 fProcess->setProcessChannelMode(QProcess::MergedChannels);
2329 fProcess->start (fEncoderPath, QStringList(fMovieTempFolderPath+fParameterFileName));
2335void G4OpenGLQtViewer::processEncodeStdout()
2337 QString tmp = fProcess->readAllStandardOutput ().data();
2338 auto start = tmp.lastIndexOf(
"ESTIMATED TIME");
2339 tmp = tmp.mid(start,tmp.indexOf(
"\n",start)-start);
2340 setRecordingInfos(tmp);
2344void G4OpenGLQtViewer::processEncodeFinished()
2348 txt = getProcessErrorMsg();
2350 setRecordingStatus(SUCCESS);
2352 setRecordingStatus(FAILED);
2358void G4OpenGLQtViewer::processLookForFinished()
2361 QString txt = getProcessErrorMsg();
2365 fEncoderPath = QString(fProcess->readAllStandardOutput ().data()).trimmed();
2367 if (fEncoderPath.contains(
" ")) {
2369 }
else if (!fEncoderPath.contains(
"ppmtompeg")) {
2379QString G4OpenGLQtViewer::getProcessErrorMsg()
2382 if (fProcess->exitCode() != 0) {
2383 switch (fProcess->error()) {
2384 case QProcess::FailedToStart:
2385 txt =
"The process failed to start. Either the invoked program is missing, or you may have insufficient permissions to invoke the program.\n";
2387 case QProcess::Crashed:
2388 txt =
"The process crashed some time after starting successfully.\n";
2390 case QProcess::Timedout:
2391 txt =
"The last waitFor...() function timed out. The state of QProcess is unchanged, and you can try calling waitFor...() again.\n";
2393 case QProcess::WriteError:
2394 txt =
"An error occurred when attempting to write to the process. For example, the process may not be running, or it may have closed its input channel.\n";
2396 case QProcess::ReadError:
2397 txt =
"An error occurred when attempting to read from the process. For example, the process may not be running.\n";
2399 case QProcess::UnknownError:
2400 txt =
"An unknown error occurred. This is the default return value of error().\n";
2410QWidget *G4OpenGLQtViewer::getParentWidget()
2416 QDialog* dialog = NULL;
2420 QWidgetList wl = QApplication::allWidgets();
2421 QWidget *widget = NULL;
2422 for (
int i=0; i < wl.size(); i++) {
2424 if ((found==
false) && (widget->inherits(
"QMainWindow"))) {
2425 dialog =
new QDialog(widget,Qt::WindowTitleHint | Qt::WindowSystemMenuHint | Qt::WindowMinMaxButtonsHint);
2431 dialog =
new QDialog();
2434 dialog=
new QDialog();
2444void G4OpenGLQtViewer::createSceneTreeWidget() {
2445 fUISceneTreeWidget =
fUiQt->GetSceneTreeWidget();
2447 if (!fUISceneTreeWidget) {
2452 QLayoutItem * wItem;
2454 if (fUISceneTreeWidget->layout()->count() ) {
2455 for(
int idx = 0; idx < fUISceneTreeWidget->layout()->count(); idx++){
2456 wItem = fUISceneTreeWidget->layout()->itemAt(idx);
2457 if (fSceneTreeWidget) {
2458 if(
dynamic_cast<QWidget *
>(wItem->widget())) {
2459 if (wItem->widget()->windowTitle() == fSceneTreeWidget->windowTitle()) {
2460 wItem->widget()->show();
2463 wItem->widget()->hide();
2474 fSceneTreeWidget =
new QWidget();
2475 QVBoxLayout* layoutSceneTree =
new QVBoxLayout();
2476 fSceneTreeWidget->setStyleSheet (
"padding: 0px ");
2478 fSceneTreeWidget->setLayout(layoutSceneTree);
2479 fSceneTreeWidget->layout()->setContentsMargins(5,5,5,5);
2480 fSceneTreeWidget->setWindowTitle(QString(
GetName().data()));
2482 if (fUISceneTreeWidget != NULL) {
2487 if (
dynamic_cast<G4OpenGLStoredQtViewer*
> (
this)) {
2488 createSceneTreeComponent();
2494void G4OpenGLQtViewer::createSceneTreeComponent(){
2496 QLayout* vLayout = fSceneTreeWidget->layout();
2499 QWidget* coutButtonWidget =
new QWidget();
2500 QHBoxLayout* layoutCoutTBButtons =
new QHBoxLayout();
2502 fFilterOutput =
new QLineEdit();
2503 fFilterOutput->setToolTip(
"Filter output by...");
2504 fFilterOutput->setStyleSheet (
"padding: 0px ");
2506 QPixmap* searchIcon =
fUiQt->getSearchIcon();
2507 fFilterOutput->addAction(*searchIcon,QLineEdit::TrailingPosition);
2508 fFilterOutput->setStyleSheet (
"border-radius:7px;");
2509 layoutCoutTBButtons->addWidget(fFilterOutput);
2511 coutButtonWidget->setLayout(layoutCoutTBButtons);
2512 vLayout->addWidget(coutButtonWidget);
2515 vLayout->setContentsMargins(0,0,0,0);
2518 fSceneTreeComponentTreeWidget =
new QTreeWidget();
2519 fSceneTreeComponentTreeWidget->setSelectionMode(QAbstractItemView::SingleSelection);
2520 fSceneTreeComponentTreeWidget->setHeaderLabel (
"Scene tree : "+QString(
GetName().data()));
2521 fSceneTreeComponentTreeWidget->setColumnHidden (1,
true);
2522 fSceneTreeComponentTreeWidget->setColumnHidden (2,
true);
2523 fSceneTreeComponentTreeWidget->setColumnHidden (3,
true);
2528 vLayout->addWidget(fSceneTreeComponentTreeWidget);
2530 connect(fSceneTreeComponentTreeWidget,SIGNAL(itemChanged(QTreeWidgetItem*,
int)),SLOT(sceneTreeComponentItemChanged(QTreeWidgetItem*,
int)));
2531 connect(fSceneTreeComponentTreeWidget,SIGNAL(itemSelectionChanged ()),SLOT(sceneTreeComponentSelected()));
2532 connect(fSceneTreeComponentTreeWidget,SIGNAL(itemDoubleClicked ( QTreeWidgetItem*,
int)),SLOT(changeColorAndTransparency( QTreeWidgetItem*,
int)));
2536 QWidget *helpWidget =
new QWidget();
2537 QHBoxLayout *helpLayout =
new QHBoxLayout();
2539 QWidget* depthWidget =
new QWidget();
2540 QWidget *showBox =
new QWidget(depthWidget);
2541 QHBoxLayout *showBoxLayout =
new QHBoxLayout();
2544 showBoxLayout->setContentsMargins(5,5,5,5);
2546 QLabel *zero =
new QLabel();
2547 zero->setText(
"Show all");
2548 QLabel *one =
new QLabel();
2549 one->setText(
"Hide all");
2550 fSceneTreeDepthSlider =
new QSlider ( Qt::Horizontal);
2551 fSceneTreeDepthSlider->setMaximum (1000);
2552 fSceneTreeDepthSlider->setMinimum (0);
2553 fSceneTreeDepthSlider->setTickPosition(QSlider::TicksAbove);
2555 fSceneTreeDepthSlider->setMinimumWidth (40);
2557 showBoxLayout->addWidget(zero);
2558 showBoxLayout->addWidget(fSceneTreeDepthSlider);
2559 showBoxLayout->addWidget(one);
2561 showBox->setLayout(showBoxLayout);
2563 helpLayout->addWidget(showBox);
2564 helpWidget->setLayout(helpLayout);
2565 helpLayout->setContentsMargins(0,0,0,0);
2567 vLayout->addWidget(helpWidget);
2569 connect( fSceneTreeDepthSlider, SIGNAL( valueChanged(
int) ),
this, SLOT( changeDepthInSceneTree(
int) ) );
2570 connect( fFilterOutput, SIGNAL( textEdited (
const QString &) ),
this, SLOT(changeSearchSelection()));
2571 fTreeItemModels.clear();
2573 fPVRootNodeCreate =
false;
2575 fMaxPOindexInserted = -1;
2581void G4OpenGLQtViewer::createViewerPropertiesWidget() {
2584 fUIViewerPropertiesWidget =
fUiQt->GetViewerPropertiesWidget();
2586 if (!fUIViewerPropertiesWidget) {
2591 QLayoutItem * wItem;
2592 if (fUIViewerPropertiesWidget->layout()->count()) {
2593 while ((wItem = fUIViewerPropertiesWidget->layout()->takeAt(0)) != 0) {
2594 delete wItem->widget();
2600 QGroupBox *groupBox =
new QGroupBox();
2601 groupBox->setTitle(
GetName().data());
2602 QVBoxLayout *vbox =
new QVBoxLayout;
2605 fViewerPropertiesTableWidget =
new QTableWidget();
2607 QSizePolicy vPolicy = fViewerPropertiesTableWidget->sizePolicy();
2608 vPolicy.setVerticalStretch(4);
2610 vbox->addWidget(fViewerPropertiesTableWidget);
2611 groupBox->setLayout(vbox);
2612 fUIViewerPropertiesWidget->layout()->addWidget(groupBox);
2614 connect(fViewerPropertiesTableWidget, SIGNAL(itemChanged(QTableWidgetItem*)),
this, SLOT(tableWidgetViewerSetItemChanged(QTableWidgetItem *)));
2618 QDialog* dial =
static_cast<QDialog*
> (fUIViewerPropertiesWidget->parent());
2621 dial->setWindowTitle(QString(
"Viewer properties - ")+
GetName());
2626void G4OpenGLQtViewer::createPickInfosWidget(){
2629 fUIPickInfosWidget =
fUiQt->GetPickInfosWidget();
2631 if (!fUIPickInfosWidget) {
2636 QLayoutItem * wItem;
2637 if (fUIPickInfosWidget->layout()->count()) {
2638 while ((wItem = fUIPickInfosWidget->layout()->takeAt(0)) != 0) {
2639 delete wItem->widget();
2644 QGroupBox *groupBox =
new QGroupBox(
"");
2645 QVBoxLayout *vbox =
new QVBoxLayout;
2648 QWidget *pickingInfoWidget =
new QWidget();
2649 QHBoxLayout *pickingInfoLayout =
new QHBoxLayout();
2651 pickingInfoWidget->setStyleSheet (
"padding-left: 0px; border:0px;");
2652 pickingInfoWidget->setLayout(pickingInfoLayout);
2654 vbox->addWidget(pickingInfoWidget);
2657 fPickInfosScrollArea =
new QScrollArea();
2658 fPickInfosScrollArea->setWidgetResizable(
true);
2661 fPickInfosWidget =
new QWidget();
2662 fPickInfosWidget->setStyleSheet (
"padding: 0px ");
2664 QVBoxLayout* vLayout =
new QVBoxLayout();
2665 fPickInfosWidget->setLayout (vLayout);
2666 fPickInfosScrollArea->setWidget(fPickInfosWidget);
2668 QSizePolicy vPolicy = fPickInfosWidget->sizePolicy();
2669 vPolicy.setVerticalStretch(4);
2670 vbox->addWidget(fPickInfosScrollArea);
2671 pickingInfoLayout->setContentsMargins(0,0,0,0);
2672 vLayout->setContentsMargins(0,0,0,0);
2673 vbox->setContentsMargins(1,1,1,1);
2675 groupBox->setLayout(vbox);
2676 fUIPickInfosWidget->layout()->addWidget(groupBox);
2684void G4OpenGLQtViewer::setCheckComponent(QTreeWidgetItem* item,
bool check)
2688 const PVPath& fullPath = fTreeItemModels[item->data(0,Qt::UserRole).toInt()];
2690 if (fullPath.size() > 0) {
2699 item->setCheckState(0,Qt::Checked);
2701 item->setCheckState(0,Qt::Unchecked);
2703 updatePositivePoIndexSceneTreeWidgetQuickMap(item->data(0,Qt::UserRole).toInt(),item);
2704 int nChildCount = item->childCount();
2705 for (
int i = 0; i < nChildCount; i++) {
2706 setCheckComponent(item->child(i),check);
2711#if QT_VERSION < 0x060000
2714static void transform_point(GLdouble out[4],
const GLdouble m[16],
const GLdouble in[4])
2716#define M(row,col) m[col*4+row]
2718 M(0, 0) * in[0] +
M(0, 1) * in[1] +
M(0, 2) * in[2] +
M(0, 3) * in[3];
2720 M(1, 0) * in[0] +
M(1, 1) * in[1] +
M(1, 2) * in[2] +
M(1, 3) * in[3];
2722 M(2, 0) * in[0] +
M(2, 1) * in[1] +
M(2, 2) * in[2] +
M(2, 3) * in[3];
2724 M(3, 0) * in[0] +
M(3, 1) * in[1] +
M(3, 2) * in[2] +
M(3, 3) * in[3];
2727inline GLint project_point(GLdouble objx, GLdouble objy, GLdouble objz,
2728 const GLdouble model[16],
const GLdouble proj[16],
2729 const GLint viewport[4],
2730 GLdouble * winx, GLdouble * winy, GLdouble * winz)
2732 GLdouble in[4], out[4];
2738 transform_point(out, model, in);
2739 transform_point(in, proj, out);
2748 *winx = viewport[0] + (1 + in[0]) * viewport[2] / 2;
2749 *winy = viewport[1] + (1 + in[1]) * viewport[3] / 2;
2751 *winz = (1 + in[2]) / 2;
2754static void render_text(QOpenGLWidget& widget,
2755 double world_x,
double world_y,
double world_z,
2756 double offset_x,
double offset_y,
2758 const QColor& color,
2761 GLdouble model[4][4];
2762 glGetDoublev(GL_MODELVIEW_MATRIX, &model[0][0]);
2763 GLdouble proj[4][4];
2764 glGetDoublev(GL_PROJECTION_MATRIX, &proj[0][0]);
2766 glGetIntegerv(GL_VIEWPORT, &view[0]);
2768 GLdouble textPosX = 0, textPosY = 0, textPosZ = 0;
2769 project_point(world_x, world_y, world_z,
2770 &model[0][0], &proj[0][0], &view[0],
2771 &textPosX, &textPosY, &textPosZ);
2773 textPosX /= GLdouble(widget.devicePixelRatio());
2774 textPosY /= GLdouble(widget.devicePixelRatio());
2776 textPosY = GLdouble(widget.height()) - textPosY;
2778 textPosX += offset_x;
2779 textPosY += offset_y;
2781 GLboolean GL_BLEND_enabled = glIsEnabled(GL_BLEND);
2783 QPainter painter(&widget);
2784 painter.setPen(color);
2785 painter.setFont(font);
2786 painter.setRenderHints(QPainter::Antialiasing | QPainter::TextAntialiasing);
2787 painter.drawText(textPosX, textPosY, text);
2790 if(GL_BLEND_enabled==GL_TRUE) {
2791 ::glEnable(GL_BLEND);
2792 ::glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
2816 QFont font = QFont();
2817 font.setPointSizeF(size);
2824 const char* textCString = textString.c_str();
2827 QFontMetrics f(font);
2828 G4double span = f.boundingRect(textCString).width();
2843#if QT_VERSION < 0x060000
2852 xmove *= fudgeFactor;
2853 ymove *= fudgeFactor;
2861 QColor color((
int)(c.
GetRed()*255),
2868 font,color,textCString);
2885 const QString& modelShortName = getModelShortName(model);
2887 if (modelShortName ==
"") {
2891 if (fSceneTreeComponentTreeWidget == NULL) {
2892 createSceneTreeComponent();
2896 if (fSceneTreeComponentTreeWidget == NULL) {
2900 fSceneTreeComponentTreeWidget->blockSignals(
true);
2904 if (!fPVRootNodeCreate) {
2907 fModelShortNameItem = createTreeWidgetItem(pPVModel->
GetFullPVPath(),
2915 fPVRootNodeCreate =
true;
2918 bool added = parseAndInsertInSceneTree(fModelShortNameItem,pPVModel,0,modelShortName,0,currentPOIndex);
2922 fSceneTreeComponentTreeWidget->blockSignals(
false);
2931QTreeWidgetItem* G4OpenGLQtViewer::createTreeWidgetItem(
2932 const PVPath& fullPath
2933 ,
const QString& name
2936 ,
const QString& logicalName
2937 ,Qt::CheckState state
2938 ,QTreeWidgetItem * parentTreeNode
2943 if (fullPath.size() > fSceneTreeDepth) {
2944 fSceneTreeDepth = (
unsigned int)fullPath.size();
2946 if (fSceneTreeDepthSlider) {
2947 fSceneTreeDepthSlider->setTickInterval(1000/(fSceneTreeDepth+1));
2950 QTreeWidgetItem * newItem = NULL;
2951 if (parentTreeNode == NULL) {
2952 newItem =
new QTreeWidgetItem();
2953 fSceneTreeComponentTreeWidget->addTopLevelItem(newItem);
2955 newItem =
new QTreeWidgetItem(parentTreeNode);
2956 fSceneTreeComponentTreeWidget->addTopLevelItem(parentTreeNode);
2960 newItem->setText(0,name);
2961 newItem->setData(1,Qt::UserRole,copyNb);
2962 newItem->setText(2,QString::number(POIndex));
2963 newItem->setData(0, Qt::UserRole, POIndex);
2964 newItem->setText(3,logicalName);
2965 newItem->setFlags(newItem->flags()|Qt::ItemIsUserCheckable);
2966 newItem->setCheckState(0,state);
2967 newItem->setExpanded(
true);
2968 updatePositivePoIndexSceneTreeWidgetQuickMap(POIndex,newItem);
2970 changeQColorForTreeWidgetItem(newItem,QColor((
int)(color.
GetRed()*255),
2976 if ((state == Qt::Unchecked) && (POIndex == -1)) {
2977 newItem->setForeground (0, QBrush( Qt::gray) );
2980 newItem->setToolTip (0,QString(
2981 "This node exists in the geometry but has not been\n")+
2982 "drawn, perhaps because it has been set invisible. It \n"+
2983 "cannot be made visible with a click on the button.\n"+
2984 "To see it, change the visibility, for example, with \n"+
2985 "/vis/geometry/set/visibility " + logicalName +
" 0 true\n"+
2986 "and rebuild the view with /vis/viewer/rebuild.\n"+
2987 "Click here will only show/hide all child components");
2990 newItem->setToolTip (0,QString(
"double-click to change the color"));
2996 state = Qt::Unchecked;
2997 newItem->setCheckState(0,state);
2998 updatePositivePoIndexSceneTreeWidgetQuickMap(POIndex,newItem);
3001 fTreeItemModels.insert(std::pair <int, PVPath > (POIndex,fullPath) );
3005 changeOpenCloseVisibleHiddenSelectedColorSceneTreeElement(newItem);
3022bool G4OpenGLQtViewer::parseAndInsertInSceneTree(
3023 QTreeWidgetItem * parentItem
3025 ,
unsigned int fullPathIndex
3026 ,
const QString& parentRoot
3027 ,
unsigned int currentIndexInTreeSceneHandler
3028 ,
int currentPVPOIndex
3031 if (parentItem == NULL) {
3037 std::ostringstream oss;
3038 oss << fullPath.at(fullPathIndex).GetCopyNo();
3039 std::string currentPVName = G4String(fullPath.at(fullPathIndex).GetPhysicalVolume()->GetName()+
" ["+oss.str()+
"]").data();
3041 int currentPVCopyNb = fullPath.at(fullPathIndex).GetCopyNo();
3051 QTreeWidgetItem* subItem = NULL;
3052 QList<QTreeWidgetItem *> parentItemList;
3060 if ((currentIndexInTreeSceneHandler == (fullPath.size()-1)) && ((color.
GetAlpha() == 1.))) {
3062 QString lookForString = QString(currentPVName.c_str());
3063 for (
int i = 0;i < parentItem->childCount(); i++ ) {
3064 if (parentItem->child(i)->text(0) == lookForString) {
3065 parentItemList.push_back(parentItem->child(i));
3070 for (
int i = 0; i < parentItemList.size(); ++i) {
3071 const std::string& parentItemName = parentItemList.at(i)->text(0).toStdString();
3072 int parentItemCopyNb = parentItemList.at(i)->data(1,Qt::UserRole).toInt();
3073 int parentItemPOIndex = parentItemList.at(i)->data(0,Qt::UserRole).toInt();
3081 if (((parentRoot == fTouchableVolumes) && (currentPVCopyNb == parentItemCopyNb)
3082 && (currentPVName == parentItemName)) ||
3084 ((parentRoot != fTouchableVolumes) && (currentPVCopyNb == parentItemCopyNb)
3086 && (currentPVName == parentItemName) && (currentPVPOIndex == parentItemPOIndex) )) {
3089 bool sameTransform =
true;
3090 if (parentItemPOIndex >= 0) {
3091 const PVPath& fullPathTmp = fTreeItemModels[parentItemPOIndex];
3092 if (fullPathTmp.size() > 0) {
3093 if (fullPathTmp.at(fullPathTmp.size()-1).GetTransform () == pPVModel->
GetTransformation ()) {
3094 sameTransform =
true;
3096 sameTransform =
false;
3102 if (sameTransform ==
true) {
3109 if (currentIndexInTreeSceneHandler == (fullPath.size()-1)) {
3111 parentItemList.at(i)->setText(2,QString::number(currentPVPOIndex));
3112 parentItemList.at(i)->setData(0, Qt::UserRole,currentPVPOIndex);
3114 fTreeItemModels.insert(std::pair <int, PVPath >(currentPVPOIndex,fullPath) );
3118 parentItemList.at(i)->setFont (0,f);
3121 parentItemList.at(i)->setForeground (0,QBrush());
3124 parentItemList.at(i)->setToolTip (0,
"");
3126 changeQColorForTreeWidgetItem(parentItemList.at(i),QColor((
int)(color.
GetRed()*255),
3133 parentItemList.at(i)->setCheckState(0,Qt::Checked);
3134 updatePositivePoIndexSceneTreeWidgetQuickMap(currentPVPOIndex,parentItemList.at(i));
3138 subItem = parentItemList.at(i);
3142 }
else if (currentIndexInTreeSceneHandler < (fullPath.size()-1)) {
3143 subItem = parentItemList.at(i);
3150 if (currentIndexInTreeSceneHandler == (fullPath.size()-1)) {
3151 createTreeWidgetItem(fullPath,
3152 QString(currentPVName.c_str()),
3155 QString(fullPath.at(fullPathIndex).GetPhysicalVolume()->GetLogicalVolume()->GetName().data()),
3160 if (currentPVPOIndex > fMaxPOindexInserted) {
3161 fMaxPOindexInserted = currentPVPOIndex;
3167 if (subItem == NULL) {
3169 if (currentIndexInTreeSceneHandler < (fullPath.size()-1)) {
3170 subItem = createTreeWidgetItem(fullPath,
3171 QString(currentPVName.c_str()),
3174 QString(fullPath.at(fullPathIndex).GetPhysicalVolume()->GetLogicalVolume()->GetName().data()),
3181 return parseAndInsertInSceneTree(subItem,pPVModel,fullPathIndex+1,parentRoot,currentIndexInTreeSceneHandler+1,currentPVPOIndex);
3187void G4OpenGLQtViewer::changeOpenCloseVisibleHiddenSelectedColorSceneTreeElement(
3188 QTreeWidgetItem* subItem
3192 QTreeWidgetItem* oldItem = NULL;
3194 QTreeWidgetItem* foundItem = getOldTreeWidgetItem(subItem->data(0,Qt::UserRole).toInt());
3196 if (foundItem != NULL) {
3197 if (isSameSceneTreeElement(foundItem,subItem)) {
3198 oldItem = foundItem;
3201 if (foundItem == NULL) {
3204 std::map <int, QTreeWidgetItem*>::const_iterator i;
3205 i = fOldPositivePoIndexSceneTreeWidgetQuickMap.cbegin();
3206 while (i != fOldPositivePoIndexSceneTreeWidgetQuickMap.cend()) {
3207 if (isSameSceneTreeElement(i->second,subItem)) {
3208 oldItem = i->second;
3209 i = fOldPositivePoIndexSceneTreeWidgetQuickMap.cend();
3215 if (oldItem == NULL) {
3217 while (a < fOldNullPoIndexSceneTreeWidgetQuickVector.size()) {
3218 if (isSameSceneTreeElement(fOldNullPoIndexSceneTreeWidgetQuickVector[a],subItem)) {
3219 oldItem = fOldNullPoIndexSceneTreeWidgetQuickVector[a];
3220 a = fOldNullPoIndexSceneTreeWidgetQuickVector.size();
3229 if (oldItem != NULL) {
3230 subItem->setFlags(oldItem->flags());
3231 subItem->setCheckState(0,oldItem->checkState(0));
3232 subItem->setSelected(oldItem->isSelected());
3233 subItem->setExpanded(oldItem->isExpanded ());
3238 std::map <int, QTreeWidgetItem* >::iterator it;
3241 int oldPOIndex = oldItem->data(0,Qt::UserRole).toInt();
3242 it = fOldPositivePoIndexSceneTreeWidgetQuickMap.find(oldPOIndex);
3246 std::map <int, QColor >::iterator itVis;
3247 itVis = fOldVisAttrColorMap.find(oldPOIndex);
3249 QColor oldVisAttrColor;
3250 const QColor& newVisAttrColor = subItem->data(2,Qt::UserRole).value<QColor>();
3252 bool visAttrChange =
false;
3254 if (itVis != fOldVisAttrColorMap.end()) {
3255 oldVisAttrColor = itVis->second;
3256 if (oldVisAttrColor != newVisAttrColor) {
3257 visAttrChange =
true;
3260 visAttrChange =
true;
3263 if (visAttrChange) {
3264 fOldVisAttrColorMap.insert(std::pair <int, QColor > (subItem->data(0,Qt::UserRole).toInt(),newVisAttrColor) );
3268 if (it != fOldPositivePoIndexSceneTreeWidgetQuickMap.end()) {
3269 color = (it->second)->data(2,Qt::UserRole).value<QColor>();
3271 color = oldItem->data(2,Qt::UserRole).value<QColor>();
3273 changeQColorForTreeWidgetItem(subItem,color);
3285bool G4OpenGLQtViewer::isSameSceneTreeElement(
3286 QTreeWidgetItem* parentOldItem
3287 ,QTreeWidgetItem* parentNewItem
3293 int newCpNumber = -1;
3294 int oldCpNumber = -1;
3296 bool firstWhile =
true;
3298 while ((parentOldItem != NULL) && (parentNewItem != NULL)) {
3302 oldPO = parentOldItem->data(0,Qt::UserRole).toInt();
3303 newPO = parentNewItem->data(0,Qt::UserRole).toInt();
3309 const PVPath& oldFullPath = fOldTreeItemModels[oldPO];
3310 const PVPath& newFullPath = fTreeItemModels[newPO];
3311 if ((oldFullPath.size() > 0) &&
3312 (newFullPath.size() > 0)) {
3313 if (oldFullPath.size() != newFullPath.size()) {
3316 if (oldFullPath.at(oldFullPath.size()-1).GetTransform () == newFullPath.at(newFullPath.size()-1).GetTransform ()) {
3317 newCpNumber = newFullPath.at(newFullPath.size()-1).GetCopyNo();
3318 oldCpNumber = oldFullPath.at(oldFullPath.size()-1).GetCopyNo();
3327 if (oldCpNumber == -1) {
3328 oldCpNumber = parentOldItem->data(1,Qt::UserRole).toInt();
3330 if (newCpNumber == -1) {
3331 newCpNumber = parentNewItem->data(1,Qt::UserRole).toInt();
3333 if ((oldCpNumber != newCpNumber) ||
3335 (parentOldItem->text(0) != parentNewItem->text(0)) ) {
3338 }
else if ((parentOldItem->text(0) != parentNewItem->text(0)) ||
3339 (parentOldItem->text(3) != parentNewItem->text(3))) {
3342 parentOldItem = parentOldItem->parent();
3343 parentNewItem = parentNewItem->parent();
3354 ,
const std::string& modelDescription
3358 QString modelShortName = getModelShortName(model);
3363 const G4Text& g4Text =
dynamic_cast<const G4Text&
>(visible);
3366 catch (
const std::bad_cast&) {
3373 if (g4Marker.
GetInfo() !=
"") {
3374 modelShortName = g4Marker.
GetInfo();
3377 catch (
const std::bad_cast&) {}
3379 if (modelShortName ==
"") {
3383 if (fSceneTreeComponentTreeWidget == NULL) {
3384 createSceneTreeComponent();
3388 if (fSceneTreeComponentTreeWidget == NULL) {
3392 fSceneTreeComponentTreeWidget->blockSignals(
true);
3396 QList<QTreeWidgetItem *> resItem;
3397 resItem = fSceneTreeComponentTreeWidget->findItems (modelShortName, Qt::MatchExactly, 0 );
3398 QTreeWidgetItem * currentItem = NULL;
3399 const PVPath tmpFullPath;
3401 if (resItem.empty()) {
3402 currentItem = createTreeWidgetItem(tmpFullPath,
3411 currentItem = resItem.first();
3415 const QList<QTreeWidgetItem *>&
3416 resItems = fSceneTreeComponentTreeWidget->findItems (QString(modelDescription.c_str()), Qt::MatchFixedString| Qt::MatchCaseSensitive|Qt::MatchRecursive, 0 );
3418 bool alreadyPresent =
false;
3419 for (
int i = 0; i < resItems.size(); ++i) {
3420 if (currentPOIndex == resItems.at(i)->data(0,Qt::UserRole).toInt()) {
3421 alreadyPresent =
true;
3424 if (!alreadyPresent) {
3425 createTreeWidgetItem(tmpFullPath,
3434 fSceneTreeComponentTreeWidget->blockSignals(
false);
3442QString G4OpenGLQtViewer::getModelShortName(
const G4String& model) {
3444 QString modelShortName = model.data();
3445 if (modelShortName.mid(0,modelShortName.indexOf(
" ")) ==
"G4PhysicalVolumeModel") {
3446 modelShortName = fTouchableVolumes;
3448 if (modelShortName.mid(0,2) ==
"G4") {
3449 modelShortName = modelShortName.mid(2);
3451 if (modelShortName.indexOf(
"Model") != -1) {
3452 modelShortName = modelShortName.mid(0,modelShortName.indexOf(
"Model"));
3455 return modelShortName;
3463 if (fSceneTreeComponentTreeWidget == NULL) {
3469 if (fLastSceneTreeWidgetAskForIterator != fLastSceneTreeWidgetAskForIteratorEnd) {
3470 fLastSceneTreeWidgetAskForIterator++;
3472 QTreeWidgetItem* item = getTreeWidgetItem(POindex);
3475 if ( item->checkState(0) == Qt::Checked) {
3483bool G4OpenGLQtViewer::parseAndCheckVisibility(QTreeWidgetItem * treeNode,
int POindex){
3484 bool isFound =
false;
3485 for (
int i = 0; i < treeNode->childCount() ; ++i) {
3487 if (treeNode->child(i)->data(0,Qt::UserRole).toInt() == POindex) {
3488 if (treeNode->child(i)->checkState(0) == Qt::Checked) {
3492 isFound = parseAndCheckVisibility(treeNode->child(i),POindex);
3501std::string G4OpenGLQtViewer::parseSceneTreeAndSaveState(){
3502 std::string commandLine =
"";
3503 for (
int b=0;b<fSceneTreeComponentTreeWidget->topLevelItemCount();b++) {
3504 commandLine += parseSceneTreeElementAndSaveState(fSceneTreeComponentTreeWidget->topLevelItem(b),1)+
"\n";
3506 if (commandLine !=
"") {
3507 commandLine = std::string(
"# Disable auto refresh and quieten vis messages whilst scene and\n") +
3508 "# trajectories are established:\n" +
3509 "/vis/viewer/set/autoRefresh false\n" +
3510 "/vis/verbose errors" +
3512 "# Re-establish auto refreshing and verbosity:\n" +
3513 "/vis/viewer/set/autoRefresh true\n" +
3514 "/vis/verbose confirmations\n";
3520std::string G4OpenGLQtViewer::parseSceneTreeElementAndSaveState(QTreeWidgetItem* item,
unsigned int level){
3522 std::string str( level,
' ' );
3523 std::string commandLine =
"\n#"+ str +
"PV Name: " + item->text(0).toStdString();
3525 if (item->text(3) !=
"") {
3526 commandLine +=
" LV Name: "+item->text(3).toStdString()+
"\n";
3528 commandLine +=
"/vis/geometry/set/visibility " + item->text(3).toStdString() +
" ! ";
3529 if (item->checkState(0) == Qt::Checked) {
3532 if (item->checkState(0) == Qt::Unchecked) {
3538 const QColor& c = item->data(2,Qt::UserRole).value<QColor>();
3539 std::stringstream red;
3540 red << ((double)c.red())/255;
3541 std::stringstream green;
3542 green << (double)c.green()/255;
3543 std::stringstream blue;
3544 blue << ((double)c.blue())/255;
3545 std::stringstream
alpha;
3546 alpha << ((double)c.alpha())/255;
3548 commandLine +=
"/vis/geometry/set/colour " + item->text(3).toStdString() +
" ! " + red.str() +
" " + green.str() +
" " + blue.str() +
" " +
alpha.str()+
"\n";
3551 commandLine +=
"\n";
3555 for (
int b=0;b< item->childCount();b++) {
3556 commandLine += parseSceneTreeElementAndSaveState(item->child(b),level+1);
3563void G4OpenGLQtViewer::sceneTreeComponentItemChanged(QTreeWidgetItem* item,
int) {
3565 if (fCheckSceneTreeComponentSignalLock ==
false) {
3566 fCheckSceneTreeComponentSignalLock =
true;
3568 if (item->checkState(0) == Qt::Checked) {
3571 setCheckComponent(item,checked);
3574 fCheckSceneTreeComponentSignalLock =
false;
3579void G4OpenGLQtViewer::sceneTreeComponentSelected() {
3582void G4OpenGLQtViewer::changeDepthInSceneTree (
int val){
3585 if (fSceneTreeComponentTreeWidget == NULL) {
3596 double depth = 1 + ((double)val)/1000 * ((
double)fSceneTreeDepth+1);
3599 fCheckSceneTreeComponentSignalLock =
true;
3602 G4bool currentAutoRefresh =
fVP.IsAutoRefresh();
3603 fVP.SetAutoRefresh(
false);
3605 for (
int b=0;b<fSceneTreeComponentTreeWidget->topLevelItemCount();b++) {
3606 changeDepthOnSceneTreeItem(depth,1.,fSceneTreeComponentTreeWidget->topLevelItem(b));
3610 fVP.SetAutoRefresh(currentAutoRefresh);
3614 fCheckSceneTreeComponentSignalLock =
false;
3619void G4OpenGLQtViewer::changeColorAndTransparency(QTreeWidgetItem* item,
int) {
3624 const QColor& old = QColor(item->data(2,Qt::UserRole).value<QColor>());
3626 const QColor& color = QColorDialog::getColor(old,
3627 fSceneTreeComponentTreeWidget,
3628 " Get color and transparency",
3629 QColorDialog::ShowAlphaChannel);
3631 if (color.isValid()) {
3633 changeColorAndTransparency(item->data(0,Qt::UserRole).toInt(),
3634 G4Colour (((
G4double)color.red())/255,
3640 changeQColorForTreeWidgetItem(item,color);
3645void G4OpenGLQtViewer::changeColorAndTransparency(GLuint index,
G4Color color) {
3649 if (iPO >= 0 && fTreeItemModels.find(iPO) != fTreeItemModels.end()) {
3650 const PVPath& fullPath = fTreeItemModels[iPO];
3652 if (fullPath.size()) {
3663 QTreeWidgetItem* item = getTreeWidgetItem(poIndex);
3667 const QColor& color = item->data(2,Qt::UserRole).value<QColor>();
3679const std::vector<G4ModelingParameters::VisAttributesModifier>*
3682 static std::vector<G4ModelingParameters::VisAttributesModifier>
3683 privateVisAttributesModifiers;
3685 privateVisAttributesModifiers.clear();
3738 return &privateVisAttributesModifiers;
3742void G4OpenGLQtViewer::changeSearchSelection()
3744 const QString& searchText = fFilterOutput->text();
3745 if (fSceneTreeComponentTreeWidget == NULL) {
3750 for (
int a=0; a<fSceneTreeComponentTreeWidget->topLevelItemCount(); a++) {
3751 fSceneTreeComponentTreeWidget->topLevelItem(a)->setExpanded(
false);
3752 fSceneTreeComponentTreeWidget->topLevelItem(a)->setSelected(
false);
3753 clearSceneTreeSelection(fSceneTreeComponentTreeWidget->topLevelItem(a));
3756 QList<QTreeWidgetItem *> itemList = fSceneTreeComponentTreeWidget->findItems (searchText,Qt::MatchContains | Qt::MatchRecursive,0);
3758 for (
int i = 0; i < itemList.size(); ++i) {
3759 QTreeWidgetItem* expandParentItem = itemList.at(i);
3760 while (expandParentItem->parent() != NULL) {
3761 expandParentItem->parent()->setExpanded(
true);
3762 expandParentItem = expandParentItem->parent();
3764 itemList.at(i)->setSelected(
true);
3770void G4OpenGLQtViewer::clearSceneTreeSelection(QTreeWidgetItem* item) {
3771 for (
int a=0; a<item->childCount(); a++) {
3772 item->child(a)->setSelected(
false);
3773 item->child(a)->setExpanded(
false);
3774 clearSceneTreeSelection(item->child(a));
3780bool G4OpenGLQtViewer::isPVVolume(QTreeWidgetItem* item) {
3781 QTreeWidgetItem* sParent = item;
3782 while (sParent->parent() != NULL) {
3783 sParent = sParent->parent();
3785 if (sParent->text(0) != fTouchableVolumes) {
3789 if (item->text(0) == fTouchableVolumes) {
3796void G4OpenGLQtViewer::changeDepthOnSceneTreeItem(
3798 ,
double currentDepth
3799 ,QTreeWidgetItem* item
3801 double transparencyLevel = 0.;
3806 if (isPVVolume(item)) {
3807 if ((lookForDepth-currentDepth) < 0) {
3808 item->setCheckState(0,Qt::Checked);
3809 updatePositivePoIndexSceneTreeWidgetQuickMap(item->data(0,Qt::UserRole).toInt(),item);
3810 transparencyLevel = 1;
3811 }
else if ((lookForDepth-currentDepth) > 1 ){
3812 item->setCheckState(0,Qt::Unchecked);
3813 updatePositivePoIndexSceneTreeWidgetQuickMap(item->data(0,Qt::UserRole).toInt(),item);
3814 transparencyLevel = 0;
3816 item->setCheckState(0,Qt::Checked);
3817 updatePositivePoIndexSceneTreeWidgetQuickMap(item->data(0,Qt::UserRole).toInt(),item);
3818 transparencyLevel = 1-(lookForDepth-currentDepth);
3822 if (item->data(0,Qt::UserRole).toInt() >= 0) {
3835 if (((color.
GetAlpha()-transparencyLevel) > 0.000001) ||
3836 ((color.
GetAlpha()-transparencyLevel) < -0.000001)) {
3837 if ((item->text(3) !=
"")) {
3843 changeQColorForTreeWidgetItem(item,QColor((
int)(color.
GetRed()*255),
3846 (
int)(transparencyLevel*255)));
3851 for (
int b=0;b< item->childCount();b++) {
3852 changeDepthOnSceneTreeItem(lookForDepth,currentDepth+1,item->child(b));
3860 if (fSceneTreeComponentTreeWidget) {
3862 if (fSceneTreeComponentTreeWidget->topLevelItemCount () > 0) {
3864 fPVRootNodeCreate =
false;
3867 fOldPositivePoIndexSceneTreeWidgetQuickMap.clear();
3868 fOldNullPoIndexSceneTreeWidgetQuickVector.clear();
3869 fOldTreeItemModels.clear();
3872 for (
int b =0; b <fSceneTreeComponentTreeWidget->topLevelItemCount();b++) {
3881 int poIndex = fSceneTreeComponentTreeWidget->topLevelItem(b)->data(0,Qt::UserRole).toInt();
3882 if (poIndex != -1) {
3883 fOldPositivePoIndexSceneTreeWidgetQuickMap.insert(std::pair <int, QTreeWidgetItem*> (poIndex,cloneWidgetItem(fSceneTreeComponentTreeWidget->topLevelItem(b))));
3885 fOldNullPoIndexSceneTreeWidgetQuickVector.push_back(cloneWidgetItem(fSceneTreeComponentTreeWidget->topLevelItem(b)));
3889 cloneSceneTree(fSceneTreeComponentTreeWidget->topLevelItem(b));
3893 fOldTreeItemModels.insert(fTreeItemModels.begin(), fTreeItemModels.end());
3896 int tmp2 = fSceneTreeComponentTreeWidget->topLevelItemCount();
3898 delete fSceneTreeComponentTreeWidget->takeTopLevelItem (0);
3899 tmp2 = fSceneTreeComponentTreeWidget->topLevelItemCount();
3901 fPositivePoIndexSceneTreeWidgetQuickMap.clear();
3904 fOldLastSceneTreeWidgetAskForIterator = fOldPositivePoIndexSceneTreeWidgetQuickMap.begin();
3905 fOldLastSceneTreeWidgetAskForIteratorEnd = fOldPositivePoIndexSceneTreeWidgetQuickMap.end();
3906 fSceneTreeDepth = 1;
3907 fModelShortNameItem = NULL;
3908 fMaxPOindexInserted = -1;
3921QTreeWidgetItem * G4OpenGLQtViewer::cloneWidgetItem(QTreeWidgetItem* item) {
3923 QTreeWidgetItem* cloneItem =
new QTreeWidgetItem();
3927 cloneItem->setText(0,item->text(0));
3928 cloneItem->setData(1,Qt::UserRole,item->data(1,Qt::UserRole).toInt());
3929 cloneItem->setText(2,item->text(2));
3930 cloneItem->setData(0, Qt::UserRole,item->data(0,Qt::UserRole).toInt());
3931 cloneItem->setText(3,item->text(3));
3932 cloneItem->setFlags(item->flags());
3933 cloneItem->setToolTip(0,item->toolTip(0));
3934 cloneItem->setCheckState(0,item->checkState(0));
3935 cloneItem->setSelected(item->isSelected());
3936 cloneItem->setExpanded(item->isExpanded ());
3938 cloneItem->setData(2,Qt::UserRole,item->data(2,Qt::UserRole).value<QColor>());
3947void G4OpenGLQtViewer::cloneSceneTree(
3948 QTreeWidgetItem* rootItem
3951 for (
int b=0;b< rootItem->childCount();b++) {
3953 QTreeWidgetItem *child = rootItem->child(b);
3956 int poIndex = child->data(0,Qt::UserRole).toInt();
3957 if (poIndex != -1) {
3958 fOldPositivePoIndexSceneTreeWidgetQuickMap.insert(std::pair <int, QTreeWidgetItem*> (poIndex,cloneWidgetItem(child)));
3960 fOldNullPoIndexSceneTreeWidgetQuickVector.push_back(cloneWidgetItem(child));
3962 cloneSceneTree(child);
3970 void G4OpenGLQtViewer::updatePositivePoIndexSceneTreeWidgetQuickMap(
int POindex,QTreeWidgetItem* item) {
3973 std::map <int, QTreeWidgetItem*>::iterator i;
3974 i = fPositivePoIndexSceneTreeWidgetQuickMap.find(POindex);
3976 if (i == fPositivePoIndexSceneTreeWidgetQuickMap.end()) {
3977 fPositivePoIndexSceneTreeWidgetQuickMap.insert(std::pair <int, QTreeWidgetItem*> (POindex,item) );
3978 fLastSceneTreeWidgetAskForIterator = fPositivePoIndexSceneTreeWidgetQuickMap.end();
3979 fLastSceneTreeWidgetAskForIteratorEnd = fPositivePoIndexSceneTreeWidgetQuickMap.end();
3987void G4OpenGLQtViewer::changeQColorForTreeWidgetItem(QTreeWidgetItem* item,
const QColor& qc) {
3989 int POIndex = item->data(0,Qt::UserRole).toInt();
3990 updatePositivePoIndexSceneTreeWidgetQuickMap(POIndex,item );
3992 QPixmap pixmap = QPixmap(QSize(16, 16));
3993 if (item->data(0,Qt::UserRole).toInt() != -1) {
3996 pixmap.fill (QColor(255,255,255,255));
3998 QPainter painter(&pixmap);
3999 painter.setPen(Qt::black);
4000 painter.drawRect(0,0,15,15);
4002 item->setIcon(0,pixmap);
4003 item->setData(2,Qt::UserRole,qc);
4012QTreeWidgetItem* G4OpenGLQtViewer::getTreeWidgetItem(
int POindex){
4015 if (POindex == -1) {
4019 if (fPositivePoIndexSceneTreeWidgetQuickMap.size() == 0){
4023 if (fLastSceneTreeWidgetAskForIterator != fLastSceneTreeWidgetAskForIteratorEnd) {
4024 if (POindex == fLastSceneTreeWidgetAskForIterator->first) {
4025 if (fLastSceneTreeWidgetAskForIterator->second != NULL) {
4026 return fLastSceneTreeWidgetAskForIterator->second;
4032 fLastSceneTreeWidgetAskForIterator = fPositivePoIndexSceneTreeWidgetQuickMap.find(POindex);
4033 fLastSceneTreeWidgetAskForIteratorEnd = fPositivePoIndexSceneTreeWidgetQuickMap.end();
4035 if (fLastSceneTreeWidgetAskForIterator != fPositivePoIndexSceneTreeWidgetQuickMap.end()) {
4036 return fLastSceneTreeWidgetAskForIterator->second;
4045QTreeWidgetItem* G4OpenGLQtViewer::getOldTreeWidgetItem(
int POindex){
4049 if (POindex == -1) {
4053 if (fOldPositivePoIndexSceneTreeWidgetQuickMap.size() == 0){
4059 if (fOldLastSceneTreeWidgetAskForIterator != fOldLastSceneTreeWidgetAskForIteratorEnd) {
4060 fOldLastSceneTreeWidgetAskForIterator++;
4063 if (fOldLastSceneTreeWidgetAskForIterator != fOldPositivePoIndexSceneTreeWidgetQuickMap.end()) {
4064 if (POindex == fOldLastSceneTreeWidgetAskForIterator->first) {
4065 if (fOldLastSceneTreeWidgetAskForIterator->second != NULL) {
4066 return fOldLastSceneTreeWidgetAskForIterator->second;
4072 fOldLastSceneTreeWidgetAskForIterator = fOldPositivePoIndexSceneTreeWidgetQuickMap.find(POindex);
4073 fOldLastSceneTreeWidgetAskForIteratorEnd = fOldPositivePoIndexSceneTreeWidgetQuickMap.end();
4075 if (fOldLastSceneTreeWidgetAskForIterator != fOldPositivePoIndexSceneTreeWidgetQuickMap.end()) {
4076 return fOldLastSceneTreeWidgetAskForIterator->second;
4089 if (fUISceneTreeWidget == NULL) {
4092 if (fSceneTreeComponentTreeWidget == NULL) {
4097 fSceneTreeComponentTreeWidget->sortItems (0, Qt::AscendingOrder );
4115 d_style =
fVP.GetDrawingStyle();
4119 if (
fUiQt)
fUiQt->SetIconWireframeSelected();
4121 fDrawingWireframe->setChecked(
true);
4122 fDrawingLineRemoval->setChecked(
false);
4123 fDrawingSurfaceRemoval->setChecked(
false);
4124 fDrawingLineSurfaceRemoval->setChecked(
false);
4129 fDrawingLineRemoval->setChecked(
true);
4130 fDrawingWireframe->setChecked(
false);
4131 fDrawingSurfaceRemoval->setChecked(
false);
4132 fDrawingLineSurfaceRemoval->setChecked(
false);
4137 fDrawingSurfaceRemoval->setChecked(
true);
4138 fDrawingWireframe->setChecked(
false);
4139 fDrawingLineRemoval->setChecked(
false);
4140 fDrawingLineSurfaceRemoval->setChecked(
false);
4145 fDrawingLineSurfaceRemoval->setChecked(
true);
4146 fDrawingWireframe->setChecked(
false);
4147 fDrawingLineRemoval->setChecked(
false);
4148 fDrawingSurfaceRemoval->setChecked(
false);
4149 fDrawingLineSurfaceRemoval->setChecked(
false);
4159 fProjectionOrtho->setChecked(
true);
4160 fProjectionPerspective->setChecked(
false);
4163 if (
fUiQt)
fUiQt->SetIconPerspectiveSelected();
4165 fProjectionPerspective->setChecked(
true);
4166 fProjectionOrtho->setChecked(
false);
4172 if (
fUiQt && fContextMenu) {
4173 if (
fUiQt->IsIconPickSelected()) {
4174 fMousePickAction->setChecked(
true);
4175 fMouseZoomOutAction->setChecked(
false);
4176 fMouseZoomInAction->setChecked(
false);
4177 fMouseRotateAction->setChecked(
false);
4178 fMouseMoveAction->setChecked(
false);
4179 }
else if (
fUiQt->IsIconZoomOutSelected()) {
4180 fMouseZoomOutAction->setChecked(
true);
4181 fMousePickAction->setChecked(
false);
4182 fMouseZoomInAction->setChecked(
false);
4183 fMouseRotateAction->setChecked(
false);
4184 fMouseMoveAction->setChecked(
false);
4185 }
else if (
fUiQt->IsIconZoomInSelected()) {
4186 fMouseZoomInAction->setChecked(
true);
4187 fMousePickAction->setChecked(
false);
4188 fMouseZoomOutAction->setChecked(
false);
4189 fMouseRotateAction->setChecked(
false);
4190 fMouseMoveAction->setChecked(
false);
4191 }
else if (
fUiQt->IsIconRotateSelected()) {
4192 fMouseRotateAction->setChecked(
true);
4193 fMousePickAction->setChecked(
false);
4194 fMouseZoomOutAction->setChecked(
false);
4195 fMouseZoomInAction->setChecked(
false);
4196 fMouseMoveAction->setChecked(
false);
4197 }
else if (
fUiQt->IsIconMoveSelected()) {
4198 fMouseMoveAction->setChecked(
true);
4199 fMousePickAction->setChecked(
false);
4200 fMouseZoomOutAction->setChecked(
false);
4201 fMouseZoomInAction->setChecked(
false);
4202 fMouseRotateAction->setChecked(
false);
4213 if (!fSceneTreeWidget) {
4214 createSceneTreeWidget();
4230 if (!fViewerPropertiesTableWidget) {
4231 createViewerPropertiesWidget();
4233 int treeWidgetInfosIgnoredCommands = 0;
4243 if ((path->
GetCommandEntry()-fTreeWidgetInfosIgnoredCommands) != fViewerPropertiesTableWidget->rowCount()) {
4244 fViewerPropertiesTableWidget->clear();
4247 fViewerPropertiesTableWidget->blockSignals(
true);
4250 fViewerPropertiesTableWidget->setColumnCount (2);
4251 fViewerPropertiesTableWidget->setRowCount (path->
GetCommandEntry()-fTreeWidgetInfosIgnoredCommands);
4252 fViewerPropertiesTableWidget->setHorizontalHeaderLabels(QStringList() << tr(
"Property")
4254 fViewerPropertiesTableWidget->verticalHeader()->setVisible(
false);
4255 fViewerPropertiesTableWidget->setAlternatingRowColors (
true);
4263 QString params =
"";
4266 if (
fVP.IsAutoRefresh()) {
4272 if (
fVP.IsAuxEdgeVisible()) {
4278 params = QString().number(
fVP.GetBackgroundColour().GetRed()) +
" "+
4279 QString().number(
fVP.GetBackgroundColour().GetGreen()) +
" "+
4280 QString().number(
fVP.GetBackgroundColour().GetBlue()) +
" "+
4281 QString().number(
fVP.GetBackgroundColour().GetAlpha());
4284 params = QString().number(
fVP. IsCulling ());
4289 params =
"intersection";
4293 params = QString().number(
fVP.GetDefaultVisAttributes()->GetColor().GetRed()) +
" "+
4294 QString().number(
fVP.GetDefaultVisAttributes()->GetColor().GetGreen()) +
" "+
4295 QString().number(
fVP.GetDefaultVisAttributes()->GetColor().GetBlue()) +
" "+
4296 QString().number(
fVP.GetDefaultVisAttributes()->GetColor().GetAlpha());
4299 params = QString().number(
fVP.GetDefaultTextVisAttributes()->GetColor().GetRed()) +
" "+
4300 QString().number(
fVP.GetDefaultTextVisAttributes()->GetColor().GetGreen()) +
" "+
4301 QString().number(
fVP.GetDefaultTextVisAttributes()->GetColor().GetBlue()) +
" "+
4302 QString().number(
fVP.GetDefaultTextVisAttributes()->GetColor().GetAlpha());
4312 params = QString().number(
fVP.GetExplodeFactor()) +
" " + QString(
G4String(
G4BestUnit(
fVP.GetExplodeFactor(),
"Length")).data());
4314 }
else if(commandTmp->
GetCommandName() ==
"globalLineWidthScale") {
4315 params = QString().number(
fVP.GetGlobalLineWidthScale());
4318 params = QString().number(
fVP.GetGlobalMarkerScale());
4330 if (
fVP.IsMarkerNotHidden()) {
4337 if (
fVP.GetLightsMoveWithCamera()) {
4345 params = QString().number(direction.
theta()/CLHEP::degree)+
" "+ QString().number(direction.
phi()/CLHEP::degree)+
" deg";
4352 params = QString().number(
fVP.GetLightpointDirection().x()) +
" "+
4353 QString().number(
fVP.GetLightpointDirection().y()) +
" "+
4354 QString().number(
fVP.GetLightpointDirection().z());
4356 }
else if(commandTmp->
GetCommandName() ==
"lineSegmentsPerCircle") {
4357 params = QString().number(
fVP.GetNoOfSides());
4360 if (
fVP.IsPicking()) {
4367 if (
fVP.GetFieldHalfAngle() == 0.) {
4368 params =
"orthogonal";
4370 params = QString(
"perspective ") + QString().number(
fVP.GetFieldHalfAngle()/CLHEP::degree) +
" deg";
4375 params =
"constrainUpDirection";
4377 params =
"freeRotation";
4381 if (
fVP.IsSection()) {
4382 params = QString(
"on ") +
4384 QString().number(
fVP.GetSectionPlane().normal().x())
4385 +
" " + QString().number(
fVP.GetSectionPlane().normal().y())
4386 +
" " + QString().number(
fVP.GetSectionPlane().normal().z());
4393 params =
"wireframe";
4408 params = QString().number(up.
theta()/CLHEP::degree)+
" "+ QString().number(up.
phi()/CLHEP::degree)+
" deg";
4416 params = QString().number(up.
x())+
" "+ QString().number(up.
y())+
" "+QString().number(up.
z())+
" ";
4421 params = QString().number(direction.
theta()/CLHEP::degree)+
" "+ QString().number(direction.
phi()/CLHEP::degree)+
" deg";
4429 params = QString().number(direction.
x())+
" "+ QString().number(direction.
y())+
" "+QString().number(direction.
z());
4459 QTableWidgetItem *nameItem;
4460 QTableWidgetItem *paramItem;
4463 QList<QTableWidgetItem *> list = fViewerPropertiesTableWidget->findItems (commandTmp->
GetCommandName().data(),Qt::MatchExactly);
4464 if (list.size() == 1) {
4465 nameItem = list.first();
4466 paramItem = fViewerPropertiesTableWidget->item(nameItem->row(),1);
4469 nameItem =
new QTableWidgetItem();
4470 paramItem =
new QTableWidgetItem();
4471 fViewerPropertiesTableWidget->setItem(a-treeWidgetInfosIgnoredCommands, 0, nameItem);
4472 fViewerPropertiesTableWidget->setItem(a-treeWidgetInfosIgnoredCommands, 1, paramItem);
4477 for(
G4int i_thGuidance=0; i_thGuidance < n_guidanceEntry; i_thGuidance++ ) {
4478 guidance += QString((
char*)(commandTmp->
GetGuidanceLine(i_thGuidance)).data()) +
"\n";
4481 nameItem->setToolTip(guidance);
4482 paramItem->setToolTip(GetCommandParameterList(commandTmp));
4484 fViewerPropertiesTableWidget->setRowHeight(a-treeWidgetInfosIgnoredCommands,15);
4489 paramItem->setText(params);
4491 nameItem->setFlags(Qt::NoItemFlags);
4492 nameItem->setForeground(QBrush());
4495 treeWidgetInfosIgnoredCommands++;
4499 for (
int i=0; i<treeWidgetInfosIgnoredCommands; i++) {
4500 fViewerPropertiesTableWidget->removeRow (fViewerPropertiesTableWidget->rowCount() - 1);
4504 if (!fViewerPropertiesTableWidgetIsInit) {
4505 fViewerPropertiesTableWidgetIsInit =
true;
4507 fViewerPropertiesTableWidget->resizeColumnsToContents();
4509 int x = fViewerPropertiesTableWidget->horizontalHeader()->length();
4510 int y = fViewerPropertiesTableWidget->verticalHeader()->length()+ fViewerPropertiesTableWidget->horizontalHeader()->sizeHint().height() + 2;
4516 QDialog* dial =
static_cast<QDialog*
> (fUIViewerPropertiesWidget->parent());
4518 dial->resize(x+56,y+46);
4521 fViewerPropertiesTableWidget->blockSignals(
false);
4523 fTreeWidgetInfosIgnoredCommands = treeWidgetInfosIgnoredCommands;
4531 fLastPickPoint = QPoint(aX,aY);
4537 if (!fPickInfosWidget) {
4538 createPickInfosWidget();
4541#if QT_VERSION < 0x060000
4544 if (qGLW) qGLW->makeCurrent();}
4547 const std::vector < G4OpenGLViewerPickMap* > & pickMapVector =
GetPickDetails(aX,aY);
4550 if (fPickInfosWidget) {
4551 QLayoutItem * wItem;
4552 if (fPickInfosWidget->layout()->count()) {
4553 while ((wItem = fPickInfosWidget->layout()->takeAt(0)) != 0) {
4554 delete wItem->widget();
4560 if (!fPickInfosWidget) {
4561 createPickInfosWidget();
4566 G4int nPickedObjectsWithAttributes = 0;
4567 for (
unsigned int a=0; a< pickMapVector.size(); a++) {
4568 const auto& pickMap = pickMapVector[a];
4570 std::ostringstream label;
4571 std::ostringstream content;
4572 std::string txt = pickMap->getAttributes()[0].data();
4573 if (pickMapVector[a]->getAttributes().size()) {
4574 ++nPickedObjectsWithAttributes;
4576 std::size_t pos1 = txt.find(
':');
4577 std::string storeKey = txt.substr(0,pos1);
4579 if (storeKey ==
"G4PhysicalVolumeModel") {
4582 std::size_t pos2 = txt.find(
':',pos1+1);
4583 std::size_t pos3 = txt.find(
'\n',pos2+1);
4584 label << txt.substr(pos2+1,pos3-pos2-1);
4586 }
else if (storeKey ==
"G4TrajectoriesModel") {
4588 label <<
"Trajectory:";
4589 std::size_t pos2 = txt.find(
':',pos1+1);
4590 std::size_t pos3 = txt.find(
'\n',pos2+1);
4591 label <<
" Run:" << txt.substr(pos2+1,pos3-pos2-1);
4592 std::size_t pos4 = txt.find(
':',pos3+1);
4593 std::size_t pos5 = txt.find(
'\n',pos4+1);
4594 label <<
", Event:" << txt.substr(pos4+1,pos5-pos4-1);
4598 label <<
"Hit number:" << a <<
", PickName: " << pickMap->getPickName();
4603 content << pickMap->print().data();
4604 G4int thisPickName = pickMap->getPickName();
4605 while (++a < pickMapVector.size()) {
4606 const auto& a_pickMap = pickMapVector[a];
4607 if (a_pickMap->getPickName() == thisPickName) {
4608 content << a_pickMap->print().data();
4615 QPushButton* pickCoutButton =
new QPushButton(label.str().c_str());
4616 pickCoutButton->setStyleSheet (
"text-align: left; padding: 1px; border: 0px;");
4617 pickCoutButton->setIcon(*fTreeIconClosed);
4618 fPickInfosWidget->layout()->addWidget(pickCoutButton);
4623 newStr = QStringList(QString(content.str().c_str()).trimmed());
4625 QTextEdit* ed =
new QTextEdit();
4626 ed->setReadOnly(
true);
4627 fPickInfosWidget->layout()->addWidget(ed);
4628 ed->setVisible((
false));
4629 ed->append(newStr.join(
""));
4631 std::cout << pickCoutButton->text().toStdString() <<
" "<< fPickInfosWidget->layout()->count()-1<< std::endl;
4632 int tmp = fPickInfosWidget->layout()->count()-1;
4633 connect(pickCoutButton, &QPushButton::clicked , [
this, tmp](){ this->toggleSceneTreeComponentPickingCout(tmp);});
4638 QLabel * pushUp =
new QLabel(
"");
4639 QSizePolicy vPolicy = QSizePolicy(QSizePolicy::Minimum,QSizePolicy::Minimum);
4640 vPolicy.setVerticalStretch(10);
4641 pushUp->setSizePolicy(vPolicy);
4642 fPickInfosWidget->layout()->addWidget(pushUp);
4647 changeColorAndTransparency(fLastHighlightName,fLastHighlightColor);
4649 if (pickMapVector.size() > 0 ) {
4651 fLastHighlightName = pickMapVector[0]->getPickName();
4654 changeColorAndTransparency(fLastHighlightName,
G4Color(1,1,1,1));
4658 QDialog* dial =
static_cast<QDialog*
> (fUIPickInfosWidget->parent());
4661 std::ostringstream oss;
4662 if (nPickedObjectsWithAttributes == 0) {
4664 }
else if (nPickedObjectsWithAttributes == 1) {
4667 oss << nPickedObjectsWithAttributes <<
" objects";
4669 oss <<
" selected - " <<
GetName();
4670 dial->setWindowTitle(oss.str().c_str());
4673 fPickInfosScrollArea->setVisible(
true);
4677void G4OpenGLQtViewer::toggleSceneTreeComponentPickingCout(
int pickItem) {
4682 for (
int a=0; a<fPickInfosWidget->layout()->count(); a++) {
4683 w = fPickInfosWidget->layout()->itemAt(a)->widget();
4684 QTextEdit* ed =
dynamic_cast<QTextEdit*
>(w);
4685 QPushButton* button;
4687 if (a == pickItem) {
4688 w->setVisible(!w->isVisible());
4690 w->setVisible(
false);
4693 button =
dynamic_cast<QPushButton*
>(fPickInfosWidget->layout()->itemAt(a-1)->widget());
4695 if (button->isVisible()) {
4696 button->setIcon(*fTreeIconOpen);
4698 button->setIcon(*fTreeIconClosed);
4707void G4OpenGLQtViewer::currentTabActivated(
int currentTab) {
4708 if (
fUiQt->GetViewerTabWidget()->tabText(currentTab) ==
GetName().data()) {
4709 createViewerPropertiesWidget();
4716void G4OpenGLQtViewer::tableWidgetViewerSetItemChanged(QTableWidgetItem * item) {
4719 QTableWidgetItem* previous = fViewerPropertiesTableWidget->item(fViewerPropertiesTableWidget->row(item),0);
4721 fViewerPropertiesTableWidget->blockSignals(
true);
4723 + previous->text().toStdString()
4725 + item->text().toStdString()).c_str());
4726 fViewerPropertiesTableWidget->blockSignals(
false);
4736 if (
GetName() !=
fUiQt->GetViewerTabWidget()->tabText(
fUiQt->GetViewerTabWidget()->currentIndex()).toStdString().c_str()) {
4750QString G4OpenGLQtViewer::GetCommandParameterList (
4757 if( n_parameterEntry > 0 ) {
4762 for(
G4int i_thParameter=0; i_thParameter<n_parameterEntry; i_thParameter++ ) {
4764 txt +=
"\nParameter : " + QString((
char*)(param->
GetParameterName()).data()) +
"\n";
4767 txt +=
" Parameter type : " + QString(QChar(param->
GetParameterType())) +
"\n";
4769 txt +=
" Omittable : True\n";
4771 txt +=
" Omittable : False\n";
4774 txt +=
" Default value : taken from the current value\n";
4776 txt +=
" Default value : " + QString((
char*)(param->
GetDefaultValue()).data())+
"\n";
4779 txt +=
" Parameter range : " + QString((
char*)(param->
GetParameterRange()).data())+
"\n";
4789#ifdef G4MULTITHREADED
4794 G4bool visSubThreadEstablished =
false;
4795 G4bool qObjectsSwitched =
false;
4805 if (qGLWidget ==
nullptr)
return;
4808 qGLWidget->doneCurrent();
4811 fQGLContextMainThread = QThread::currentThread();
4818 if (qGLWidget ==
nullptr)
return;
4827 if(qGLWidget->context()) qGLWidget->context()->moveToThread(fQGLContextVisSubThread);
4831 qObjectsSwitched =
true;
4840 if (qGLWidget ==
nullptr)
return;
4843 fQGLContextVisSubThread = QThread::currentThread();
4848 visSubThreadEstablished =
true;
4859 qGLWidget->makeCurrent();
4866 if (qGLWidget ==
nullptr)
return;
4869 qGLWidget->doneCurrent();
4872 if(qGLWidget->context()) qGLWidget->context()->moveToThread(fQGLContextMainThread);
4879 if (qGLWidget ==
nullptr)
return;
4881 qGLWidget->makeCurrent();
4883 visSubThreadEstablished =
false;
4884 qObjectsSwitched =
false;
G4TemplateAutoLock< G4Mutex > G4AutoLock
G4ThreadLocal T * G4GeomSplitter< T >::offset
QOpenGLWidget G4QGLWidgetType
HepGeom::Point3D< G4double > G4Point3D
#define G4CONDITION_INITIALIZER
#define G4MUTEX_INITIALIZER
#define G4CONDITIONWAITLAMBDA(cond, mutex, lambda)
#define G4CONDITIONBROADCAST(cond)
HepGeom::Vector3D< G4double > G4Vector3D
G4GLOB_DLL std::ostream G4cerr
G4GLOB_DLL std::ostream G4cout
G4double GetAlpha() const
G4double GetGreen() const
void setRecordingInfos(QString)
void G4MouseReleaseEvent(QMouseEvent *evnt)
virtual G4bool ReadyToDraw()
void rotateQtSceneToggle(float, float)
void G4MouseMoveEvent(QMouseEvent *event)
void G4keyPressEvent(QKeyEvent *event)
void updateViewerPropertiesTableWidget()
void addNonPVSceneTreeElement(const G4String &model, int currentPVPOIndex, const std::string &modelDescription, const G4Visible &visible)
void displayRecordingStatus()
QString getSaveFileName()
void G4MousePressEvent(QMouseEvent *event)
void G4wheelEvent(QWheelEvent *event)
void G4MouseDoubleClickEvent()
void updateSceneTreeWidget()
G4OpenGLQtViewer(G4OpenGLSceneHandler &scene)
bool isTouchableVisible(int POindex)
void moveScene(float, float, float, bool)
void G4manageContextMenuEvent(QContextMenuEvent *e)
QString getTempFolderPath()
void rotateQtScene(float, float)
bool exportImage(std::string name="", int width=-1, int height=-1)
void updateToolbarAndMouseContextMenu()
void updateKeyModifierState(const Qt::KeyboardModifiers &)
void G4keyReleaseEvent(QKeyEvent *event)
virtual void CreateMainWindow(G4QGLWidgetType *, const QString &)
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!//
G4Colour getColorForPoIndex(int poIndex)
void updatePickInfosWidget(int, int)
virtual ~G4OpenGLQtViewer()
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!//
void DrawText(const G4Text &)
QString setEncoderPath(QString path)
bool generateMpegEncoderParameters()
virtual void updateQWidget()=0
QString setSaveFileName(QString path)
void addPVSceneTreeElement(const G4String &model, G4PhysicalVolumeModel *pPVModel, int currentPVPOIndex)
const std::vector< G4ModelingParameters::VisAttributesModifier > * GetPrivateVisAttributesModifiers() const
QString setTempFolderPath(QString path)
void displaySceneTreeComponent()
std::vector< std::string > fExportImageFormatVector
friend class G4OpenGLSceneHandler
void rotateSceneToggle(G4double dx, G4double dy)
unsigned int getWinHeight() const
std::string fExportImageFormat
void ResizeWindow(unsigned int, unsigned int)
unsigned int getWinWidth() const
std::string fDefaultExportImageFormat
bool setExportFilename(G4String name, G4bool inc=true)
G4OpenGLViewer(G4OpenGLSceneHandler &scene)
void addExportImageFormat(std::string format)
G4bool antialiasing_enabled
virtual void DrawText(const G4Text &)
void setExportSize(G4int, G4int)
void rotateScene(G4double dx, G4double dy)
G4bool transparency_enabled
virtual bool exportImage(std::string name="", int width=-1, int height=-1)
std::string getRealPrintFilename()
GLdouble getSceneNearWidth()
const std::vector< G4OpenGLViewerPickMap * > & GetPickDetails(GLdouble x, GLdouble y)
const G4Transform3D & GetTransformation() const
const std::vector< G4PhysicalVolumeNodeID > & GetFullPVPath() const
static G4Qt * getInstance()
G4double GetYOffset() const
G4double GetXOffset() const
G4int GetCommandEntry() const
G4UIcommand * GetCommand(G4int i)
G4UIcommandTree * FindCommandTree(const char *commandPath)
std::size_t GetParameterEntries() const
const G4String & GetGuidanceLine(G4int i) const
G4UIparameter * GetParameter(G4int i) const
std::size_t GetGuidanceEntries() const
const G4String & GetCommandName() const
G4UIcommandTree * GetTree() const
G4int ApplyCommand(const char *aCommand)
G4UIsession * GetG4UIWindow() const
static G4UImanager * GetUIpointer()
const G4String & GetParameterCandidates() const
const G4String & GetParameterGuidance() const
G4bool IsOmittable() const
const G4String & GetParameterRange() const
G4bool GetCurrentAsDefault() const
char GetParameterType() const
const G4String & GetParameterName() const
const G4String & GetDefaultValue() const
G4Interactor GetMainInteractor()
G4Point3D GetPosition() const
void SetTouchable(const std::vector< G4PhysicalVolumeModel::G4PhysicalVolumeNodeID > &fullPath)
const G4String & GetName() const
virtual void DoneWithMasterThread()
G4VSceneHandler & fSceneHandler
void SetNeedKernelVisit(G4bool need)
virtual void SwitchToMasterThread()
virtual void SwitchToVisSubThread()
void TouchableSetVisibility(const std::vector< G4PhysicalVolumeModel::G4PhysicalVolumeNodeID > &fullPath, G4bool visibility)
G4VViewer(G4VSceneHandler &, G4int id, const G4String &name="")
void TouchableSetColour(const std::vector< G4PhysicalVolumeModel::G4PhysicalVolumeNodeID > &fullPath, const G4Colour &)
virtual void MovingToVisSubThread()
virtual void DoneWithVisSubThread()
virtual const G4String & GetInfo() const
const char * name(G4int ptype)