2013-11-02 02:48:30 +01:00
# include "scenewidget.hpp"
# include <QEvent>
# include <QResizeEvent>
2014-02-25 13:09:37 +01:00
# include <QTimer>
2014-07-14 11:19:59 +02:00
# include <QShortcut>
2015-03-19 23:27:14 +01:00
# include <QLayout>
2013-11-02 02:48:30 +01:00
2016-02-15 15:16:48 +01:00
# include <extern/osgQt/GraphicsWindowQt>
2015-03-28 20:15:17 +01:00
# include <osg/GraphicsContext>
# include <osgViewer/CompositeViewer>
# include <osgViewer/ViewerEventHandlers>
2015-03-28 21:26:16 +01:00
# include <osg/LightModel>
2016-03-10 16:56:14 -05:00
2015-03-28 20:15:17 +01:00
# include <components/resource/scenemanager.hpp>
2015-07-18 03:52:38 +02:00
# include <components/resource/resourcesystem.hpp>
2016-01-06 15:00:58 +01:00
# include <components/sceneutil/lightmanager.hpp>
2015-03-28 20:15:17 +01:00
2014-07-08 12:39:12 +02:00
# include "../widget/scenetoolmode.hpp"
2014-03-23 15:14:26 +01:00
2016-03-10 04:29:24 -05:00
# include "../../model/prefs/state.hpp"
2014-03-23 15:14:26 +01:00
# include "lighting.hpp"
2016-01-06 14:34:39 +01:00
# include "mask.hpp"
2016-03-14 00:04:11 -04:00
# include "cameracontroller.hpp"
2014-02-27 15:23:14 +01:00
2015-03-19 23:27:14 +01:00
namespace CSVRender
{
2013-11-02 02:48:30 +01:00
2015-03-28 20:15:17 +01:00
RenderWidget : : RenderWidget ( QWidget * parent , Qt : : WindowFlags f )
2015-03-19 23:27:14 +01:00
: QWidget ( parent , f )
2015-03-20 00:39:24 +01:00
, mRootNode ( 0 )
2015-03-19 23:27:14 +01:00
{
2013-11-02 02:48:30 +01:00
2015-03-23 15:55:57 +01:00
osgViewer : : CompositeViewer & viewer = CompositeViewer : : get ( ) ;
2014-02-27 15:23:14 +01:00
2015-03-19 23:27:14 +01:00
osg : : DisplaySettings * ds = osg : : DisplaySettings : : instance ( ) . get ( ) ;
2015-03-20 00:39:24 +01:00
//ds->setNumMultiSamples(8);
2015-03-23 15:55:57 +01:00
2015-03-19 23:27:14 +01:00
osg : : ref_ptr < osg : : GraphicsContext : : Traits > traits = new osg : : GraphicsContext : : Traits ;
traits - > windowName = " " ;
traits - > windowDecoration = true ;
traits - > x = 0 ;
traits - > y = 0 ;
traits - > width = width ( ) ;
traits - > height = height ( ) ;
traits - > doubleBuffer = true ;
traits - > alpha = ds - > getMinimumNumAlphaBits ( ) ;
traits - > stencil = ds - > getMinimumNumStencilBits ( ) ;
traits - > sampleBuffers = ds - > getMultiSamples ( ) ;
traits - > samples = ds - > getNumMultiSamples ( ) ;
2015-03-20 00:39:24 +01:00
// Doesn't make much sense as we're running on demand updates, and there seems to be a bug with the refresh rate when running multiple QGLWidgets
traits - > vsync = false ;
2014-02-25 13:09:37 +01:00
2015-03-23 15:55:57 +01:00
mView = new osgViewer : : View ;
osg : : ref_ptr < osgQt : : GraphicsWindowQt > window = new osgQt : : GraphicsWindowQt ( traits . get ( ) ) ;
2015-03-19 23:27:14 +01:00
QLayout * layout = new QHBoxLayout ( this ) ;
2015-09-18 18:29:22 +02:00
layout - > setContentsMargins ( 0 , 0 , 0 , 0 ) ;
2015-03-19 23:27:14 +01:00
layout - > addWidget ( window - > getGLWidget ( ) ) ;
setLayout ( layout ) ;
2014-02-25 13:39:49 +01:00
2015-03-23 15:55:57 +01:00
mView - > getCamera ( ) - > setGraphicsContext ( window ) ;
mView - > getCamera ( ) - > setClearColor ( osg : : Vec4 ( 0.2 , 0.2 , 0.6 , 1.0 ) ) ;
mView - > getCamera ( ) - > setViewport ( new osg : : Viewport ( 0 , 0 , traits - > width , traits - > height ) ) ;
mView - > getCamera ( ) - > setProjectionMatrixAsPerspective ( 30.0f , static_cast < double > ( traits - > width ) / static_cast < double > ( traits - > height ) , 1.0f , 10000.0f ) ;
2014-02-25 14:09:07 +01:00
2016-01-06 15:00:58 +01:00
SceneUtil : : LightManager * lightMgr = new SceneUtil : : LightManager ;
lightMgr - > setStartLight ( 1 ) ;
2016-01-06 23:20:26 +01:00
lightMgr - > setLightingMask ( Mask_Lighting ) ;
2016-01-06 15:00:58 +01:00
mRootNode = lightMgr ;
2015-03-28 20:15:17 +01:00
2015-03-28 21:26:16 +01:00
mView - > getCamera ( ) - > getOrCreateStateSet ( ) - > setMode ( GL_NORMALIZE , osg : : StateAttribute : : ON ) ;
mView - > getCamera ( ) - > getOrCreateStateSet ( ) - > setMode ( GL_CULL_FACE , osg : : StateAttribute : : ON ) ;
2015-03-25 23:35:10 +01:00
2015-03-23 15:55:57 +01:00
mView - > setSceneData ( mRootNode ) ;
2015-03-20 00:39:24 +01:00
// Press S to reveal profiling stats
2015-03-23 15:55:57 +01:00
mView - > addEventHandler ( new osgViewer : : StatsHandler ) ;
2016-01-06 14:34:39 +01:00
mView - > getCamera ( ) - > setCullMask ( ~ ( Mask_UpdateVisitor ) ) ;
2015-03-23 15:55:57 +01:00
viewer . addView ( mView ) ;
viewer . setDone ( false ) ;
viewer . realize ( ) ;
}
2015-03-28 20:15:17 +01:00
RenderWidget : : ~ RenderWidget ( )
2015-03-23 15:55:57 +01:00
{
CompositeViewer : : get ( ) . removeView ( mView ) ;
}
2015-03-28 20:15:17 +01:00
void RenderWidget : : flagAsModified ( )
2015-03-23 15:55:57 +01:00
{
mView - > requestRedraw ( ) ;
}
2015-03-28 20:15:17 +01:00
void RenderWidget : : setVisibilityMask ( int mask )
2015-03-26 00:27:39 +01:00
{
2016-01-06 23:20:26 +01:00
mView - > getCamera ( ) - > setCullMask ( mask | Mask_ParticleSystem | Mask_Lighting ) ;
2015-03-26 00:27:39 +01:00
}
2016-02-16 16:02:29 +01:00
osg : : Camera * RenderWidget : : getCamera ( )
{
return mView - > getCamera ( ) ;
}
2015-03-28 20:15:17 +01:00
// --------------------------------------------------
2015-03-23 15:55:57 +01:00
CompositeViewer : : CompositeViewer ( )
2015-06-14 19:19:23 +02:00
: mSimulationTime ( 0.0 )
2015-03-23 15:55:57 +01:00
{
# if QT_VERSION >= 0x050000
// Qt5 is currently crashing and reporting "Cannot make QOpenGLContext current in a different thread" when the viewer is run multi-threaded, this is regression from Qt4
osgViewer : : ViewerBase : : ThreadingModel threadingModel = osgViewer : : ViewerBase : : SingleThreaded ;
# else
2015-11-13 20:59:39 +01:00
osgViewer : : ViewerBase : : ThreadingModel threadingModel = osgViewer : : ViewerBase : : DrawThreadPerContext ;
2015-03-23 15:55:57 +01:00
# endif
setThreadingModel ( threadingModel ) ;
// disable the default setting of viewer.done() by pressing Escape.
setKeyEventSetsDone ( 0 ) ;
2014-02-25 14:09:07 +01:00
2015-03-19 23:27:14 +01:00
// Only render when the camera position changed, or content flagged dirty
//setRunFrameScheme(osgViewer::ViewerBase::ON_DEMAND);
2015-03-20 00:39:24 +01:00
setRunFrameScheme ( osgViewer : : ViewerBase : : CONTINUOUS ) ;
2014-02-25 13:09:37 +01:00
2015-03-19 23:27:14 +01:00
connect ( & mTimer , SIGNAL ( timeout ( ) ) , this , SLOT ( update ( ) ) ) ;
2015-03-28 20:15:17 +01:00
mTimer . start ( 10 ) ;
2015-03-19 23:27:14 +01:00
}
2014-09-28 14:08:14 +10:00
2015-03-23 15:55:57 +01:00
CompositeViewer & CompositeViewer : : get ( )
2015-03-19 23:27:14 +01:00
{
2015-03-23 15:55:57 +01:00
static CompositeViewer sThis ;
return sThis ;
2015-03-19 23:27:14 +01:00
}
2014-09-28 15:43:33 +10:00
2015-03-23 15:55:57 +01:00
void CompositeViewer : : update ( )
2015-03-19 23:27:14 +01:00
{
2016-03-14 00:04:11 -04:00
double dt = mFrameTimer . time_s ( ) ;
2015-06-14 19:19:23 +02:00
mFrameTimer . setStartTick ( ) ;
2016-03-14 00:04:11 -04:00
emit simulationUpdated ( dt ) ;
mSimulationTime + = dt ;
2015-06-14 19:19:23 +02:00
frame ( mSimulationTime ) ;
2015-03-19 23:27:14 +01:00
}
2014-09-29 06:27:54 +10:00
2015-03-28 20:15:17 +01:00
// ---------------------------------------------------
2016-03-10 04:29:24 -05:00
SceneWidget : : SceneWidget ( boost : : shared_ptr < Resource : : ResourceSystem > resourceSystem , QWidget * parent , Qt : : WindowFlags f ,
bool retrieveInput )
2015-03-28 20:15:17 +01:00
: RenderWidget ( parent , f )
2015-07-18 03:52:38 +02:00
, mResourceSystem ( resourceSystem )
2015-03-28 20:15:17 +01:00
, mLighting ( NULL )
2015-03-28 21:26:16 +01:00
, mHasDefaultAmbient ( false )
2016-03-14 00:04:11 -04:00
, mPrevMouseX ( 0 )
, mPrevMouseY ( 0 )
, mFreeCamControl ( new FreeCameraController ( ) )
2016-03-18 16:06:35 -04:00
, mOrbitCamControl ( new OrbitCameraController ( ) )
2016-03-14 00:04:11 -04:00
, mCurrentCamControl ( mFreeCamControl . get ( ) )
2016-03-18 14:02:24 -04:00
, mCamPositionSet ( false )
2015-03-28 20:15:17 +01:00
{
2016-04-02 13:59:57 -04:00
mOrbitCamControl - > setPickingMask ( Mask_Reference | Mask_Terrain ) ;
2016-03-14 00:04:11 -04:00
selectNavigationMode ( " free " ) ;
2015-03-28 21:26:16 +01:00
// we handle lighting manually
mView - > setLightingMode ( osgViewer : : View : : NO_LIGHT ) ;
2015-03-28 20:15:17 +01:00
setLighting ( & mLightingDay ) ;
2015-09-17 12:41:56 +02:00
2016-01-06 14:39:48 +01:00
mResourceSystem - > getSceneManager ( ) - > setParticleSystemMask ( Mask_ParticleSystem ) ;
2015-09-17 12:41:56 +02:00
/// \todo make shortcut configurable
QShortcut * focusToolbar = new QShortcut ( Qt : : Key_T , this , 0 , 0 , Qt : : WidgetWithChildrenShortcut ) ;
connect ( focusToolbar , SIGNAL ( activated ( ) ) , this , SIGNAL ( focusToolbarRequest ( ) ) ) ;
2016-03-10 04:29:24 -05:00
connect ( & CSMPrefs : : State : : get ( ) , SIGNAL ( settingChanged ( const CSMPrefs : : Setting * ) ) ,
this , SLOT ( settingChanged ( const CSMPrefs : : Setting * ) ) ) ;
// TODO update this outside of the constructor where virtual methods can be used
if ( retrieveInput )
{
CSMPrefs : : get ( ) [ " 3D Scene Input " ] . update ( ) ;
CSMPrefs : : get ( ) [ " Tooltips " ] . update ( ) ;
}
2016-03-14 00:04:11 -04:00
connect ( & CompositeViewer : : get ( ) , SIGNAL ( simulationUpdated ( double ) ) , this , SLOT ( update ( double ) ) ) ;
2015-03-28 20:15:17 +01:00
}
SceneWidget : : ~ SceneWidget ( )
{
// Since we're holding on to the scene templates past the existance of this graphics context, we'll need to manually release the created objects
2015-07-18 03:52:38 +02:00
mResourceSystem - > getSceneManager ( ) - > releaseGLObjects ( mView - > getCamera ( ) - > getGraphicsContext ( ) - > getState ( ) ) ;
2015-03-28 20:15:17 +01:00
}
void SceneWidget : : setLighting ( Lighting * lighting )
{
if ( mLighting )
mLighting - > deactivate ( ) ;
mLighting = lighting ;
2015-03-28 21:26:16 +01:00
mLighting - > activate ( mRootNode ) ;
osg : : Vec4f ambient = mLighting - > getAmbientColour ( mHasDefaultAmbient ? & mDefaultAmbient : 0 ) ;
setAmbient ( ambient ) ;
2015-03-28 20:15:17 +01:00
flagAsModified ( ) ;
}
2015-03-28 21:26:16 +01:00
void SceneWidget : : setAmbient ( const osg : : Vec4f & ambient )
{
osg : : ref_ptr < osg : : StateSet > stateset = new osg : : StateSet ;
osg : : ref_ptr < osg : : LightModel > lightmodel = new osg : : LightModel ;
lightmodel - > setAmbientIntensity ( ambient ) ;
stateset - > setMode ( GL_LIGHTING , osg : : StateAttribute : : ON ) ;
stateset - > setMode ( GL_LIGHT0 , osg : : StateAttribute : : ON ) ;
stateset - > setAttributeAndModes ( lightmodel , osg : : StateAttribute : : ON ) ;
mRootNode - > setStateSet ( stateset ) ;
}
2015-03-28 20:15:17 +01:00
void SceneWidget : : selectLightingMode ( const std : : string & mode )
{
if ( mode = = " day " )
setLighting ( & mLightingDay ) ;
else if ( mode = = " night " )
setLighting ( & mLightingNight ) ;
else if ( mode = = " bright " )
setLighting ( & mLightingBright ) ;
}
CSVWidget : : SceneToolMode * SceneWidget : : makeLightingSelector ( CSVWidget : : SceneToolbar * parent )
{
CSVWidget : : SceneToolMode * tool = new CSVWidget : : SceneToolMode ( parent , " Lighting Mode " ) ;
/// \todo replace icons
tool - > addButton ( " :scenetoolbar/day " , " day " ,
" Day "
" <ul><li>Cell specific ambient in interiors</li> "
" <li>Low ambient in exteriors</li> "
" <li>Strong directional light source</li> "
" <li>This mode closely resembles day time in-game</li></ul> " ) ;
tool - > addButton ( " :scenetoolbar/night " , " night " ,
" Night "
" <ul><li>Cell specific ambient in interiors</li> "
" <li>Low ambient in exteriors</li> "
" <li>Weak directional light source</li> "
" <li>This mode closely resembles night time in-game</li></ul> " ) ;
tool - > addButton ( " :scenetoolbar/bright " , " bright " ,
" Bright "
" <ul><li>Maximum ambient</li> "
" <li>Strong directional light source</li></ul> " ) ;
connect ( tool , SIGNAL ( modeChanged ( const std : : string & ) ) ,
this , SLOT ( selectLightingMode ( const std : : string & ) ) ) ;
return tool ;
}
void SceneWidget : : setDefaultAmbient ( const osg : : Vec4f & colour )
{
mDefaultAmbient = colour ;
mHasDefaultAmbient = true ;
2015-03-28 21:26:16 +01:00
setAmbient ( mLighting - > getAmbientColour ( & mDefaultAmbient ) ) ;
2015-03-28 20:15:17 +01:00
}
2016-03-10 04:29:24 -05:00
void SceneWidget : : mousePressEvent ( QMouseEvent * event )
{
2016-03-14 00:04:11 -04:00
mMouseMode = mapButton ( event ) ;
2016-03-10 04:29:24 -05:00
2016-03-14 00:04:11 -04:00
mPrevMouseX = event - > x ( ) ;
mPrevMouseY = event - > y ( ) ;
2016-03-10 04:29:24 -05:00
}
void SceneWidget : : mouseReleaseEvent ( QMouseEvent * event )
{
2016-03-14 00:04:11 -04:00
mMouseMode = " " ;
}
void SceneWidget : : mouseMoveEvent ( QMouseEvent * event )
{
mCurrentCamControl - > handleMouseMoveEvent ( mMouseMode , event - > x ( ) - mPrevMouseX , event - > y ( ) - mPrevMouseY ) ;
2016-03-10 04:29:24 -05:00
2016-03-14 00:04:11 -04:00
mPrevMouseX = event - > x ( ) ;
mPrevMouseY = event - > y ( ) ;
}
void SceneWidget : : wheelEvent ( QWheelEvent * event )
{
mCurrentCamControl - > handleMouseMoveEvent ( " t-navi " , event - > delta ( ) , 0 ) ;
}
void SceneWidget : : keyPressEvent ( QKeyEvent * event )
{
mCurrentCamControl - > handleKeyEvent ( event , true ) ;
}
void SceneWidget : : keyReleaseEvent ( QKeyEvent * event )
{
mCurrentCamControl - > handleKeyEvent ( event , false ) ;
}
void SceneWidget : : update ( double dt )
{
2016-03-18 14:02:24 -04:00
if ( mCamPositionSet )
{
mCurrentCamControl - > update ( dt ) ;
}
else
{
2016-04-02 13:36:12 -04:00
mCurrentCamControl - > setup ( mRootNode , Mask_Reference | Mask_Terrain , CameraController : : WorldUp ) ;
2016-03-18 14:02:24 -04:00
mCamPositionSet = true ;
}
2016-03-10 04:29:24 -05:00
}
void SceneWidget : : settingChanged ( const CSMPrefs : : Setting * setting )
{
2016-04-04 18:42:57 -04:00
if ( * setting = = " 3D Scene Input/p-navi-free-sensitivity " )
2016-03-25 22:19:44 -04:00
{
mFreeCamControl - > setCameraSensitivity ( setting - > toDouble ( ) ) ;
2016-04-04 18:42:57 -04:00
}
else if ( * setting = = " 3D Scene Input/p-navi-orbit-sensitivity " )
{
2016-03-25 22:19:44 -04:00
mOrbitCamControl - > setCameraSensitivity ( setting - > toDouble ( ) ) ;
}
2016-04-04 18:42:57 -04:00
else if ( * setting = = " 3D Scene Input/p-navi-free-invert " )
{
mFreeCamControl - > setInverted ( setting - > isTrue ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/p-navi-orbit-invert " )
{
mOrbitCamControl - > setInverted ( setting - > isTrue ( ) ) ;
}
2016-03-25 22:19:44 -04:00
else if ( * setting = = " 3D Scene Input/s-navi-sensitivity " )
{
mFreeCamControl - > setSecondaryMovementMultiplier ( setting - > toDouble ( ) ) ;
mOrbitCamControl - > setSecondaryMovementMultiplier ( setting - > toDouble ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/navi-wheel-factor " )
{
mFreeCamControl - > setWheelMovementMultiplier ( setting - > toDouble ( ) ) ;
mOrbitCamControl - > setWheelMovementMultiplier ( setting - > toDouble ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/navi-free-lin-speed " )
{
mFreeCamControl - > setLinearSpeed ( setting - > toDouble ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/navi-free-rot-speed " )
{
mFreeCamControl - > setRotationalSpeed ( setting - > toDouble ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/navi-free-speed-mult " )
{
mFreeCamControl - > setSpeedMultiplier ( setting - > toDouble ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/navi-orbit-rot-speed " )
{
mOrbitCamControl - > setOrbitSpeed ( setting - > toDouble ( ) ) ;
}
else if ( * setting = = " 3D Scene Input/navi-orbit-speed-mult " )
{
mOrbitCamControl - > setOrbitSpeedMultiplier ( setting - > toDouble ( ) ) ;
}
else
{
storeMappingSetting ( setting ) ;
}
2016-03-10 04:29:24 -05:00
}
2016-03-10 16:56:14 -05:00
void SceneWidget : : selectNavigationMode ( const std : : string & mode )
{
if ( mode = = " 1st " )
2016-03-14 00:04:11 -04:00
{
mCurrentCamControl - > setCamera ( NULL ) ;
mCurrentCamControl = mFreeCamControl . get ( ) ;
mCurrentCamControl - > setCamera ( getCamera ( ) ) ;
2016-03-18 13:57:48 -04:00
mFreeCamControl - > fixUpAxis ( CameraController : : WorldUp ) ;
2016-03-14 00:04:11 -04:00
}
2016-03-10 16:56:14 -05:00
else if ( mode = = " free " )
2016-03-14 00:04:11 -04:00
{
mCurrentCamControl - > setCamera ( NULL ) ;
mCurrentCamControl = mFreeCamControl . get ( ) ;
mCurrentCamControl - > setCamera ( getCamera ( ) ) ;
mFreeCamControl - > unfixUpAxis ( ) ;
}
2016-03-10 16:56:14 -05:00
else if ( mode = = " orbit " )
2016-03-14 00:04:11 -04:00
{
mCurrentCamControl - > setCamera ( NULL ) ;
mCurrentCamControl = mOrbitCamControl . get ( ) ;
mCurrentCamControl - > setCamera ( getCamera ( ) ) ;
}
2016-03-10 16:56:14 -05:00
}
2016-03-10 04:29:24 -05:00
bool SceneWidget : : storeMappingSetting ( const CSMPrefs : : Setting * setting )
{
if ( setting - > getParent ( ) - > getKey ( ) ! = " 3D Scene Input " )
return false ;
static const char * const sMappingSettings [ ] =
{
" p-navi " , " s-navi " ,
0
} ;
for ( int i = 0 ; sMappingSettings [ i ] ; + + i )
if ( setting - > getKey ( ) = = sMappingSettings [ i ] )
{
QString value = QString : : fromUtf8 ( setting - > toString ( ) . c_str ( ) ) ;
Qt : : MouseButton button = Qt : : NoButton ;
if ( value . endsWith ( " Left Mouse-Button " ) )
button = Qt : : LeftButton ;
else if ( value . endsWith ( " Right Mouse-Button " ) )
button = Qt : : RightButton ;
else if ( value . endsWith ( " Middle Mouse-Button " ) )
button = Qt : : MiddleButton ;
else
return false ;
bool ctrl = value . startsWith ( " Ctrl- " ) ;
mButtonMapping [ std : : make_pair ( button , ctrl ) ] = sMappingSettings [ i ] ;
return true ;
}
return false ;
}
std : : string SceneWidget : : mapButton ( QMouseEvent * event )
{
std : : pair < Qt : : MouseButton , bool > phyiscal (
event - > button ( ) , event - > modifiers ( ) & Qt : : ControlModifier ) ;
std : : map < std : : pair < Qt : : MouseButton , bool > , std : : string > : : const_iterator iter =
mButtonMapping . find ( phyiscal ) ;
if ( iter ! = mButtonMapping . end ( ) )
return iter - > second ;
return " " ;
}
2013-11-02 02:48:30 +01:00
}