123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496 |
- #include "OSGWidget.hpp"
- #include "PickHandler.hpp"
- #include <osg/Camera>
- #include <osg/DisplaySettings>
- #include <osg/Geode>
- #include <osg/Material>
- #include <osg/Shape>
- #include <osg/ShapeDrawable>
- #include <osg/StateSet>
- #include <osgDB/WriteFile>
- #include <osgGA/EventQueue>
- #include <osgGA/TrackballManipulator>
- #include <osgUtil/IntersectionVisitor>
- #include <osgUtil/PolytopeIntersector>
- #include <osgViewer/View>
- #include <osgViewer/ViewerEventHandlers>
- #include <cassert>
- #include <stdexcept>
- #include <vector>
- #include <QDebug>
- #include <QKeyEvent>
- #include <QPainter>
- #include <QWheelEvent>
- namespace
- {
- #ifdef WITH_SELECTION_PROCESSING
- QRect makeRectangle( const QPoint& first, const QPoint& second )
- {
- // Relative to the first point, the second point may be in either one of the
- // four quadrants of an Euclidean coordinate system.
- //
- // We enumerate them in counter-clockwise order, starting from the lower-right
- // quadrant that corresponds to the default case:
- //
- // |
- // (3) | (4)
- // |
- // -------|-------
- // |
- // (2) | (1)
- // |
- if( second.x() >= first.x() && second.y() >= first.y() )
- return QRect( first, second );
- else if( second.x() < first.x() && second.y() >= first.y() )
- return QRect( QPoint( second.x(), first.y() ), QPoint( first.x(), second.y() ) );
- else if( second.x() < first.x() && second.y() < first.y() )
- return QRect( second, first );
- else if( second.x() >= first.x() && second.y() < first.y() )
- return QRect( QPoint( first.x(), second.y() ), QPoint( second.x(), first.y() ) );
- // Should never reach that point...
- return QRect();
- }
- #endif
- }
- namespace osgWidget
- {
- void Viewer::setupThreading()
- {
- if( _threadingModel == SingleThreaded )
- {
- if(_threadsRunning)
- stopThreading();
- }
- else
- {
- if(!_threadsRunning)
- startThreading();
- }
- }
- }
- OSGWidget::OSGWidget( QWidget* parent,
- Qt::WindowFlags f )
- : QOpenGLWidget( parent,
- f )
- , graphicsWindow_( new osgViewer::GraphicsWindowEmbedded( this->x(),
- this->y(),
- this->width(),
- this->height() ) )
- , viewer_( new osgWidget::Viewer )
- , selectionActive_( false )
- , selectionFinished_( true )
- {
- osg::Sphere* sphere = new osg::Sphere( osg::Vec3( 0.f, 0.f, 0.f ), 0.25f );
- osg::ShapeDrawable* sd = new osg::ShapeDrawable( sphere );
- sd->setColor( osg::Vec4( 1.f, 0.f, 0.f, 1.f ) );
- sd->setName( "A nice sphere" );
- osg::Geode* geode = new osg::Geode;
- geode->addDrawable( sd );
- // Set material for basic lighting and enable depth tests. Else, the sphere
- // will suffer from rendering errors.
- {
- osg::StateSet* stateSet = geode->getOrCreateStateSet();
- osg::Material* material = new osg::Material;
- material->setColorMode( osg::Material::AMBIENT_AND_DIFFUSE );
- stateSet->setAttributeAndModes( material, osg::StateAttribute::ON );
- stateSet->setMode( GL_DEPTH_TEST, osg::StateAttribute::ON );
- }
- float aspectRatio = static_cast<float>( this->width() / 2 ) / static_cast<float>( this->height() );
- auto pixelRatio = this->devicePixelRatio();
- osg::Camera* camera = new osg::Camera;
- camera->setViewport( 0, 0, this->width() / 2 * pixelRatio, this->height() * pixelRatio );
- camera->setClearColor( osg::Vec4( 0.f, 0.f, 1.f, 1.f ) );
- camera->setProjectionMatrixAsPerspective( 30.f, aspectRatio, 1.f, 1000.f );
- camera->setGraphicsContext( graphicsWindow_ );
- osgViewer::View* view = new osgViewer::View;
- view->setCamera( camera );
- view->setSceneData( geode );
- view->addEventHandler( new osgViewer::StatsHandler );
- #ifdef WITH_PICK_HANDLER
- view->addEventHandler( new PickHandler( this->devicePixelRatio() ) );
- #endif
- osgGA::TrackballManipulator* manipulator = new osgGA::TrackballManipulator;
- manipulator->setAllowThrow( false );
- view->setCameraManipulator( manipulator );
- osg::Camera* sideCamera = new osg::Camera;
- sideCamera->setViewport( this->width() /2 * pixelRatio, 0,
- this->width() /2 * pixelRatio, this->height() * pixelRatio );
- sideCamera->setClearColor( osg::Vec4( 0.f, 0.f, 1.f, 1.f ) );
- sideCamera->setProjectionMatrixAsPerspective( 30.f, aspectRatio, 1.f, 1000.f );
- sideCamera->setGraphicsContext( graphicsWindow_ );
- osgViewer::View* sideView = new osgViewer::View;
- sideView->setCamera( sideCamera );
- sideView->setSceneData( geode );
- sideView->addEventHandler( new osgViewer::StatsHandler );
- sideView->setCameraManipulator( new osgGA::TrackballManipulator );
- viewer_->addView( view );
- viewer_->addView( sideView );
- viewer_->setThreadingModel( osgViewer::CompositeViewer::SingleThreaded );
- viewer_->realize();
- // This ensures that the widget will receive keyboard events. This focus
- // policy is not set by default. The default, Qt::NoFocus, will result in
- // keyboard events that are ignored.
- this->setFocusPolicy( Qt::StrongFocus );
- this->setMinimumSize( 100, 100 );
- // Ensures that the widget receives mouse move events even though no
- // mouse button has been pressed. We require this in order to let the
- // graphics window switch viewports properly.
- this->setMouseTracking( true );
- }
- OSGWidget::~OSGWidget()
- {
- }
- void OSGWidget::paintEvent( QPaintEvent* /* paintEvent */ )
- {
- this->makeCurrent();
- QPainter painter( this );
- painter.setRenderHint( QPainter::Antialiasing );
- this->paintGL();
- #ifdef WITH_SELECTION_PROCESSING
- if( selectionActive_ && !selectionFinished_ )
- {
- painter.setPen( Qt::black );
- painter.setBrush( Qt::transparent );
- painter.drawRect( makeRectangle( selectionStart_, selectionEnd_ ) );
- }
- #endif
- painter.end();
- this->doneCurrent();
- }
- void OSGWidget::paintGL()
- {
- viewer_->frame();
- }
- void OSGWidget::resizeGL( int width, int height )
- {
- this->getEventQueue()->windowResize( this->x(), this->y(), width, height );
- graphicsWindow_->resized( this->x(), this->y(), width, height );
- this->onResize( width, height );
- }
- void OSGWidget::keyPressEvent( QKeyEvent* event )
- {
- QString keyString = event->text();
- const char* keyData = keyString.toLocal8Bit().data();
- if( event->key() == Qt::Key_S )
- {
- #ifdef WITH_SELECTION_PROCESSING
- selectionActive_ = !selectionActive_;
- #endif
- // Further processing is required for the statistics handler here, so we do
- // not return right away.
- }
- else if( event->key() == Qt::Key_D )
- {
- osgDB::writeNodeFile( *viewer_->getView(0)->getSceneData(),
- "/tmp/sceneGraph.osg" );
- return;
- }
- else if( event->key() == Qt::Key_H )
- {
- this->onHome();
- return;
- }
- this->getEventQueue()->keyPress( osgGA::GUIEventAdapter::KeySymbol( *keyData ) );
- }
- void OSGWidget::keyReleaseEvent( QKeyEvent* event )
- {
- QString keyString = event->text();
- const char* keyData = keyString.toLocal8Bit().data();
- this->getEventQueue()->keyRelease( osgGA::GUIEventAdapter::KeySymbol( *keyData ) );
- }
- void OSGWidget::mouseMoveEvent( QMouseEvent* event )
- {
- // Note that we have to check the buttons mask in order to see whether the
- // left button has been pressed. A call to `button()` will only result in
- // `Qt::NoButton` for mouse move events.
- if( selectionActive_ && event->buttons() & Qt::LeftButton )
- {
- selectionEnd_ = event->pos();
- // Ensures that new paint events are created while the user moves the
- // mouse.
- this->update();
- }
- else
- {
- auto pixelRatio = this->devicePixelRatio();
- this->getEventQueue()->mouseMotion( static_cast<float>( event->x() * pixelRatio ),
- static_cast<float>( event->y() * pixelRatio ) );
- }
- }
- void OSGWidget::mousePressEvent( QMouseEvent* event )
- {
- // Selection processing
- if( selectionActive_ && event->button() == Qt::LeftButton )
- {
- selectionStart_ = event->pos();
- selectionEnd_ = selectionStart_; // Deletes the old selection
- selectionFinished_ = false; // As long as this is set, the rectangle will be drawn
- }
- // Normal processing
- else
- {
- // 1 = left mouse button
- // 2 = middle mouse button
- // 3 = right mouse button
- unsigned int button = 0;
- switch( event->button() )
- {
- case Qt::LeftButton:
- button = 1;
- break;
- case Qt::MiddleButton:
- button = 2;
- break;
- case Qt::RightButton:
- button = 3;
- break;
- default:
- break;
- }
- auto pixelRatio = this->devicePixelRatio();
- this->getEventQueue()->mouseButtonPress( static_cast<float>( event->x() * pixelRatio ),
- static_cast<float>( event->y() * pixelRatio ),
- button );
- }
- }
- void OSGWidget::mouseReleaseEvent(QMouseEvent* event)
- {
- // Selection processing: Store end position and obtain selected objects
- // through polytope intersection.
- if( selectionActive_ && event->button() == Qt::LeftButton )
- {
- selectionEnd_ = event->pos();
- selectionFinished_ = true; // Will force the painter to stop drawing the
- // selection rectangle
- this->processSelection();
- }
- // Normal processing
- else
- {
- // 1 = left mouse button
- // 2 = middle mouse button
- // 3 = right mouse button
- unsigned int button = 0;
- switch( event->button() )
- {
- case Qt::LeftButton:
- button = 1;
- break;
- case Qt::MiddleButton:
- button = 2;
- break;
- case Qt::RightButton:
- button = 3;
- break;
- default:
- break;
- }
- auto pixelRatio = this->devicePixelRatio();
- this->getEventQueue()->mouseButtonRelease( static_cast<float>( pixelRatio * event->x() ),
- static_cast<float>( pixelRatio * event->y() ),
- button );
- }
- }
- void OSGWidget::wheelEvent( QWheelEvent* event )
- {
- // Ignore wheel events as long as the selection is active.
- if( selectionActive_ )
- return;
- event->accept();
- int delta = event->angleDelta().y();
- osgGA::GUIEventAdapter::ScrollingMotion motion = delta > 0 ? osgGA::GUIEventAdapter::SCROLL_UP
- : osgGA::GUIEventAdapter::SCROLL_DOWN;
- this->getEventQueue()->mouseScroll( motion );
- }
- bool OSGWidget::event( QEvent* event )
- {
- bool handled = QOpenGLWidget::event( event );
- // This ensures that the OSG widget is always going to be repainted after the
- // user performed some interaction. Doing this in the event handler ensures
- // that we don't forget about some event and prevents duplicate code.
- switch( event->type() )
- {
- case QEvent::KeyPress:
- case QEvent::KeyRelease:
- case QEvent::MouseButtonDblClick:
- case QEvent::MouseButtonPress:
- case QEvent::MouseButtonRelease:
- case QEvent::MouseMove:
- case QEvent::Wheel:
- this->update();
- break;
- default:
- break;
- }
- return handled;
- }
- void OSGWidget::onHome()
- {
- osgViewer::ViewerBase::Views views;
- viewer_->getViews( views );
- for( std::size_t i = 0; i < views.size(); i++ )
- {
- osgViewer::View* view = views.at(i);
- view->home();
- }
- }
- void OSGWidget::onResize( int width, int height )
- {
- std::vector<osg::Camera*> cameras;
- viewer_->getCameras( cameras );
- assert( cameras.size() == 2 );
- auto pixelRatio = this->devicePixelRatio();
- cameras[0]->setViewport( 0, 0, width / 2 * pixelRatio, height * pixelRatio );
- cameras[1]->setViewport( width / 2 * pixelRatio, 0, width / 2 * pixelRatio, height * pixelRatio );
- }
- osgGA::EventQueue* OSGWidget::getEventQueue() const
- {
- osgGA::EventQueue* eventQueue = graphicsWindow_->getEventQueue();
- if( eventQueue )
- return eventQueue;
- else
- throw std::runtime_error( "Unable to obtain valid event queue");
- }
- void OSGWidget::processSelection()
- {
- #ifdef WITH_SELECTION_PROCESSING
- QRect selectionRectangle = makeRectangle( selectionStart_, selectionEnd_ );
- auto widgetHeight = this->height();
- auto pixelRatio = this->devicePixelRatio();
- double xMin = selectionRectangle.left();
- double xMax = selectionRectangle.right();
- double yMin = widgetHeight - selectionRectangle.bottom();
- double yMax = widgetHeight - selectionRectangle.top();
- xMin *= pixelRatio;
- yMin *= pixelRatio;
- xMax *= pixelRatio;
- yMax *= pixelRatio;
- osgUtil::PolytopeIntersector* polytopeIntersector
- = new osgUtil::PolytopeIntersector( osgUtil::PolytopeIntersector::WINDOW,
- xMin, yMin,
- xMax, yMax );
- // This limits the amount of intersections that are reported by the
- // polytope intersector. Using this setting, a single drawable will
- // appear at most once while calculating intersections. This is the
- // preferred and expected behaviour.
- polytopeIntersector->setIntersectionLimit( osgUtil::Intersector::LIMIT_ONE_PER_DRAWABLE );
- osgUtil::IntersectionVisitor iv( polytopeIntersector );
- for( unsigned int viewIndex = 0; viewIndex < viewer_->getNumViews(); viewIndex++ )
- {
- qDebug() << "View index:" << viewIndex;
- osgViewer::View* view = viewer_->getView( viewIndex );
- if( !view )
- throw std::runtime_error( "Unable to obtain valid view for selection processing" );
- osg::Camera* camera = view->getCamera();
- if( !camera )
- throw std::runtime_error( "Unable to obtain valid camera for selection processing" );
- camera->accept( iv );
- if( !polytopeIntersector->containsIntersections() )
- continue;
- auto intersections = polytopeIntersector->getIntersections();
- for( auto&& intersection : intersections )
- qDebug() << "Selected a drawable:" << QString::fromStdString( intersection.drawable->getName() );
- }
- #endif
- }
|