OSGWidget.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496
  1. #include "OSGWidget.hpp"
  2. #include "PickHandler.hpp"
  3. #include <osg/Camera>
  4. #include <osg/DisplaySettings>
  5. #include <osg/Geode>
  6. #include <osg/Material>
  7. #include <osg/Shape>
  8. #include <osg/ShapeDrawable>
  9. #include <osg/StateSet>
  10. #include <osgDB/WriteFile>
  11. #include <osgGA/EventQueue>
  12. #include <osgGA/TrackballManipulator>
  13. #include <osgUtil/IntersectionVisitor>
  14. #include <osgUtil/PolytopeIntersector>
  15. #include <osgViewer/View>
  16. #include <osgViewer/ViewerEventHandlers>
  17. #include <cassert>
  18. #include <stdexcept>
  19. #include <vector>
  20. #include <QDebug>
  21. #include <QKeyEvent>
  22. #include <QPainter>
  23. #include <QWheelEvent>
  24. namespace
  25. {
  26. #ifdef WITH_SELECTION_PROCESSING
  27. QRect makeRectangle( const QPoint& first, const QPoint& second )
  28. {
  29. // Relative to the first point, the second point may be in either one of the
  30. // four quadrants of an Euclidean coordinate system.
  31. //
  32. // We enumerate them in counter-clockwise order, starting from the lower-right
  33. // quadrant that corresponds to the default case:
  34. //
  35. // |
  36. // (3) | (4)
  37. // |
  38. // -------|-------
  39. // |
  40. // (2) | (1)
  41. // |
  42. if( second.x() >= first.x() && second.y() >= first.y() )
  43. return QRect( first, second );
  44. else if( second.x() < first.x() && second.y() >= first.y() )
  45. return QRect( QPoint( second.x(), first.y() ), QPoint( first.x(), second.y() ) );
  46. else if( second.x() < first.x() && second.y() < first.y() )
  47. return QRect( second, first );
  48. else if( second.x() >= first.x() && second.y() < first.y() )
  49. return QRect( QPoint( first.x(), second.y() ), QPoint( second.x(), first.y() ) );
  50. // Should never reach that point...
  51. return QRect();
  52. }
  53. #endif
  54. }
  55. namespace osgWidget
  56. {
  57. void Viewer::setupThreading()
  58. {
  59. if( _threadingModel == SingleThreaded )
  60. {
  61. if(_threadsRunning)
  62. stopThreading();
  63. }
  64. else
  65. {
  66. if(!_threadsRunning)
  67. startThreading();
  68. }
  69. }
  70. }
  71. OSGWidget::OSGWidget( QWidget* parent,
  72. Qt::WindowFlags f )
  73. : QOpenGLWidget( parent,
  74. f )
  75. , graphicsWindow_( new osgViewer::GraphicsWindowEmbedded( this->x(),
  76. this->y(),
  77. this->width(),
  78. this->height() ) )
  79. , viewer_( new osgWidget::Viewer )
  80. , selectionActive_( false )
  81. , selectionFinished_( true )
  82. {
  83. osg::Sphere* sphere = new osg::Sphere( osg::Vec3( 0.f, 0.f, 0.f ), 0.25f );
  84. osg::ShapeDrawable* sd = new osg::ShapeDrawable( sphere );
  85. sd->setColor( osg::Vec4( 1.f, 0.f, 0.f, 1.f ) );
  86. sd->setName( "A nice sphere" );
  87. osg::Geode* geode = new osg::Geode;
  88. geode->addDrawable( sd );
  89. // Set material for basic lighting and enable depth tests. Else, the sphere
  90. // will suffer from rendering errors.
  91. {
  92. osg::StateSet* stateSet = geode->getOrCreateStateSet();
  93. osg::Material* material = new osg::Material;
  94. material->setColorMode( osg::Material::AMBIENT_AND_DIFFUSE );
  95. stateSet->setAttributeAndModes( material, osg::StateAttribute::ON );
  96. stateSet->setMode( GL_DEPTH_TEST, osg::StateAttribute::ON );
  97. }
  98. float aspectRatio = static_cast<float>( this->width() / 2 ) / static_cast<float>( this->height() );
  99. auto pixelRatio = this->devicePixelRatio();
  100. osg::Camera* camera = new osg::Camera;
  101. camera->setViewport( 0, 0, this->width() / 2 * pixelRatio, this->height() * pixelRatio );
  102. camera->setClearColor( osg::Vec4( 0.f, 0.f, 1.f, 1.f ) );
  103. camera->setProjectionMatrixAsPerspective( 30.f, aspectRatio, 1.f, 1000.f );
  104. camera->setGraphicsContext( graphicsWindow_ );
  105. osgViewer::View* view = new osgViewer::View;
  106. view->setCamera( camera );
  107. view->setSceneData( geode );
  108. view->addEventHandler( new osgViewer::StatsHandler );
  109. #ifdef WITH_PICK_HANDLER
  110. view->addEventHandler( new PickHandler( this->devicePixelRatio() ) );
  111. #endif
  112. osgGA::TrackballManipulator* manipulator = new osgGA::TrackballManipulator;
  113. manipulator->setAllowThrow( false );
  114. view->setCameraManipulator( manipulator );
  115. osg::Camera* sideCamera = new osg::Camera;
  116. sideCamera->setViewport( this->width() /2 * pixelRatio, 0,
  117. this->width() /2 * pixelRatio, this->height() * pixelRatio );
  118. sideCamera->setClearColor( osg::Vec4( 0.f, 0.f, 1.f, 1.f ) );
  119. sideCamera->setProjectionMatrixAsPerspective( 30.f, aspectRatio, 1.f, 1000.f );
  120. sideCamera->setGraphicsContext( graphicsWindow_ );
  121. osgViewer::View* sideView = new osgViewer::View;
  122. sideView->setCamera( sideCamera );
  123. sideView->setSceneData( geode );
  124. sideView->addEventHandler( new osgViewer::StatsHandler );
  125. sideView->setCameraManipulator( new osgGA::TrackballManipulator );
  126. viewer_->addView( view );
  127. viewer_->addView( sideView );
  128. viewer_->setThreadingModel( osgViewer::CompositeViewer::SingleThreaded );
  129. viewer_->realize();
  130. // This ensures that the widget will receive keyboard events. This focus
  131. // policy is not set by default. The default, Qt::NoFocus, will result in
  132. // keyboard events that are ignored.
  133. this->setFocusPolicy( Qt::StrongFocus );
  134. this->setMinimumSize( 100, 100 );
  135. // Ensures that the widget receives mouse move events even though no
  136. // mouse button has been pressed. We require this in order to let the
  137. // graphics window switch viewports properly.
  138. this->setMouseTracking( true );
  139. }
  140. OSGWidget::~OSGWidget()
  141. {
  142. }
  143. void OSGWidget::paintEvent( QPaintEvent* /* paintEvent */ )
  144. {
  145. this->makeCurrent();
  146. QPainter painter( this );
  147. painter.setRenderHint( QPainter::Antialiasing );
  148. this->paintGL();
  149. #ifdef WITH_SELECTION_PROCESSING
  150. if( selectionActive_ && !selectionFinished_ )
  151. {
  152. painter.setPen( Qt::black );
  153. painter.setBrush( Qt::transparent );
  154. painter.drawRect( makeRectangle( selectionStart_, selectionEnd_ ) );
  155. }
  156. #endif
  157. painter.end();
  158. this->doneCurrent();
  159. }
  160. void OSGWidget::paintGL()
  161. {
  162. viewer_->frame();
  163. }
  164. void OSGWidget::resizeGL( int width, int height )
  165. {
  166. this->getEventQueue()->windowResize( this->x(), this->y(), width, height );
  167. graphicsWindow_->resized( this->x(), this->y(), width, height );
  168. this->onResize( width, height );
  169. }
  170. void OSGWidget::keyPressEvent( QKeyEvent* event )
  171. {
  172. QString keyString = event->text();
  173. const char* keyData = keyString.toLocal8Bit().data();
  174. if( event->key() == Qt::Key_S )
  175. {
  176. #ifdef WITH_SELECTION_PROCESSING
  177. selectionActive_ = !selectionActive_;
  178. #endif
  179. // Further processing is required for the statistics handler here, so we do
  180. // not return right away.
  181. }
  182. else if( event->key() == Qt::Key_D )
  183. {
  184. osgDB::writeNodeFile( *viewer_->getView(0)->getSceneData(),
  185. "/tmp/sceneGraph.osg" );
  186. return;
  187. }
  188. else if( event->key() == Qt::Key_H )
  189. {
  190. this->onHome();
  191. return;
  192. }
  193. this->getEventQueue()->keyPress( osgGA::GUIEventAdapter::KeySymbol( *keyData ) );
  194. }
  195. void OSGWidget::keyReleaseEvent( QKeyEvent* event )
  196. {
  197. QString keyString = event->text();
  198. const char* keyData = keyString.toLocal8Bit().data();
  199. this->getEventQueue()->keyRelease( osgGA::GUIEventAdapter::KeySymbol( *keyData ) );
  200. }
  201. void OSGWidget::mouseMoveEvent( QMouseEvent* event )
  202. {
  203. // Note that we have to check the buttons mask in order to see whether the
  204. // left button has been pressed. A call to `button()` will only result in
  205. // `Qt::NoButton` for mouse move events.
  206. if( selectionActive_ && event->buttons() & Qt::LeftButton )
  207. {
  208. selectionEnd_ = event->pos();
  209. // Ensures that new paint events are created while the user moves the
  210. // mouse.
  211. this->update();
  212. }
  213. else
  214. {
  215. auto pixelRatio = this->devicePixelRatio();
  216. this->getEventQueue()->mouseMotion( static_cast<float>( event->x() * pixelRatio ),
  217. static_cast<float>( event->y() * pixelRatio ) );
  218. }
  219. }
  220. void OSGWidget::mousePressEvent( QMouseEvent* event )
  221. {
  222. // Selection processing
  223. if( selectionActive_ && event->button() == Qt::LeftButton )
  224. {
  225. selectionStart_ = event->pos();
  226. selectionEnd_ = selectionStart_; // Deletes the old selection
  227. selectionFinished_ = false; // As long as this is set, the rectangle will be drawn
  228. }
  229. // Normal processing
  230. else
  231. {
  232. // 1 = left mouse button
  233. // 2 = middle mouse button
  234. // 3 = right mouse button
  235. unsigned int button = 0;
  236. switch( event->button() )
  237. {
  238. case Qt::LeftButton:
  239. button = 1;
  240. break;
  241. case Qt::MiddleButton:
  242. button = 2;
  243. break;
  244. case Qt::RightButton:
  245. button = 3;
  246. break;
  247. default:
  248. break;
  249. }
  250. auto pixelRatio = this->devicePixelRatio();
  251. this->getEventQueue()->mouseButtonPress( static_cast<float>( event->x() * pixelRatio ),
  252. static_cast<float>( event->y() * pixelRatio ),
  253. button );
  254. }
  255. }
  256. void OSGWidget::mouseReleaseEvent(QMouseEvent* event)
  257. {
  258. // Selection processing: Store end position and obtain selected objects
  259. // through polytope intersection.
  260. if( selectionActive_ && event->button() == Qt::LeftButton )
  261. {
  262. selectionEnd_ = event->pos();
  263. selectionFinished_ = true; // Will force the painter to stop drawing the
  264. // selection rectangle
  265. this->processSelection();
  266. }
  267. // Normal processing
  268. else
  269. {
  270. // 1 = left mouse button
  271. // 2 = middle mouse button
  272. // 3 = right mouse button
  273. unsigned int button = 0;
  274. switch( event->button() )
  275. {
  276. case Qt::LeftButton:
  277. button = 1;
  278. break;
  279. case Qt::MiddleButton:
  280. button = 2;
  281. break;
  282. case Qt::RightButton:
  283. button = 3;
  284. break;
  285. default:
  286. break;
  287. }
  288. auto pixelRatio = this->devicePixelRatio();
  289. this->getEventQueue()->mouseButtonRelease( static_cast<float>( pixelRatio * event->x() ),
  290. static_cast<float>( pixelRatio * event->y() ),
  291. button );
  292. }
  293. }
  294. void OSGWidget::wheelEvent( QWheelEvent* event )
  295. {
  296. // Ignore wheel events as long as the selection is active.
  297. if( selectionActive_ )
  298. return;
  299. event->accept();
  300. int delta = event->angleDelta().y();
  301. osgGA::GUIEventAdapter::ScrollingMotion motion = delta > 0 ? osgGA::GUIEventAdapter::SCROLL_UP
  302. : osgGA::GUIEventAdapter::SCROLL_DOWN;
  303. this->getEventQueue()->mouseScroll( motion );
  304. }
  305. bool OSGWidget::event( QEvent* event )
  306. {
  307. bool handled = QOpenGLWidget::event( event );
  308. // This ensures that the OSG widget is always going to be repainted after the
  309. // user performed some interaction. Doing this in the event handler ensures
  310. // that we don't forget about some event and prevents duplicate code.
  311. switch( event->type() )
  312. {
  313. case QEvent::KeyPress:
  314. case QEvent::KeyRelease:
  315. case QEvent::MouseButtonDblClick:
  316. case QEvent::MouseButtonPress:
  317. case QEvent::MouseButtonRelease:
  318. case QEvent::MouseMove:
  319. case QEvent::Wheel:
  320. this->update();
  321. break;
  322. default:
  323. break;
  324. }
  325. return handled;
  326. }
  327. void OSGWidget::onHome()
  328. {
  329. osgViewer::ViewerBase::Views views;
  330. viewer_->getViews( views );
  331. for( std::size_t i = 0; i < views.size(); i++ )
  332. {
  333. osgViewer::View* view = views.at(i);
  334. view->home();
  335. }
  336. }
  337. void OSGWidget::onResize( int width, int height )
  338. {
  339. std::vector<osg::Camera*> cameras;
  340. viewer_->getCameras( cameras );
  341. assert( cameras.size() == 2 );
  342. auto pixelRatio = this->devicePixelRatio();
  343. cameras[0]->setViewport( 0, 0, width / 2 * pixelRatio, height * pixelRatio );
  344. cameras[1]->setViewport( width / 2 * pixelRatio, 0, width / 2 * pixelRatio, height * pixelRatio );
  345. }
  346. osgGA::EventQueue* OSGWidget::getEventQueue() const
  347. {
  348. osgGA::EventQueue* eventQueue = graphicsWindow_->getEventQueue();
  349. if( eventQueue )
  350. return eventQueue;
  351. else
  352. throw std::runtime_error( "Unable to obtain valid event queue");
  353. }
  354. void OSGWidget::processSelection()
  355. {
  356. #ifdef WITH_SELECTION_PROCESSING
  357. QRect selectionRectangle = makeRectangle( selectionStart_, selectionEnd_ );
  358. auto widgetHeight = this->height();
  359. auto pixelRatio = this->devicePixelRatio();
  360. double xMin = selectionRectangle.left();
  361. double xMax = selectionRectangle.right();
  362. double yMin = widgetHeight - selectionRectangle.bottom();
  363. double yMax = widgetHeight - selectionRectangle.top();
  364. xMin *= pixelRatio;
  365. yMin *= pixelRatio;
  366. xMax *= pixelRatio;
  367. yMax *= pixelRatio;
  368. osgUtil::PolytopeIntersector* polytopeIntersector
  369. = new osgUtil::PolytopeIntersector( osgUtil::PolytopeIntersector::WINDOW,
  370. xMin, yMin,
  371. xMax, yMax );
  372. // This limits the amount of intersections that are reported by the
  373. // polytope intersector. Using this setting, a single drawable will
  374. // appear at most once while calculating intersections. This is the
  375. // preferred and expected behaviour.
  376. polytopeIntersector->setIntersectionLimit( osgUtil::Intersector::LIMIT_ONE_PER_DRAWABLE );
  377. osgUtil::IntersectionVisitor iv( polytopeIntersector );
  378. for( unsigned int viewIndex = 0; viewIndex < viewer_->getNumViews(); viewIndex++ )
  379. {
  380. qDebug() << "View index:" << viewIndex;
  381. osgViewer::View* view = viewer_->getView( viewIndex );
  382. if( !view )
  383. throw std::runtime_error( "Unable to obtain valid view for selection processing" );
  384. osg::Camera* camera = view->getCamera();
  385. if( !camera )
  386. throw std::runtime_error( "Unable to obtain valid camera for selection processing" );
  387. camera->accept( iv );
  388. if( !polytopeIntersector->containsIntersections() )
  389. continue;
  390. auto intersections = polytopeIntersector->getIntersections();
  391. for( auto&& intersection : intersections )
  392. qDebug() << "Selected a drawable:" << QString::fromStdString( intersection.drawable->getName() );
  393. }
  394. #endif
  395. }