mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 13:56:41 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			615 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			615 lines
		
	
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*-c++-*- OpenSceneGraph - Copyright (C) 2009 Wang Rui
 | |
|  *
 | |
|  * This library is open source and may be redistributed and/or modified under
 | |
|  * the terms of the OpenSceneGraph Public License (OSGPL) version 0.0 or
 | |
|  * (at your option) any later version.  The full license is in LICENSE file
 | |
|  * included with this distribution, and on the openscenegraph.org website.
 | |
|  *
 | |
|  * This library is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * OpenSceneGraph Public License for more details.
 | |
| */
 | |
| 
 | |
| #include "GraphicsWindowQt"
 | |
| 
 | |
| #include <osg/DeleteHandler>
 | |
| #include <osgViewer/ViewerBase>
 | |
| #include <QInputEvent>
 | |
| #include <QPointer>
 | |
| #include <QWindow>
 | |
| 
 | |
| using namespace osgQt;
 | |
| 
 | |
| #define GETDEVICEPIXELRATIO() devicePixelRatio()
 | |
| 
 | |
| GLWidget::GLWidget( QWidget* parent, const QGLWidget* shareWidget, Qt::WindowFlags f)
 | |
| : QGLWidget(parent, shareWidget, f), _gw( nullptr )
 | |
| {
 | |
|     _devicePixelRatio = GETDEVICEPIXELRATIO();
 | |
| }
 | |
| 
 | |
| GLWidget::GLWidget( QGLContext* context, QWidget* parent, const QGLWidget* shareWidget, Qt::WindowFlags f)
 | |
| : QGLWidget(context, parent, shareWidget, f), _gw( nullptr )
 | |
| {
 | |
|     _devicePixelRatio = GETDEVICEPIXELRATIO();
 | |
| }
 | |
| 
 | |
| GLWidget::GLWidget( const QGLFormat& format, QWidget* parent, const QGLWidget* shareWidget, Qt::WindowFlags f)
 | |
| : QGLWidget(format, parent, shareWidget, f), _gw( nullptr )
 | |
| {
 | |
|     _devicePixelRatio = GETDEVICEPIXELRATIO();
 | |
| }
 | |
| 
 | |
| GLWidget::~GLWidget()
 | |
| {
 | |
|     // close GraphicsWindowQt and remove the reference to us
 | |
|     if( _gw )
 | |
|     {
 | |
|         _gw->close();
 | |
|         _gw->_widget = nullptr;
 | |
|         _gw = nullptr;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void GLWidget::processDeferredEvents()
 | |
| {
 | |
|     QQueue<QEvent::Type> deferredEventQueueCopy;
 | |
|     {
 | |
|         QMutexLocker lock(&_deferredEventQueueMutex);
 | |
|         deferredEventQueueCopy = _deferredEventQueue;
 | |
|         _eventCompressor.clear();
 | |
|         _deferredEventQueue.clear();
 | |
|     }
 | |
| 
 | |
|     while (!deferredEventQueueCopy.isEmpty())
 | |
|     {
 | |
|         QEvent event(deferredEventQueueCopy.dequeue());
 | |
|         QGLWidget::event(&event);
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool GLWidget::event( QEvent* event )
 | |
| {
 | |
| 
 | |
|     // QEvent::Hide
 | |
|     //
 | |
|     // workaround "Qt-workaround" that does glFinish before hiding the widget
 | |
|     // (the Qt workaround was seen at least in Qt 4.6.3 and 4.7.0)
 | |
|     //
 | |
|     // Qt makes the context current, performs glFinish, and releases the context.
 | |
|     // This makes the problem in OSG multithreaded environment as the context
 | |
|     // is active in another thread, thus it can not be made current for the purpose
 | |
|     // of glFinish in this thread.
 | |
| 
 | |
|     // QEvent::ParentChange
 | |
|     //
 | |
|     // Reparenting GLWidget may create a new underlying window and a new GL context.
 | |
|     // Qt will then call doneCurrent on the GL context about to be deleted. The thread
 | |
|     // where old GL context was current has no longer current context to render to and
 | |
|     // we cannot make new GL context current in this thread.
 | |
| 
 | |
|     // We workaround above problems by deferring execution of problematic event requests.
 | |
|     // These events has to be enqueue and executed later in a main GUI thread (GUI operations
 | |
|     // outside the main thread are not allowed) just before makeCurrent is called from the
 | |
|     // right thread. The good place for doing that is right after swap in a swapBuffersImplementation.
 | |
| 
 | |
|     if (event->type() == QEvent::Hide)
 | |
|     {
 | |
|         // enqueue only the last of QEvent::Hide and QEvent::Show
 | |
|         enqueueDeferredEvent(QEvent::Hide, QEvent::Show);
 | |
|         return true;
 | |
|     }
 | |
|     else if (event->type() == QEvent::Show)
 | |
|     {
 | |
|         // enqueue only the last of QEvent::Show or QEvent::Hide
 | |
|         enqueueDeferredEvent(QEvent::Show, QEvent::Hide);
 | |
|         return true;
 | |
|     }
 | |
|     else if (event->type() == QEvent::ParentChange)
 | |
|     {
 | |
|         // enqueue only the last QEvent::ParentChange
 | |
|         enqueueDeferredEvent(QEvent::ParentChange);
 | |
|         return true;
 | |
|     }
 | |
|     else if (event->type() == QEvent::PlatformSurface && static_cast<QPlatformSurfaceEvent*>(event)->surfaceEventType() == QPlatformSurfaceEvent::SurfaceAboutToBeDestroyed)
 | |
|     {
 | |
|         if (_gw)
 | |
|             _gw->close();
 | |
|     }
 | |
| 
 | |
|     // perform regular event handling
 | |
|     return QGLWidget::event( event );
 | |
| }
 | |
| 
 | |
| void GLWidget::resizeEvent( QResizeEvent* event )
 | |
| {
 | |
|     if (_gw == nullptr || !_gw->valid())
 | |
|         return;
 | |
|     const QSize& size = event->size();
 | |
| 
 | |
|     int scaled_width = static_cast<int>(size.width()*_devicePixelRatio);
 | |
|     int scaled_height = static_cast<int>(size.height()*_devicePixelRatio);
 | |
|     _gw->resized( x(), y(), scaled_width,  scaled_height);
 | |
|     _gw->getEventQueue()->windowResize( x(), y(), scaled_width, scaled_height );
 | |
|     _gw->requestRedraw();
 | |
| }
 | |
| 
 | |
| void GLWidget::moveEvent( QMoveEvent* event )
 | |
| {
 | |
|     if (_gw == nullptr || !_gw->valid())
 | |
|         return;
 | |
|     const QPoint& pos = event->pos();
 | |
|     int scaled_width = static_cast<int>(width()*_devicePixelRatio);
 | |
|     int scaled_height = static_cast<int>(height()*_devicePixelRatio);
 | |
|     _gw->resized( pos.x(), pos.y(), scaled_width,  scaled_height );
 | |
|     _gw->getEventQueue()->windowResize( pos.x(), pos.y(), scaled_width,  scaled_height );
 | |
| }
 | |
| 
 | |
| void GLWidget::glDraw()
 | |
| {
 | |
|     _gw->requestRedraw();
 | |
| }
 | |
| 
 | |
| GraphicsWindowQt::GraphicsWindowQt( osg::GraphicsContext::Traits* traits, QWidget* parent, const QGLWidget* shareWidget, Qt::WindowFlags f )
 | |
| :   _realized(false)
 | |
| {
 | |
| 
 | |
|     _widget = nullptr;
 | |
|     _traits = traits;
 | |
|     init( parent, shareWidget, f );
 | |
| }
 | |
| 
 | |
| GraphicsWindowQt::GraphicsWindowQt( GLWidget* widget )
 | |
| :   _realized(false)
 | |
| {
 | |
|     _widget = widget;
 | |
|     _traits = _widget ? createTraits( _widget ) : new osg::GraphicsContext::Traits;
 | |
|     init( nullptr, nullptr, Qt::WindowFlags() );
 | |
| }
 | |
| 
 | |
| GraphicsWindowQt::~GraphicsWindowQt()
 | |
| {
 | |
|     close();
 | |
| 
 | |
|     // remove reference from GLWidget
 | |
|     if ( _widget )
 | |
|         _widget->_gw = nullptr;
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::init( QWidget* parent, const QGLWidget* shareWidget, Qt::WindowFlags f )
 | |
| {
 | |
|     // update _widget and parent by WindowData
 | |
|     WindowData* windowData = _traits.get() ? dynamic_cast<WindowData*>(_traits->inheritedWindowData.get()) : nullptr;
 | |
|     if ( !_widget )
 | |
|         _widget = windowData ? windowData->_widget : nullptr;
 | |
|     if ( !parent )
 | |
|         parent = windowData ? windowData->_parent : nullptr;
 | |
| 
 | |
|     // create widget if it does not exist
 | |
|     _ownsWidget = _widget == nullptr;
 | |
|     if ( !_widget )
 | |
|     {
 | |
|         // shareWidget
 | |
|         if ( !shareWidget ) {
 | |
|             GraphicsWindowQt* sharedContextQt = dynamic_cast<GraphicsWindowQt*>(_traits->sharedContext.get());
 | |
|             if ( sharedContextQt )
 | |
|                 shareWidget = sharedContextQt->getGLWidget();
 | |
|         }
 | |
| 
 | |
|         // WindowFlags
 | |
|         Qt::WindowFlags flags = f | Qt::Window | Qt::CustomizeWindowHint;
 | |
|         if ( _traits->windowDecoration )
 | |
|             flags |= Qt::WindowTitleHint | Qt::WindowMinMaxButtonsHint | Qt::WindowSystemMenuHint | Qt::WindowCloseButtonHint;
 | |
| 
 | |
|         // create widget
 | |
|         _widget = new GLWidget( traits2qglFormat( _traits.get() ), parent, shareWidget, flags );
 | |
|     }
 | |
| 
 | |
|     // set widget name and position
 | |
|     // (do not set it when we inherited the widget)
 | |
|     if ( _ownsWidget )
 | |
|     {
 | |
|         _widget->setWindowTitle( _traits->windowName.c_str() );
 | |
|         _widget->move( _traits->x, _traits->y );
 | |
|         if ( !_traits->supportsResize ) _widget->setFixedSize( _traits->width, _traits->height );
 | |
|         else _widget->resize( _traits->width, _traits->height );
 | |
|     }
 | |
| 
 | |
|     // initialize widget properties
 | |
|     _widget->setAutoBufferSwap( false );
 | |
|     _widget->setMouseTracking( true );
 | |
|     _widget->setGraphicsWindow( this );
 | |
|     useCursor( _traits->useCursor );
 | |
| 
 | |
|     // initialize State
 | |
|     setState( new osg::State );
 | |
|     getState()->setGraphicsContext(this);
 | |
| 
 | |
|     // initialize contextID
 | |
|     if ( _traits.valid() && _traits->sharedContext.valid() )
 | |
|     {
 | |
|         getState()->setContextID( _traits->sharedContext->getState()->getContextID() );
 | |
|         incrementContextIDUsageCount( getState()->getContextID() );
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         getState()->setContextID( osg::GraphicsContext::createNewContextID() );
 | |
|     }
 | |
| 
 | |
|     // make sure the event queue has the correct window rectangle size and input range
 | |
|     getEventQueue()->syncWindowRectangleWithGraphicsContext();
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| QGLFormat GraphicsWindowQt::traits2qglFormat( const osg::GraphicsContext::Traits* traits )
 | |
| {
 | |
|     QGLFormat format( QGLFormat::defaultFormat() );
 | |
| 
 | |
|     format.setAlphaBufferSize( traits->alpha );
 | |
|     format.setRedBufferSize( traits->red );
 | |
|     format.setGreenBufferSize( traits->green );
 | |
|     format.setBlueBufferSize( traits->blue );
 | |
|     format.setDepthBufferSize( traits->depth );
 | |
|     format.setStencilBufferSize( traits->stencil );
 | |
|     format.setSampleBuffers( traits->sampleBuffers );
 | |
|     format.setSamples( traits->samples );
 | |
| 
 | |
|     format.setAlpha( traits->alpha>0 );
 | |
|     format.setDepth( traits->depth>0 );
 | |
|     format.setStencil( traits->stencil>0 );
 | |
|     format.setDoubleBuffer( traits->doubleBuffer );
 | |
|     format.setSwapInterval( traits->vsync ? 1 : 0 );
 | |
|     format.setStereo( traits->quadBufferStereo ? 1 : 0 );
 | |
| 
 | |
|     return format;
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::qglFormat2traits( const QGLFormat& format, osg::GraphicsContext::Traits* traits )
 | |
| {
 | |
|     traits->red = format.redBufferSize();
 | |
|     traits->green = format.greenBufferSize();
 | |
|     traits->blue = format.blueBufferSize();
 | |
|     traits->alpha = format.alpha() ? format.alphaBufferSize() : 0;
 | |
|     traits->depth = format.depth() ? format.depthBufferSize() : 0;
 | |
|     traits->stencil = format.stencil() ? format.stencilBufferSize() : 0;
 | |
| 
 | |
|     traits->sampleBuffers = format.sampleBuffers() ? 1 : 0;
 | |
|     traits->samples = format.samples();
 | |
| 
 | |
|     traits->quadBufferStereo = format.stereo();
 | |
|     traits->doubleBuffer = format.doubleBuffer();
 | |
| 
 | |
|     traits->vsync = format.swapInterval() >= 1;
 | |
| }
 | |
| 
 | |
| osg::GraphicsContext::Traits* GraphicsWindowQt::createTraits( const QGLWidget* widget )
 | |
| {
 | |
|     osg::GraphicsContext::Traits *traits = new osg::GraphicsContext::Traits;
 | |
| 
 | |
|     qglFormat2traits( widget->format(), traits );
 | |
| 
 | |
|     QRect r = widget->geometry();
 | |
|     traits->x = r.x();
 | |
|     traits->y = r.y();
 | |
|     traits->width = r.width();
 | |
|     traits->height = r.height();
 | |
| 
 | |
|     traits->windowName = widget->windowTitle().toLocal8Bit().data();
 | |
|     Qt::WindowFlags f = widget->windowFlags();
 | |
|     traits->windowDecoration = ( f & Qt::WindowTitleHint ) &&
 | |
|                             ( f & Qt::WindowMinMaxButtonsHint ) &&
 | |
|                             ( f & Qt::WindowSystemMenuHint );
 | |
|     QSizePolicy sp = widget->sizePolicy();
 | |
|     traits->supportsResize = sp.horizontalPolicy() != QSizePolicy::Fixed ||
 | |
|                             sp.verticalPolicy() != QSizePolicy::Fixed;
 | |
| 
 | |
|     return traits;
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::setWindowRectangleImplementation( int x, int y, int width, int height )
 | |
| {
 | |
|     if ( _widget == nullptr )
 | |
|         return false;
 | |
| 
 | |
|     _widget->setGeometry( x, y, width, height );
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::getWindowRectangle( int& x, int& y, int& width, int& height )
 | |
| {
 | |
|     if ( _widget )
 | |
|     {
 | |
|         const QRect& geom = _widget->geometry();
 | |
|         x = geom.x();
 | |
|         y = geom.y();
 | |
|         width = geom.width();
 | |
|         height = geom.height();
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::setWindowDecorationImplementation( bool windowDecoration )
 | |
| {
 | |
|     Qt::WindowFlags flags = Qt::Window|Qt::CustomizeWindowHint;//|Qt::WindowStaysOnTopHint;
 | |
|     if ( windowDecoration )
 | |
|         flags |= Qt::WindowTitleHint|Qt::WindowMinMaxButtonsHint|Qt::WindowSystemMenuHint;
 | |
|     _traits->windowDecoration = windowDecoration;
 | |
| 
 | |
|     if ( _widget )
 | |
|     {
 | |
|         _widget->setWindowFlags( flags );
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::getWindowDecoration() const
 | |
| {
 | |
|     return _traits->windowDecoration;
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::grabFocus()
 | |
| {
 | |
|     if ( _widget )
 | |
|         _widget->setFocus( Qt::ActiveWindowFocusReason );
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::grabFocusIfPointerInWindow()
 | |
| {
 | |
|     if ( _widget->underMouse() )
 | |
|         _widget->setFocus( Qt::ActiveWindowFocusReason );
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::raiseWindow()
 | |
| {
 | |
|     if ( _widget )
 | |
|         _widget->raise();
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::setWindowName( const std::string& name )
 | |
| {
 | |
|     if ( _widget )
 | |
|         _widget->setWindowTitle( name.c_str() );
 | |
| }
 | |
| 
 | |
| std::string GraphicsWindowQt::getWindowName()
 | |
| {
 | |
|     return _widget ? _widget->windowTitle().toStdString() : "";
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::useCursor( bool cursorOn )
 | |
| {
 | |
|     if ( _widget )
 | |
|     {
 | |
|         _traits->useCursor = cursorOn;
 | |
|         if ( !cursorOn ) _widget->setCursor( Qt::BlankCursor );
 | |
|         else _widget->setCursor( _currentCursor );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::setCursor( MouseCursor cursor )
 | |
| {
 | |
|     if ( cursor==InheritCursor && _widget )
 | |
|     {
 | |
|         _widget->unsetCursor();
 | |
|     }
 | |
| 
 | |
|     switch ( cursor )
 | |
|     {
 | |
|     case NoCursor: _currentCursor = Qt::BlankCursor; break;
 | |
|     case RightArrowCursor: case LeftArrowCursor: _currentCursor = Qt::ArrowCursor; break;
 | |
|     case InfoCursor: _currentCursor = Qt::SizeAllCursor; break;
 | |
|     case DestroyCursor: _currentCursor = Qt::ForbiddenCursor; break;
 | |
|     case HelpCursor: _currentCursor = Qt::WhatsThisCursor; break;
 | |
|     case CycleCursor: _currentCursor = Qt::ForbiddenCursor; break;
 | |
|     case SprayCursor: _currentCursor = Qt::SizeAllCursor; break;
 | |
|     case WaitCursor: _currentCursor = Qt::WaitCursor; break;
 | |
|     case TextCursor: _currentCursor = Qt::IBeamCursor; break;
 | |
|     case CrosshairCursor: _currentCursor = Qt::CrossCursor; break;
 | |
|     case HandCursor: _currentCursor = Qt::OpenHandCursor; break;
 | |
|     case UpDownCursor: _currentCursor = Qt::SizeVerCursor; break;
 | |
|     case LeftRightCursor: _currentCursor = Qt::SizeHorCursor; break;
 | |
|     case TopSideCursor: case BottomSideCursor: _currentCursor = Qt::UpArrowCursor; break;
 | |
|     case LeftSideCursor: case RightSideCursor: _currentCursor = Qt::SizeHorCursor; break;
 | |
|     case TopLeftCorner: _currentCursor = Qt::SizeBDiagCursor; break;
 | |
|     case TopRightCorner: _currentCursor = Qt::SizeFDiagCursor; break;
 | |
|     case BottomRightCorner: _currentCursor = Qt::SizeBDiagCursor; break;
 | |
|     case BottomLeftCorner: _currentCursor = Qt::SizeFDiagCursor; break;
 | |
|     default: break;
 | |
|     };
 | |
|     if ( _widget ) _widget->setCursor( _currentCursor );
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::valid() const
 | |
| {
 | |
|     return _widget && _widget->isValid();
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::realizeImplementation()
 | |
| {
 | |
|     // save the current context
 | |
|     // note: this will save only Qt-based contexts
 | |
|     const QGLContext *savedContext = QGLContext::currentContext();
 | |
| 
 | |
|     // initialize GL context for the widget
 | |
|     if ( !valid() )
 | |
|         _widget->glInit();
 | |
| 
 | |
|     // make current
 | |
|     _realized = true;
 | |
|     bool result = makeCurrent();
 | |
|     _realized = false;
 | |
| 
 | |
|     // fail if we do not have current context
 | |
|     if ( !result )
 | |
|     {
 | |
|         if ( savedContext )
 | |
|             const_cast< QGLContext* >( savedContext )->makeCurrent();
 | |
| 
 | |
|         OSG_WARN << "Window realize: Can make context current." << std::endl;
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     _realized = true;
 | |
| 
 | |
|     // make sure the event queue has the correct window rectangle size and input range
 | |
|     getEventQueue()->syncWindowRectangleWithGraphicsContext();
 | |
| 
 | |
|     // make this window's context not current
 | |
|     // note: this must be done as we will probably make the context current from another thread
 | |
|     //       and it is not allowed to have one context current in two threads
 | |
|     if( !releaseContext() )
 | |
|         OSG_WARN << "Window realize: Can not release context." << std::endl;
 | |
| 
 | |
|     // restore previous context
 | |
|     if ( savedContext )
 | |
|         const_cast< QGLContext* >( savedContext )->makeCurrent();
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::isRealizedImplementation() const
 | |
| {
 | |
|     return _realized;
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::closeImplementation()
 | |
| {
 | |
|     if ( _widget )
 | |
|         _widget->close();
 | |
|     _realized = false;
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::runOperations()
 | |
| {
 | |
|     // While in graphics thread this is last chance to do something useful before
 | |
|     // graphics thread will execute its operations.
 | |
|     if (_widget->getNumDeferredEvents() > 0)
 | |
|         _widget->processDeferredEvents();
 | |
| 
 | |
|     if (QGLContext::currentContext() != _widget->context())
 | |
|         _widget->makeCurrent();
 | |
| 
 | |
|     GraphicsWindow::runOperations();
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::makeCurrentImplementation()
 | |
| {
 | |
|     if (_widget->getNumDeferredEvents() > 0)
 | |
|         _widget->processDeferredEvents();
 | |
| 
 | |
|     _widget->makeCurrent();
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| bool GraphicsWindowQt::releaseContextImplementation()
 | |
| {
 | |
|     _widget->doneCurrent();
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::swapBuffersImplementation()
 | |
| {
 | |
|     // QOpenGLContext complains if we swap on an non-exposed QWindow
 | |
|     if (!_widget || !_widget->windowHandle()->isExposed())
 | |
|         return;
 | |
| 
 | |
|     // FIXME: the processDeferredEvents should really be executed in a GUI (main) thread context but
 | |
|     // I couln't find any reliable way to do this. For now, lets hope non of *GUI thread only operations* will
 | |
|     // be executed in a QGLWidget::event handler. On the other hand, calling GUI only operations in the
 | |
|     // QGLWidget event handler is an indication of a Qt bug.
 | |
|     if (_widget->getNumDeferredEvents() > 0)
 | |
|         _widget->processDeferredEvents();
 | |
| 
 | |
|     // We need to call makeCurrent here to restore our previously current context
 | |
|     // which may be changed by the processDeferredEvents function.
 | |
|     _widget->makeCurrent();
 | |
|     _widget->swapBuffers();
 | |
| }
 | |
| 
 | |
| void GraphicsWindowQt::requestWarpPointer( float x, float y )
 | |
| {
 | |
|     if ( _widget )
 | |
|         QCursor::setPos( _widget->mapToGlobal(QPoint((int)x,(int)y)) );
 | |
| }
 | |
| 
 | |
| 
 | |
| class QtWindowingSystem : public osg::GraphicsContext::WindowingSystemInterface
 | |
| {
 | |
| public:
 | |
| 
 | |
|     QtWindowingSystem()
 | |
|     {
 | |
|         OSG_INFO << "QtWindowingSystemInterface()" << std::endl;
 | |
|     }
 | |
| 
 | |
|     ~QtWindowingSystem()
 | |
|     {
 | |
|         if (osg::Referenced::getDeleteHandler())
 | |
|         {
 | |
|             osg::Referenced::getDeleteHandler()->setNumFramesToRetainObjects(0);
 | |
|             osg::Referenced::getDeleteHandler()->flushAll();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Access the Qt windowing system through this singleton class.
 | |
|     static QtWindowingSystem* getInterface()
 | |
|     {
 | |
|         static QtWindowingSystem* qtInterface = new QtWindowingSystem;
 | |
|         return qtInterface;
 | |
|     }
 | |
| 
 | |
|     // Return the number of screens present in the system
 | |
|     unsigned int getNumScreens( const osg::GraphicsContext::ScreenIdentifier& /*si*/ ) override
 | |
|     {
 | |
|         OSG_WARN << "osgQt: getNumScreens() not implemented yet." << std::endl;
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     // Return the resolution of specified screen
 | |
|     // (0,0) is returned if screen is unknown
 | |
|     void getScreenSettings( const osg::GraphicsContext::ScreenIdentifier& /*si*/, osg::GraphicsContext::ScreenSettings & /*resolution*/ ) override
 | |
|     {
 | |
|         OSG_WARN << "osgQt: getScreenSettings() not implemented yet." << std::endl;
 | |
|     }
 | |
| 
 | |
|     // Set the resolution for given screen
 | |
|     bool setScreenSettings( const osg::GraphicsContext::ScreenIdentifier& /*si*/, const osg::GraphicsContext::ScreenSettings & /*resolution*/ ) override
 | |
|     {
 | |
|         OSG_WARN << "osgQt: setScreenSettings() not implemented yet." << std::endl;
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     // Enumerates available resolutions
 | |
|     void enumerateScreenSettings( const osg::GraphicsContext::ScreenIdentifier& /*screenIdentifier*/, osg::GraphicsContext::ScreenSettingsList & /*resolution*/ ) override
 | |
|     {
 | |
|         OSG_WARN << "osgQt: enumerateScreenSettings() not implemented yet." << std::endl;
 | |
|     }
 | |
| 
 | |
|     // Create a graphics context with given traits
 | |
|     osg::GraphicsContext* createGraphicsContext( osg::GraphicsContext::Traits* traits ) override
 | |
|     {
 | |
|         if (traits->pbuffer)
 | |
|         {
 | |
|             OSG_WARN << "osgQt: createGraphicsContext - pbuffer not implemented yet." << std::endl;
 | |
|             return nullptr;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             osg::ref_ptr< GraphicsWindowQt > window = new GraphicsWindowQt( traits );
 | |
|             if (window->valid()) return window.release();
 | |
|             else return nullptr;
 | |
|         }
 | |
|     }
 | |
| 
 | |
| private:
 | |
| 
 | |
|     // No implementation for these
 | |
|     QtWindowingSystem( const QtWindowingSystem& );
 | |
|     QtWindowingSystem& operator=( const QtWindowingSystem& );
 | |
| };
 | |
| 
 |