2011-10-20 15:02:19 -04:00
# include "renderingmanager.hpp"
2015-08-21 22:00:08 +12:00
# include <limits>
2017-02-15 20:57:12 +01:00
# include <cstdlib>
2011-10-20 18:15:30 -04:00
2015-04-12 15:34:50 +02:00
# include <osg/Light>
# include <osg/LightModel>
2015-04-21 23:27:26 +02:00
# include <osg/Fog>
2017-02-01 03:18:17 +01:00
# include <osg/Material>
2015-06-01 17:02:44 +02:00
# include <osg/PolygonMode>
2015-04-12 15:34:50 +02:00
# include <osg/Group>
2015-05-24 03:36:34 +02:00
# include <osg/UserDataContainer>
# include <osg/ComputeBoundsVisitor>
2021-06-01 12:15:25 -07:00
# include <osg/Depth>
# include <osg/ClipControl>
2011-10-20 18:15:30 -04:00
2015-05-24 03:36:34 +02:00
# include <osgUtil/LineSegmentIntersector>
2015-05-03 18:16:54 +02:00
2015-04-12 15:34:50 +02:00
# include <osgViewer/Viewer>
2013-07-29 02:32:08 +02:00
2020-04-22 15:57:24 +02:00
# include <components/nifosg/nifloader.hpp>
2020-04-22 13:34:19 +02:00
2018-08-14 23:05:43 +04:00
# include <components/debug/debuglog.hpp>
2015-04-22 19:08:56 +02:00
# include <components/resource/resourcesystem.hpp>
2016-02-05 23:03:53 +01:00
# include <components/resource/imagemanager.hpp>
2015-05-26 20:20:18 +02:00
# include <components/resource/scenemanager.hpp>
2017-02-09 03:32:40 +01:00
# include <components/resource/keyframemanager.hpp>
2021-01-08 19:33:51 +04:00
2020-12-18 00:02:51 +00:00
# include <components/shader/removedalphafunc.hpp>
2017-11-15 15:20:59 +01:00
# include <components/shader/shadermanager.hpp>
2015-04-22 19:08:56 +02:00
2015-04-21 23:27:26 +02:00
# include <components/settings/settings.hpp>
2015-04-12 15:34:50 +02:00
# include <components/sceneutil/util.hpp>
# include <components/sceneutil/lightmanager.hpp>
2015-04-14 17:29:12 +02:00
# include <components/sceneutil/statesetupdater.hpp>
2015-11-20 21:57:04 +01:00
# include <components/sceneutil/positionattitudetransform.hpp>
2016-02-09 01:02:40 +01:00
# include <components/sceneutil/workqueue.hpp>
2016-02-09 15:30:53 +01:00
# include <components/sceneutil/unrefqueue.hpp>
2017-02-01 03:00:33 +01:00
# include <components/sceneutil/writescene.hpp>
2017-11-08 01:44:49 +00:00
# include <components/sceneutil/shadow.hpp>
2015-04-14 15:55:56 +02:00
2015-06-03 01:18:36 +02:00
# include <components/terrain/terraingrid.hpp>
2017-03-09 03:49:40 +01:00
# include <components/terrain/quadtreeworld.hpp>
2015-06-03 01:18:36 +02:00
2015-04-12 15:34:50 +02:00
# include <components/esm/loadcell.hpp>
2013-08-16 13:01:52 +02:00
2018-04-07 16:11:23 +03:00
# include <components/detournavigator/navigator.hpp>
2015-11-03 02:17:42 +01:00
# include "../mwworld/cellstore.hpp"
2018-06-16 10:18:04 +04:00
# include "../mwworld/class.hpp"
2017-11-16 22:08:33 +01:00
# include "../mwgui/loadingscreen.hpp"
# include "../mwbase/windowmanager.hpp"
2021-04-03 22:25:13 -07:00
# include "../mwmechanics/actorutil.hpp"
2015-11-01 22:09:02 +01:00
2015-04-14 15:55:56 +02:00
# include "sky.hpp"
2015-04-19 17:55:56 +02:00
# include "effectmanager.hpp"
2015-05-01 18:21:50 +02:00
# include "npcanimation.hpp"
2020-04-20 18:47:14 +02:00
# include "vismask.hpp"
2015-05-04 17:41:30 +02:00
# include "pathgrid.hpp"
2015-05-21 23:54:39 +02:00
# include "camera.hpp"
2020-07-11 14:21:18 +02:00
# include "viewovershoulder.hpp"
2015-06-02 16:35:35 +02:00
# include "water.hpp"
2015-06-03 01:18:36 +02:00
# include "terrainstorage.hpp"
2018-04-07 16:11:23 +03:00
# include "navmesh.hpp"
2018-07-21 13:37:02 +03:00
# include "actorspaths.hpp"
2019-11-27 23:45:01 +01:00
# include "recastmesh.hpp"
2020-06-07 10:35:33 +04:00
# include "fogmanager.hpp"
2019-06-13 13:37:00 +00:00
# include "objectpaging.hpp"
2021-01-08 19:33:51 +04:00
# include "screenshotmanager.hpp"
2020-01-12 11:42:47 +04:00
# include "groundcover.hpp"
2021-06-01 12:15:25 -07:00
# include "postprocessor.hpp"
2017-09-21 03:08:45 -07:00
2015-04-12 15:34:50 +02:00
namespace MWRender
2012-03-08 10:46:34 +04:00
{
2021-07-09 10:05:27 -07:00
class SharedUniformStateUpdater : public SceneUtil : : StateSetUpdater
{
public :
SharedUniformStateUpdater ( )
: mLinearFac ( 0.f )
2021-08-04 17:49:57 -07:00
, mNear ( 0.f )
, mFar ( 0.f )
2021-07-09 10:05:27 -07:00
{
}
void setDefaults ( osg : : StateSet * stateset ) override
{
stateset - > addUniform ( new osg : : Uniform ( " projectionMatrix " , osg : : Matrixf { } ) ) ;
stateset - > addUniform ( new osg : : Uniform ( " linearFac " , 0.f ) ) ;
2021-08-04 17:49:57 -07:00
stateset - > addUniform ( new osg : : Uniform ( " near " , 0.f ) ) ;
stateset - > addUniform ( new osg : : Uniform ( " far " , 0.f ) ) ;
2021-07-09 10:05:27 -07:00
}
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * nv ) override
{
auto * uProjectionMatrix = stateset - > getUniform ( " projectionMatrix " ) ;
if ( uProjectionMatrix )
uProjectionMatrix - > set ( mProjectionMatrix ) ;
auto * uLinearFac = stateset - > getUniform ( " linearFac " ) ;
if ( uLinearFac )
uLinearFac - > set ( mLinearFac ) ;
2021-08-04 17:49:57 -07:00
auto * uNear = stateset - > getUniform ( " near " ) ;
if ( uNear )
uNear - > set ( mNear ) ;
auto * uFar = stateset - > getUniform ( " far " ) ;
if ( uFar )
uFar - > set ( mFar ) ;
2021-07-09 10:05:27 -07:00
}
void setProjectionMatrix ( const osg : : Matrixf & projectionMatrix )
{
mProjectionMatrix = projectionMatrix ;
}
void setLinearFac ( float linearFac )
{
mLinearFac = linearFac ;
}
2021-08-04 17:49:57 -07:00
void setNear ( float near )
{
mNear = near ;
}
void setFar ( float far )
{
mFar = far ;
}
2021-07-09 10:05:27 -07:00
private :
osg : : Matrixf mProjectionMatrix ;
float mLinearFac ;
2021-08-04 17:49:57 -07:00
float mNear ;
float mFar ;
2021-07-09 10:05:27 -07:00
} ;
2021-01-12 12:39:19 +04:00
2015-04-14 17:29:12 +02:00
class StateUpdater : public SceneUtil : : StateSetUpdater
2015-04-14 15:55:56 +02:00
{
public :
2015-04-21 23:27:26 +02:00
StateUpdater ( )
2015-07-01 03:21:39 +02:00
: mFogStart ( 0.f )
, mFogEnd ( 0.f )
2015-06-01 17:02:44 +02:00
, mWireframe ( false )
2015-04-21 23:27:26 +02:00
{
}
2020-10-16 22:18:54 +04:00
void setDefaults ( osg : : StateSet * stateset ) override
2015-04-14 15:55:56 +02:00
{
osg : : LightModel * lightModel = new osg : : LightModel ;
stateset - > setAttribute ( lightModel , osg : : StateAttribute : : ON ) ;
2015-04-21 23:27:26 +02:00
osg : : Fog * fog = new osg : : Fog ;
2015-05-26 18:22:21 +02:00
fog - > setMode ( osg : : Fog : : LINEAR ) ;
2015-04-21 23:27:26 +02:00
stateset - > setAttributeAndModes ( fog , osg : : StateAttribute : : ON ) ;
2015-06-01 17:02:44 +02:00
if ( mWireframe )
{
osg : : PolygonMode * polygonmode = new osg : : PolygonMode ;
polygonmode - > setMode ( osg : : PolygonMode : : FRONT_AND_BACK , osg : : PolygonMode : : LINE ) ;
stateset - > setAttributeAndModes ( polygonmode , osg : : StateAttribute : : ON ) ;
}
else
stateset - > removeAttribute ( osg : : StateAttribute : : POLYGONMODE ) ;
2015-04-14 15:55:56 +02:00
}
2020-10-16 22:18:54 +04:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * ) override
2015-04-14 15:55:56 +02:00
{
osg : : LightModel * lightModel = static_cast < osg : : LightModel * > ( stateset - > getAttribute ( osg : : StateAttribute : : LIGHTMODEL ) ) ;
lightModel - > setAmbientIntensity ( mAmbientColor ) ;
2015-04-21 23:27:26 +02:00
osg : : Fog * fog = static_cast < osg : : Fog * > ( stateset - > getAttribute ( osg : : StateAttribute : : FOG ) ) ;
fog - > setColor ( mFogColor ) ;
2015-07-01 03:21:39 +02:00
fog - > setStart ( mFogStart ) ;
2015-04-21 23:27:26 +02:00
fog - > setEnd ( mFogEnd ) ;
2015-04-14 15:55:56 +02:00
}
2015-04-21 23:27:26 +02:00
void setAmbientColor ( const osg : : Vec4f & col )
2015-04-14 15:55:56 +02:00
{
mAmbientColor = col ;
}
2015-04-21 23:27:26 +02:00
void setFogColor ( const osg : : Vec4f & col )
{
mFogColor = col ;
}
2015-07-01 03:21:39 +02:00
void setFogStart ( float start )
{
mFogStart = start ;
}
2015-04-21 23:27:26 +02:00
void setFogEnd ( float end )
{
mFogEnd = end ;
}
2015-06-01 17:02:44 +02:00
void setWireframe ( bool wireframe )
{
if ( mWireframe ! = wireframe )
{
mWireframe = wireframe ;
reset ( ) ;
}
}
bool getWireframe ( ) const
{
return mWireframe ;
}
2015-04-14 15:55:56 +02:00
private :
osg : : Vec4f mAmbientColor ;
2015-04-21 23:27:26 +02:00
osg : : Vec4f mFogColor ;
2015-07-01 03:21:39 +02:00
float mFogStart ;
2015-04-21 23:27:26 +02:00
float mFogEnd ;
2015-06-01 17:02:44 +02:00
bool mWireframe ;
2015-04-14 15:55:56 +02:00
} ;
2016-02-09 01:17:02 +01:00
class PreloadCommonAssetsWorkItem : public SceneUtil : : WorkItem
{
public :
PreloadCommonAssetsWorkItem ( Resource : : ResourceSystem * resourceSystem )
: mResourceSystem ( resourceSystem )
{
}
2020-10-16 22:18:54 +04:00
void doWork ( ) override
2016-02-09 01:17:02 +01:00
{
2017-02-09 03:32:40 +01:00
try
{
for ( std : : vector < std : : string > : : const_iterator it = mModels . begin ( ) ; it ! = mModels . end ( ) ; + + it )
mResourceSystem - > getSceneManager ( ) - > cacheInstance ( * it ) ;
for ( std : : vector < std : : string > : : const_iterator it = mTextures . begin ( ) ; it ! = mTextures . end ( ) ; + + it )
mResourceSystem - > getImageManager ( ) - > getImage ( * it ) ;
for ( std : : vector < std : : string > : : const_iterator it = mKeyframes . begin ( ) ; it ! = mKeyframes . end ( ) ; + + it )
mResourceSystem - > getKeyframeManager ( ) - > get ( * it ) ;
}
catch ( std : : exception & )
{
// ignore error (will be shown when these are needed proper)
}
2016-02-09 01:17:02 +01:00
}
std : : vector < std : : string > mModels ;
std : : vector < std : : string > mTextures ;
2017-02-09 03:32:40 +01:00
std : : vector < std : : string > mKeyframes ;
2016-02-09 01:17:02 +01:00
private :
Resource : : ResourceSystem * mResourceSystem ;
} ;
2018-04-07 16:11:23 +03:00
RenderingManager : : RenderingManager ( osgViewer : : Viewer * viewer , osg : : ref_ptr < osg : : Group > rootNode ,
Resource : : ResourceSystem * resourceSystem , SceneUtil : : WorkQueue * workQueue ,
2019-01-22 10:08:48 +04:00
const std : : string & resourcePath , DetourNavigator : : Navigator & navigator )
2015-04-12 15:34:50 +02:00
: mViewer ( viewer )
, mRootNode ( rootNode )
, mResourceSystem ( resourceSystem )
2017-02-14 03:37:45 +01:00
, mWorkQueue ( workQueue )
2016-02-09 15:30:53 +01:00
, mUnrefQueue ( new SceneUtil : : UnrefQueue )
2018-04-07 16:11:23 +03:00
, mNavigator ( navigator )
2021-03-30 18:12:57 -07:00
, mMinimumAmbientLuminance ( 0.f )
2015-06-11 23:16:05 +02:00
, mNightEyeFactor ( 0.f )
2018-08-01 20:18:37 +04:00
, mFieldOfViewOverridden ( false )
2017-09-20 11:42:15 -07:00
, mFieldOfViewOverride ( 0.f )
2014-06-22 16:10:52 +02:00
{
2021-08-04 17:49:57 -07:00
bool reverseZ = SceneUtil : : getReverseZ ( ) ;
2021-06-01 12:15:25 -07:00
if ( reverseZ )
Log ( Debug : : Info ) < < " Using reverse-z depth buffer " ;
else
Log ( Debug : : Info ) < < " Using standard depth buffer " ;
2021-04-16 11:55:40 -07:00
auto lightingMethod = SceneUtil : : LightManager : : getLightingMethodFromString ( Settings : : Manager : : getString ( " lighting method " , " Shaders " ) ) ;
2020-04-20 18:47:14 +02:00
resourceSystem - > getSceneManager ( ) - > setParticleSystemMask ( MWRender : : Mask_ParticleSystem ) ;
2020-03-14 16:39:32 +04:00
// Shadows and radial fog have problems with fixed-function mode
2021-04-16 11:55:40 -07:00
bool forceShaders = Settings : : Manager : : getBool ( " radial fog " , " Shaders " )
| | Settings : : Manager : : getBool ( " force shaders " , " Shaders " )
| | Settings : : Manager : : getBool ( " enable shadows " , " Shadows " )
2021-07-09 10:05:27 -07:00
| | lightingMethod ! = SceneUtil : : LightingMethod : : FFP
| | reverseZ ;
2020-03-14 16:39:32 +04:00
resourceSystem - > getSceneManager ( ) - > setForceShaders ( forceShaders ) ;
2019-02-20 13:37:00 +00:00
// FIXME: calling dummy method because terrain needs to know whether lighting is clamped
2021-04-09 17:41:33 -07:00
resourceSystem - > getSceneManager ( ) - > setClampLighting ( Settings : : Manager : : getBool ( " clamp lighting " , " Shaders " ) ) ;
2016-02-20 17:57:19 +01:00
resourceSystem - > getSceneManager ( ) - > setAutoUseNormalMaps ( Settings : : Manager : : getBool ( " auto use object normal maps " , " Shaders " ) ) ;
resourceSystem - > getSceneManager ( ) - > setNormalMapPattern ( Settings : : Manager : : getString ( " normal map pattern " , " Shaders " ) ) ;
2016-03-22 21:00:31 +01:00
resourceSystem - > getSceneManager ( ) - > setNormalHeightMapPattern ( Settings : : Manager : : getString ( " normal height map pattern " , " Shaders " ) ) ;
2016-02-20 19:02:11 +01:00
resourceSystem - > getSceneManager ( ) - > setAutoUseSpecularMaps ( Settings : : Manager : : getBool ( " auto use object specular maps " , " Shaders " ) ) ;
resourceSystem - > getSceneManager ( ) - > setSpecularMapPattern ( Settings : : Manager : : getString ( " specular map pattern " , " Shaders " ) ) ;
2020-11-09 13:53:58 +03:00
resourceSystem - > getSceneManager ( ) - > setApplyLightingToEnvMaps ( Settings : : Manager : : getBool ( " apply lighting to environment maps " , " Shaders " ) ) ;
2021-02-24 18:01:06 +00:00
resourceSystem - > getSceneManager ( ) - > setConvertAlphaTestToAlphaToCoverage ( Settings : : Manager : : getBool ( " antialias alpha test " , " Shaders " ) & & Settings : : Manager : : getInt ( " antialiasing " , " Video " ) > 1 ) ;
2021-03-29 00:13:35 -07:00
2021-04-09 17:41:33 -07:00
// Let LightManager choose which backend to use based on our hint. For methods besides legacy lighting, this depends on support for various OpenGL extensions.
2021-04-16 11:55:40 -07:00
osg : : ref_ptr < SceneUtil : : LightManager > sceneRoot = new SceneUtil : : LightManager ( lightingMethod = = SceneUtil : : LightingMethod : : FFP ) ;
2021-03-14 21:42:34 -07:00
resourceSystem - > getSceneManager ( ) - > getShaderManager ( ) . setLightingMethod ( sceneRoot - > getLightingMethod ( ) ) ;
resourceSystem - > getSceneManager ( ) - > setLightingMethod ( sceneRoot - > getLightingMethod ( ) ) ;
2021-04-16 11:55:40 -07:00
resourceSystem - > getSceneManager ( ) - > setSupportedLightingMethods ( sceneRoot - > getSupportedLightingMethods ( ) ) ;
2021-04-03 22:25:13 -07:00
mMinimumAmbientLuminance = std : : clamp ( Settings : : Manager : : getFloat ( " minimum interior brightness " , " Shaders " ) , 0.f , 1.f ) ;
2021-03-30 18:12:57 -07:00
2020-04-20 18:47:14 +02:00
sceneRoot - > setLightingMask ( Mask_Lighting ) ;
2016-02-08 16:45:56 +01:00
mSceneRoot = sceneRoot ;
sceneRoot - > setStartLight ( 1 ) ;
2019-06-21 13:37:00 +00:00
sceneRoot - > setNodeMask ( Mask_Scene ) ;
sceneRoot - > setName ( " Scene Root " ) ;
2012-07-20 16:44:03 +02:00
2020-04-20 18:47:14 +02:00
int shadowCastingTraversalMask = Mask_Scene ;
2017-12-27 02:32:17 +00:00
if ( Settings : : Manager : : getBool ( " actor shadows " , " Shadows " ) )
2020-04-20 18:47:14 +02:00
shadowCastingTraversalMask | = Mask_Actor ;
2017-12-27 02:32:17 +00:00
if ( Settings : : Manager : : getBool ( " player shadows " , " Shadows " ) )
2020-04-20 18:47:14 +02:00
shadowCastingTraversalMask | = Mask_Player ;
2017-12-27 02:32:17 +00:00
if ( Settings : : Manager : : getBool ( " terrain shadows " , " Shadows " ) )
2020-04-20 18:47:14 +02:00
shadowCastingTraversalMask | = Mask_Terrain ;
2018-03-03 15:13:36 +00:00
int indoorShadowCastingTraversalMask = shadowCastingTraversalMask ;
if ( Settings : : Manager : : getBool ( " object shadows " , " Shadows " ) )
2020-04-20 18:47:14 +02:00
shadowCastingTraversalMask | = ( Mask_Object | Mask_Static ) ;
2019-02-27 19:41:07 +04:00
2021-08-04 17:49:57 -07:00
mShadowManager . reset ( new SceneUtil : : ShadowManager ( sceneRoot , mRootNode , shadowCastingTraversalMask , indoorShadowCastingTraversalMask , mResourceSystem - > getSceneManager ( ) - > getShaderManager ( ) ) ) ;
2017-04-14 15:32:18 +02:00
2018-02-26 22:27:09 +00:00
Shader : : ShaderManager : : DefineMap shadowDefines = mShadowManager - > getShadowDefines ( ) ;
2021-02-21 10:38:15 -08:00
Shader : : ShaderManager : : DefineMap lightDefines = sceneRoot - > getLightDefines ( ) ;
2017-11-22 20:07:07 +00:00
Shader : : ShaderManager : : DefineMap globalDefines = mResourceSystem - > getSceneManager ( ) - > getShaderManager ( ) . getGlobalDefines ( ) ;
2017-11-22 20:54:39 +00:00
for ( auto itr = shadowDefines . begin ( ) ; itr ! = shadowDefines . end ( ) ; itr + + )
globalDefines [ itr - > first ] = itr - > second ;
2019-02-20 13:37:00 +00:00
globalDefines [ " forcePPL " ] = Settings : : Manager : : getBool ( " force per pixel lighting " , " Shaders " ) ? " 1 " : " 0 " ;
globalDefines [ " clamp " ] = Settings : : Manager : : getBool ( " clamp lighting " , " Shaders " ) ? " 1 " : " 0 " ;
2020-03-03 20:08:59 +03:00
globalDefines [ " preLightEnv " ] = Settings : : Manager : : getBool ( " apply lighting to environment maps " , " Shaders " ) ? " 1 " : " 0 " ;
2020-03-14 16:39:32 +04:00
globalDefines [ " radialFog " ] = Settings : : Manager : : getBool ( " radial fog " , " Shaders " ) ? " 1 " : " 0 " ;
2021-01-07 18:13:51 +00:00
globalDefines [ " useGPUShader4 " ] = " 0 " ;
2019-02-20 13:37:00 +00:00
2021-03-14 21:42:34 -07:00
for ( auto itr = lightDefines . begin ( ) ; itr ! = lightDefines . end ( ) ; itr + + )
globalDefines [ itr - > first ] = itr - > second ;
2019-02-20 13:37:00 +00:00
2021-04-18 21:44:23 +01:00
// Refactor this at some point - most shaders don't care about these defines
2021-05-12 09:18:54 +04:00
float groundcoverDistance = std : : max ( 0.f , Settings : : Manager : : getFloat ( " rendering distance " , " Groundcover " ) ) ;
2021-01-11 09:59:57 +04:00
globalDefines [ " groundcoverFadeStart " ] = std : : to_string ( groundcoverDistance * 0.9f ) ;
2020-01-12 11:42:47 +04:00
globalDefines [ " groundcoverFadeEnd " ] = std : : to_string ( groundcoverDistance ) ;
2021-04-18 21:44:23 +01:00
globalDefines [ " groundcoverStompMode " ] = std : : to_string ( std : : clamp ( Settings : : Manager : : getInt ( " stomp mode " , " Groundcover " ) , 0 , 2 ) ) ;
globalDefines [ " groundcoverStompIntensity " ] = std : : to_string ( std : : clamp ( Settings : : Manager : : getInt ( " stomp intensity " , " Groundcover " ) , 0 , 2 ) ) ;
2021-03-29 00:13:35 -07:00
2021-06-01 12:15:25 -07:00
globalDefines [ " reverseZ " ] = reverseZ ? " 1 " : " 0 " ;
2017-12-17 23:55:19 +00:00
// It is unnecessary to stop/start the viewer as no frames are being rendered yet.
mResourceSystem - > getSceneManager ( ) - > getShaderManager ( ) . setGlobalDefines ( globalDefines ) ;
2011-11-03 23:47:15 -04:00
2018-07-21 13:37:02 +03:00
mNavMesh . reset ( new NavMesh ( mRootNode , Settings : : Manager : : getBool ( " enable nav mesh render " , " Navigator " ) ) ) ;
mActorsPaths . reset ( new ActorsPaths ( mRootNode , Settings : : Manager : : getBool ( " enable agents paths render " , " Navigator " ) ) ) ;
2019-11-27 23:45:01 +01:00
mRecastMesh . reset ( new RecastMesh ( mRootNode , Settings : : Manager : : getBool ( " enable recast mesh render " , " Navigator " ) ) ) ;
2015-05-04 17:41:30 +02:00
mPathgrid . reset ( new Pathgrid ( mRootNode ) ) ;
2015-05-02 22:45:27 +02:00
2016-02-08 16:45:56 +01:00
mObjects . reset ( new Objects ( mResourceSystem , sceneRoot , mUnrefQueue . get ( ) ) ) ;
2012-02-26 13:13:29 +01:00
2018-10-09 10:21:12 +04:00
if ( getenv ( " OPENMW_DONT_PRECOMPILE " ) = = nullptr )
2018-02-13 00:40:41 +00:00
{
2017-02-15 20:57:12 +01:00
mViewer - > setIncrementalCompileOperation ( new osgUtil : : IncrementalCompileOperation ) ;
2018-02-13 00:40:41 +00:00
mViewer - > getIncrementalCompileOperation ( ) - > setTargetFrameRate ( Settings : : Manager : : getFloat ( " target framerate " , " Cells " ) ) ;
}
2015-05-03 18:16:54 +02:00
2015-05-26 20:20:18 +02:00
mResourceSystem - > getSceneManager ( ) - > setIncrementalCompileOperation ( mViewer - > getIncrementalCompileOperation ( ) ) ;
2015-05-03 18:16:54 +02:00
2016-02-08 16:45:56 +01:00
mEffectManager . reset ( new EffectManager ( sceneRoot , mResourceSystem ) ) ;
2015-06-02 16:35:35 +02:00
2019-02-27 19:41:07 +04:00
const std : : string normalMapPattern = Settings : : Manager : : getString ( " normal map pattern " , " Shaders " ) ;
const std : : string heightMapPattern = Settings : : Manager : : getString ( " normal height map pattern " , " Shaders " ) ;
const std : : string specularMapPattern = Settings : : Manager : : getString ( " terrain specular map pattern " , " Shaders " ) ;
const bool useTerrainNormalMaps = Settings : : Manager : : getBool ( " auto use terrain normal maps " , " Shaders " ) ;
const bool useTerrainSpecularMaps = Settings : : Manager : : getBool ( " auto use terrain specular maps " , " Shaders " ) ;
2020-01-12 11:42:47 +04:00
mTerrainStorage . reset ( new TerrainStorage ( mResourceSystem , normalMapPattern , heightMapPattern , useTerrainNormalMaps , specularMapPattern , useTerrainSpecularMaps ) ) ;
const float lodFactor = Settings : : Manager : : getFloat ( " lod factor " , " Terrain " ) ;
2017-03-09 03:49:40 +01:00
2021-09-09 21:10:22 +00:00
bool groundcover = Settings : : Manager : : getBool ( " enabled " , " Groundcover " ) ;
bool distantTerrain = Settings : : Manager : : getBool ( " distant terrain " , " Terrain " ) ;
if ( distantTerrain | | groundcover )
2019-02-27 19:41:07 +04:00
{
const int compMapResolution = Settings : : Manager : : getInt ( " composite map resolution " , " Terrain " ) ;
int compMapPower = Settings : : Manager : : getInt ( " composite map level " , " Terrain " ) ;
compMapPower = std : : max ( - 3 , compMapPower ) ;
float compMapLevel = pow ( 2 , compMapPower ) ;
2019-02-20 13:37:00 +00:00
const int vertexLodMod = Settings : : Manager : : getInt ( " vertex lod mod " , " Terrain " ) ;
2019-02-28 12:48:04 +04:00
float maxCompGeometrySize = Settings : : Manager : : getFloat ( " max composite geometry size " , " Terrain " ) ;
maxCompGeometrySize = std : : max ( maxCompGeometrySize , 1.f ) ;
mTerrain . reset ( new Terrain : : QuadTreeWorld (
2020-01-12 11:42:47 +04:00
sceneRoot , mRootNode , mResourceSystem , mTerrainStorage . get ( ) , Mask_Terrain , Mask_PreCompile , Mask_Debug ,
2020-04-20 18:47:14 +02:00
compMapResolution , compMapLevel , lodFactor , vertexLodMod , maxCompGeometrySize ) ) ;
2019-06-13 13:37:00 +00:00
if ( Settings : : Manager : : getBool ( " object paging " , " Terrain " ) )
{
mObjectPaging . reset ( new ObjectPaging ( mResourceSystem - > getSceneManager ( ) ) ) ;
static_cast < Terrain : : QuadTreeWorld * > ( mTerrain . get ( ) ) - > addChunkManager ( mObjectPaging . get ( ) ) ;
mResourceSystem - > addResourceManager ( mObjectPaging . get ( ) ) ;
}
2019-02-27 19:41:07 +04:00
}
2017-03-09 03:49:40 +01:00
else
2020-01-12 11:42:47 +04:00
mTerrain . reset ( new Terrain : : TerrainGrid ( sceneRoot , mRootNode , mResourceSystem , mTerrainStorage . get ( ) , Mask_Terrain , Mask_PreCompile , Mask_Debug ) ) ;
2018-06-13 01:48:31 +02:00
2018-02-13 00:40:41 +00:00
mTerrain - > setTargetFrameRate ( Settings : : Manager : : getFloat ( " target framerate " , " Cells " ) ) ;
2019-02-20 13:37:00 +00:00
mTerrain - > setWorkQueue ( mWorkQueue . get ( ) ) ;
2015-04-19 17:55:56 +02:00
2021-09-09 21:10:22 +00:00
osg : : ref_ptr < SceneUtil : : CompositeStateSetUpdater > composite = new SceneUtil : : CompositeStateSetUpdater ;
2020-01-12 11:42:47 +04:00
2021-09-09 21:10:22 +00:00
if ( groundcover )
{
2020-01-12 11:42:47 +04:00
float density = Settings : : Manager : : getFloat ( " density " , " Groundcover " ) ;
density = std : : clamp ( density , 0.f , 1.f ) ;
2021-09-09 21:10:22 +00:00
mGroundcoverUpdater = new GroundcoverUpdater ;
composite - > addController ( mGroundcoverUpdater ) ;
2020-01-12 11:42:47 +04:00
mGroundcover . reset ( new Groundcover ( mResourceSystem - > getSceneManager ( ) , density ) ) ;
2021-09-09 21:10:22 +00:00
static_cast < Terrain : : QuadTreeWorld * > ( mTerrain . get ( ) ) - > addChunkManager ( mGroundcover . get ( ) ) ;
2020-01-12 11:42:47 +04:00
mResourceSystem - > addResourceManager ( mGroundcover . get ( ) ) ;
2021-05-12 09:18:54 +04:00
2021-09-09 21:10:22 +00:00
float groundcoverDistance = std : : max ( 0.f , Settings : : Manager : : getFloat ( " rendering distance " , " Groundcover " ) ) ;
mGroundcover - > setViewDistance ( groundcoverDistance ) ;
2020-01-12 11:42:47 +04:00
}
2021-07-09 10:05:27 -07:00
2021-08-17 17:45:50 -07:00
mStateUpdater = new StateUpdater ;
2021-09-09 21:10:22 +00:00
composite - > addController ( mStateUpdater ) ;
sceneRoot - > addUpdateCallback ( composite ) ;
2021-08-17 17:45:50 -07:00
mSharedUniformStateUpdater = new SharedUniformStateUpdater ;
rootNode - > addUpdateCallback ( mSharedUniformStateUpdater ) ;
2021-08-04 17:49:57 -07:00
mPostProcessor = new PostProcessor ( * this , viewer , mRootNode ) ;
2021-07-09 10:05:27 -07:00
resourceSystem - > getSceneManager ( ) - > setDepthFormat ( mPostProcessor - > getDepthFormat ( ) ) ;
2021-08-04 17:49:57 -07:00
if ( reverseZ & & ! SceneUtil : : isFloatingPointDepthFormat ( mPostProcessor - > getDepthFormat ( ) ) )
Log ( Debug : : Warning ) < < " Floating point depth format not in use but reverse-z buffer is enabled, consider disabling it. " ;
2019-06-13 13:37:00 +00:00
// water goes after terrain for correct waterculling order
2020-12-02 23:03:10 +00:00
mWater . reset ( new Water ( sceneRoot - > getParent ( 0 ) , sceneRoot , mResourceSystem , mViewer - > getIncrementalCompileOperation ( ) , resourcePath ) ) ;
2019-06-13 13:37:00 +00:00
2015-05-21 23:54:39 +02:00
mCamera . reset ( new Camera ( mViewer - > getCamera ( ) ) ) ;
2020-07-11 14:21:18 +02:00
if ( Settings : : Manager : : getBool ( " view over shoulder " , " Camera " ) )
mViewOverShoulderController . reset ( new ViewOverShoulderController ( mCamera . get ( ) ) ) ;
2015-05-21 23:54:39 +02:00
2021-01-08 19:33:51 +04:00
mScreenshotManager . reset ( new ScreenshotManager ( viewer , mRootNode , sceneRoot , mResourceSystem , mWater . get ( ) ) ) ;
2015-05-14 21:42:04 +02:00
mViewer - > setLightingMode ( osgViewer : : View : : NO_LIGHT ) ;
2011-11-03 23:47:15 -04:00
2015-04-12 15:34:50 +02:00
osg : : ref_ptr < osg : : LightSource > source = new osg : : LightSource ;
2020-04-20 18:47:14 +02:00
source - > setNodeMask ( Mask_Lighting ) ;
2015-04-12 15:34:50 +02:00
mSunLight = new osg : : Light ;
source - > setLight ( mSunLight ) ;
mSunLight - > setDiffuse ( osg : : Vec4f ( 0 , 0 , 0 , 1 ) ) ;
mSunLight - > setAmbient ( osg : : Vec4f ( 0 , 0 , 0 , 1 ) ) ;
2015-05-26 18:22:21 +02:00
mSunLight - > setSpecular ( osg : : Vec4f ( 0 , 0 , 0 , 0 ) ) ;
2015-04-12 15:34:50 +02:00
mSunLight - > setConstantAttenuation ( 1.f ) ;
2021-02-21 10:38:15 -08:00
sceneRoot - > setSunlight ( mSunLight ) ;
2016-02-08 16:45:56 +01:00
sceneRoot - > addChild ( source ) ;
2011-11-03 23:47:15 -04:00
2016-02-08 16:45:56 +01:00
sceneRoot - > getOrCreateStateSet ( ) - > setMode ( GL_CULL_FACE , osg : : StateAttribute : : ON ) ;
sceneRoot - > getOrCreateStateSet ( ) - > setMode ( GL_LIGHTING , osg : : StateAttribute : : ON ) ;
sceneRoot - > getOrCreateStateSet ( ) - > setMode ( GL_NORMALIZE , osg : : StateAttribute : : ON ) ;
2017-02-01 03:18:17 +01:00
osg : : ref_ptr < osg : : Material > defaultMat ( new osg : : Material ) ;
defaultMat - > setColorMode ( osg : : Material : : OFF ) ;
defaultMat - > setAmbient ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 1 , 1 , 1 , 1 ) ) ;
defaultMat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 1 , 1 , 1 , 1 ) ) ;
defaultMat - > setSpecular ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 0.f ) ) ;
sceneRoot - > getOrCreateStateSet ( ) - > setAttribute ( defaultMat ) ;
2015-04-14 15:55:56 +02:00
2020-06-07 10:35:33 +04:00
mFog . reset ( new FogManager ( ) ) ;
2017-10-24 14:12:41 +02:00
2020-06-07 10:35:33 +04:00
mSky . reset ( new SkyManager ( sceneRoot , resourceSystem - > getSceneManager ( ) ) ) ;
2017-10-14 18:45:29 +02:00
mSky - > setCamera ( mViewer - > getCamera ( ) ) ;
2015-04-30 23:21:25 +02:00
2015-04-14 15:55:56 +02:00
source - > setStateSetModes ( * mRootNode - > getOrCreateStateSet ( ) , osg : : StateAttribute : : ON ) ;
2015-04-21 23:27:26 +02:00
osg : : Camera : : CullingMode cullingMode = osg : : Camera : : DEFAULT_CULLING | osg : : Camera : : FAR_PLANE_CULLING ;
2011-11-03 23:47:15 -04:00
2015-05-24 02:34:20 +02:00
if ( ! Settings : : Manager : : getBool ( " small feature culling " , " Camera " ) )
2015-05-14 18:46:04 +02:00
cullingMode & = ~ ( osg : : CullStack : : SMALL_FEATURE_CULLING ) ;
else
2017-02-01 05:38:59 +01:00
{
mViewer - > getCamera ( ) - > setSmallFeatureCullingPixelSize ( Settings : : Manager : : getFloat ( " small feature culling pixel size " , " Camera " ) ) ;
2015-05-14 18:46:04 +02:00
cullingMode | = osg : : CullStack : : SMALL_FEATURE_CULLING ;
2017-02-01 05:38:59 +01:00
}
2015-04-21 23:27:26 +02:00
2015-05-14 21:42:04 +02:00
mViewer - > getCamera ( ) - > setCullingMode ( cullingMode ) ;
2015-04-14 15:55:56 +02:00
2015-05-14 21:42:04 +02:00
mViewer - > getCamera ( ) - > setComputeNearFarMode ( osg : : Camera : : DO_NOT_COMPUTE_NEAR_FAR ) ;
mViewer - > getCamera ( ) - > setCullingMode ( cullingMode ) ;
2015-04-21 23:27:26 +02:00
2020-04-20 18:47:14 +02:00
mViewer - > getCamera ( ) - > setCullMask ( ~ ( Mask_UpdateVisitor | Mask_SimpleWater ) ) ;
2020-04-22 13:34:19 +02:00
NifOsg : : Loader : : setHiddenNodeMask ( Mask_UpdateVisitor ) ;
2020-04-29 16:20:03 +03:00
NifOsg : : Loader : : setIntersectionDisabledNodeMask ( Mask_Effect ) ;
2020-11-09 14:22:48 +03:00
Nif : : NIFFile : : setLoadUnsupportedFiles ( Settings : : Manager : : getBool ( " load unsupported nif files " , " Models " ) ) ;
2015-05-20 03:35:52 +02:00
2021-07-09 10:05:27 -07:00
// TODO: Near clip should not need to be bounded like this, but too small values break OSG shadow calculations CPU-side.
// See issue: #6072
mNearClip = std : : max ( 0.005f , Settings : : Manager : : getFloat ( " near clip " , " Camera " ) ) ;
2015-05-24 02:34:20 +02:00
mViewDistance = Settings : : Manager : : getFloat ( " viewing distance " , " Camera " ) ;
2019-03-03 21:29:51 +04:00
float fov = Settings : : Manager : : getFloat ( " field of view " , " Camera " ) ;
mFieldOfView = std : : min ( std : : max ( 1.f , fov ) , 179.f ) ;
float firstPersonFov = Settings : : Manager : : getFloat ( " first person field of view " , " Camera " ) ;
mFirstPersonFieldOfView = std : : min ( std : : max ( 1.f , firstPersonFov ) , 179.f ) ;
2015-06-02 16:35:35 +02:00
mStateUpdater - > setFogEnd ( mViewDistance ) ;
2015-10-28 18:59:35 +01:00
2020-07-27 00:29:35 +03:00
mRootNode - > getOrCreateStateSet ( ) - > addUniform ( new osg : : Uniform ( " simpleWater " , false ) ) ;
2017-09-26 14:14:28 +02:00
2020-12-18 00:02:51 +00:00
// Hopefully, anything genuinely requiring the default alpha func of GL_ALWAYS explicitly sets it
2020-12-18 16:31:21 +00:00
mRootNode - > getOrCreateStateSet ( ) - > setAttribute ( Shader : : RemovedAlphaFunc : : getInstance ( GL_ALWAYS ) ) ;
2021-01-02 02:50:19 +00:00
// The transparent renderbin sets alpha testing on because that was faster on old GPUs. It's now slower and breaks things.
mRootNode - > getOrCreateStateSet ( ) - > setMode ( GL_ALPHA_TEST , osg : : StateAttribute : : OFF ) ;
2020-12-18 00:02:51 +00:00
2021-06-01 12:15:25 -07:00
if ( reverseZ )
{
osg : : ref_ptr < osg : : ClipControl > clipcontrol = new osg : : ClipControl ( osg : : ClipControl : : LOWER_LEFT , osg : : ClipControl : : ZERO_TO_ONE ) ;
2021-08-04 17:49:57 -07:00
mRootNode - > getOrCreateStateSet ( ) - > setAttributeAndModes ( SceneUtil : : createDepth ( ) , osg : : StateAttribute : : ON ) ;
2021-06-01 12:15:25 -07:00
mRootNode - > getOrCreateStateSet ( ) - > setAttributeAndModes ( clipcontrol , osg : : StateAttribute : : ON ) ;
}
2021-08-04 17:49:57 -07:00
SceneUtil : : setCameraClearDepth ( mViewer - > getCamera ( ) ) ;
2017-09-26 14:14:28 +02:00
updateProjectionMatrix ( ) ;
2015-04-14 15:55:56 +02:00
}
RenderingManager : : ~ RenderingManager ( )
{
2016-08-14 18:10:29 +02:00
// let background loading thread finish before we delete anything else
2018-10-09 10:21:12 +04:00
mWorkQueue = nullptr ;
2013-02-03 15:46:23 +01:00
}
2012-02-24 16:12:43 +01:00
2019-11-24 17:40:19 +04:00
osgUtil : : IncrementalCompileOperation * RenderingManager : : getIncrementalCompileOperation ( )
{
return mViewer - > getIncrementalCompileOperation ( ) ;
}
2015-04-12 15:34:50 +02:00
MWRender : : Objects & RenderingManager : : getObjects ( )
2014-03-05 21:45:43 +01:00
{
2015-04-12 15:34:50 +02:00
return * mObjects . get ( ) ;
2014-03-05 21:45:43 +01:00
}
2014-02-18 16:44:37 +01:00
2015-04-12 15:34:50 +02:00
Resource : : ResourceSystem * RenderingManager : : getResourceSystem ( )
2012-05-14 21:37:43 +02:00
{
2015-04-12 15:34:50 +02:00
return mResourceSystem ;
2012-05-14 21:37:43 +02:00
}
2016-02-09 20:57:30 +01:00
SceneUtil : : WorkQueue * RenderingManager : : getWorkQueue ( )
2016-02-09 01:02:40 +01:00
{
return mWorkQueue . get ( ) ;
}
2016-02-09 20:57:30 +01:00
SceneUtil : : UnrefQueue * RenderingManager : : getUnrefQueue ( )
2016-02-09 19:04:59 +01:00
{
return mUnrefQueue . get ( ) ;
}
2016-02-09 20:57:30 +01:00
Terrain : : World * RenderingManager : : getTerrain ( )
{
return mTerrain . get ( ) ;
}
2016-02-09 01:17:02 +01:00
void RenderingManager : : preloadCommonAssets ( )
{
osg : : ref_ptr < PreloadCommonAssetsWorkItem > workItem ( new PreloadCommonAssetsWorkItem ( mResourceSystem ) ) ;
mSky - > listAssetsToPreload ( workItem - > mModels , workItem - > mTextures ) ;
mWater - > listAssetsToPreload ( workItem - > mTextures ) ;
2020-12-30 22:11:32 +02:00
workItem - > mModels . push_back ( Settings : : Manager : : getString ( " xbaseanim " , " Models " ) ) ;
workItem - > mModels . push_back ( Settings : : Manager : : getString ( " xbaseanim1st " , " Models " ) ) ;
workItem - > mModels . push_back ( Settings : : Manager : : getString ( " xbaseanimfemale " , " Models " ) ) ;
workItem - > mModels . push_back ( Settings : : Manager : : getString ( " xargonianswimkna " , " Models " ) ) ;
workItem - > mKeyframes . push_back ( Settings : : Manager : : getString ( " xbaseanimkf " , " Models " ) ) ;
workItem - > mKeyframes . push_back ( Settings : : Manager : : getString ( " xbaseanim1stkf " , " Models " ) ) ;
workItem - > mKeyframes . push_back ( Settings : : Manager : : getString ( " xbaseanimfemalekf " , " Models " ) ) ;
workItem - > mKeyframes . push_back ( Settings : : Manager : : getString ( " xargonianswimknakf " , " Models " ) ) ;
2017-02-09 03:32:40 +01:00
2020-10-17 12:26:35 +04:00
workItem - > mTextures . emplace_back ( " textures/_land_default.dds " ) ;
2016-02-09 01:17:02 +01:00
2021-07-10 12:11:48 +02:00
mWorkQueue - > addWorkItem ( std : : move ( workItem ) ) ;
2016-02-09 01:17:02 +01:00
}
2016-02-07 00:36:31 +01:00
double RenderingManager : : getReferenceTime ( ) const
{
return mViewer - > getFrameStamp ( ) - > getReferenceTime ( ) ;
}
2015-12-05 00:44:04 +01:00
osg : : Group * RenderingManager : : getLightRoot ( )
{
2016-02-08 16:45:56 +01:00
return mSceneRoot . get ( ) ;
2015-12-05 00:44:04 +01:00
}
2015-06-11 23:16:05 +02:00
void RenderingManager : : setNightEyeFactor ( float factor )
{
if ( factor ! = mNightEyeFactor )
{
mNightEyeFactor = factor ;
updateAmbient ( ) ;
}
}
2015-04-14 15:55:56 +02:00
void RenderingManager : : setAmbientColour ( const osg : : Vec4f & colour )
{
2015-06-11 23:16:05 +02:00
mAmbientColor = colour ;
updateAmbient ( ) ;
2015-04-14 15:55:56 +02:00
}
2015-06-16 20:56:48 +02:00
void RenderingManager : : skySetDate ( int day , int month )
{
mSky - > setDate ( day , month ) ;
}
int RenderingManager : : skyGetMasserPhase ( ) const
{
return mSky - > getMasserPhase ( ) ;
}
int RenderingManager : : skyGetSecundaPhase ( ) const
{
return mSky - > getSecundaPhase ( ) ;
}
void RenderingManager : : skySetMoonColour ( bool red )
{
mSky - > setMoonColour ( red ) ;
}
2015-04-12 15:34:50 +02:00
void RenderingManager : : configureAmbient ( const ESM : : Cell * cell )
2012-05-23 01:32:36 +02:00
{
2021-03-31 22:35:02 -07:00
bool needsAdjusting = false ;
if ( mResourceSystem - > getSceneManager ( ) - > getLightingMethod ( ) ! = SceneUtil : : LightingMethod : : FFP )
needsAdjusting = ! cell - > isExterior ( ) & & ! ( cell - > mData . mFlags & ESM : : Cell : : QuasiEx ) ;
auto ambient = SceneUtil : : colourFromRGB ( cell - > mAmbi . mAmbient ) ;
if ( needsAdjusting )
{
2021-04-09 17:41:33 -07:00
constexpr float pR = 0.2126 ;
constexpr float pG = 0.7152 ;
constexpr float pB = 0.0722 ;
2021-03-31 22:35:02 -07:00
// we already work in linear RGB so no conversions are needed for the luminosity function
float relativeLuminance = pR * ambient . r ( ) + pG * ambient . g ( ) + pB * ambient . b ( ) ;
if ( relativeLuminance < mMinimumAmbientLuminance )
{
// brighten ambient so it reaches the minimum threshold but no more, we want to mess with content data as least we can
float targetBrightnessIncreaseFactor = mMinimumAmbientLuminance / relativeLuminance ;
if ( ambient . r ( ) = = 0.f & & ambient . g ( ) = = 0.f & & ambient . b ( ) = = 0.f )
2021-04-16 11:55:40 -07:00
ambient = osg : : Vec4 ( mMinimumAmbientLuminance , mMinimumAmbientLuminance , mMinimumAmbientLuminance , ambient . a ( ) ) ;
2021-03-31 22:35:02 -07:00
else
ambient * = targetBrightnessIncreaseFactor ;
}
}
setAmbientColour ( ambient ) ;
2012-05-29 04:54:54 +02:00
2016-02-20 19:02:11 +01:00
osg : : Vec4f diffuse = SceneUtil : : colourFromRGB ( cell - > mAmbi . mSunlight ) ;
mSunLight - > setDiffuse ( diffuse ) ;
mSunLight - > setSpecular ( diffuse ) ;
2018-08-08 23:30:33 +01:00
mSunLight - > setPosition ( osg : : Vec4f ( - 0.15f , 0.15f , 1.f , 0.f ) ) ;
2012-05-27 21:39:18 +02:00
}
2016-05-10 17:39:57 +02:00
void RenderingManager : : setSunColour ( const osg : : Vec4f & diffuse , const osg : : Vec4f & specular )
2015-04-14 15:55:56 +02:00
{
2015-05-26 18:22:21 +02:00
// need to wrap this in a StateUpdater?
2016-05-10 17:39:57 +02:00
mSunLight - > setDiffuse ( diffuse ) ;
mSunLight - > setSpecular ( specular ) ;
2015-04-14 15:55:56 +02:00
}
void RenderingManager : : setSunDirection ( const osg : : Vec3f & direction )
{
2015-05-20 02:07:18 +02:00
osg : : Vec3 position = direction * - 1 ;
2015-05-26 18:22:21 +02:00
// need to wrap this in a StateUpdater?
2015-05-20 02:07:18 +02:00
mSunLight - > setPosition ( osg : : Vec4 ( position . x ( ) , position . y ( ) , position . z ( ) , 0 ) ) ;
2015-04-14 15:55:56 +02:00
2015-05-20 02:07:18 +02:00
mSky - > setSunDirection ( position ) ;
2015-04-14 15:55:56 +02:00
}
2015-05-02 22:45:27 +02:00
void RenderingManager : : addCell ( const MWWorld : : CellStore * store )
{
2015-05-04 17:41:30 +02:00
mPathgrid - > addCell ( store ) ;
2015-06-02 16:35:35 +02:00
mWater - > changeCell ( store ) ;
2015-06-03 01:18:36 +02:00
if ( store - > getCell ( ) - > isExterior ( ) )
2020-01-12 11:42:47 +04:00
{
2015-06-03 01:18:36 +02:00
mTerrain - > loadCell ( store - > getCell ( ) - > getGridX ( ) , store - > getCell ( ) - > getGridY ( ) ) ;
2020-01-12 11:42:47 +04:00
}
2015-05-02 22:45:27 +02:00
}
2015-04-12 15:34:50 +02:00
void RenderingManager : : removeCell ( const MWWorld : : CellStore * store )
2013-08-20 09:52:27 +02:00
{
2015-05-04 17:41:30 +02:00
mPathgrid - > removeCell ( store ) ;
2018-07-21 13:37:02 +03:00
mActorsPaths - > removeCell ( store ) ;
2015-04-12 15:34:50 +02:00
mObjects - > removeCell ( store ) ;
2015-06-03 01:18:36 +02:00
if ( store - > getCell ( ) - > isExterior ( ) )
2020-01-12 11:42:47 +04:00
{
2015-06-03 01:18:36 +02:00
mTerrain - > unloadCell ( store - > getCell ( ) - > getGridX ( ) , store - > getCell ( ) - > getGridY ( ) ) ;
2020-01-12 11:42:47 +04:00
}
2015-06-16 20:36:48 +02:00
mWater - > removeCell ( store ) ;
2013-08-20 09:52:27 +02:00
}
2012-05-23 01:32:36 +02:00
2017-03-09 02:17:25 +01:00
void RenderingManager : : enableTerrain ( bool enable )
{
2019-06-13 13:37:00 +00:00
if ( ! enable )
mWater - > setCullCallback ( nullptr ) ;
2017-03-09 02:17:25 +01:00
mTerrain - > enable ( enable ) ;
}
2015-04-14 15:55:56 +02:00
void RenderingManager : : setSkyEnabled ( bool enabled )
{
mSky - > setEnabled ( enabled ) ;
2018-02-27 00:13:51 +00:00
if ( enabled )
mShadowManager - > enableOutdoorMode ( ) ;
else
mShadowManager - > enableIndoorMode ( ) ;
2015-04-14 15:55:56 +02:00
}
2018-04-07 16:11:23 +03:00
2018-06-13 01:48:31 +02:00
bool RenderingManager : : toggleBorders ( )
{
2020-06-06 14:10:24 +04:00
bool borders = ! mTerrain - > getBordersVisible ( ) ;
mTerrain - > setBordersVisible ( borders ) ;
return borders ;
2018-06-13 01:48:31 +02:00
}
2015-04-14 15:55:56 +02:00
2015-05-02 22:45:27 +02:00
bool RenderingManager : : toggleRenderMode ( RenderMode mode )
{
if ( mode = = Render_CollisionDebug | | mode = = Render_Pathgrid )
2015-05-04 17:41:30 +02:00
return mPathgrid - > toggleRenderMode ( mode ) ;
2015-05-02 22:45:27 +02:00
else if ( mode = = Render_Wireframe )
{
2015-06-01 17:02:44 +02:00
bool wireframe = ! mStateUpdater - > getWireframe ( ) ;
mStateUpdater - > setWireframe ( wireframe ) ;
return wireframe ;
2015-05-02 22:45:27 +02:00
}
2015-06-02 16:35:35 +02:00
else if ( mode = = Render_Water )
{
return mWater - > toggle ( ) ;
}
else if ( mode = = Render_Scene )
{
2021-04-20 10:42:06 +04:00
unsigned int mask = mViewer - > getCamera ( ) - > getCullMask ( ) ;
2020-04-20 18:47:14 +02:00
bool enabled = mask & Mask_Scene ;
2015-06-02 16:35:35 +02:00
enabled = ! enabled ;
if ( enabled )
2020-04-20 18:47:14 +02:00
mask | = Mask_Scene ;
2015-06-02 16:35:35 +02:00
else
2020-04-20 18:47:14 +02:00
mask & = ~ Mask_Scene ;
2015-06-02 16:35:35 +02:00
mViewer - > getCamera ( ) - > setCullMask ( mask ) ;
return enabled ;
}
2018-04-07 16:11:23 +03:00
else if ( mode = = Render_NavMesh )
{
return mNavMesh - > toggle ( ) ;
}
2018-07-21 13:37:02 +03:00
else if ( mode = = Render_ActorsPaths )
{
return mActorsPaths - > toggle ( ) ;
}
2019-11-27 23:45:01 +01:00
else if ( mode = = Render_RecastMesh )
{
return mRecastMesh - > toggle ( ) ;
}
2015-05-02 22:45:27 +02:00
return false ;
}
2015-04-21 23:27:26 +02:00
void RenderingManager : : configureFog ( const ESM : : Cell * cell )
{
2020-06-07 10:35:33 +04:00
mFog - > configure ( mViewDistance , cell ) ;
2015-04-21 23:27:26 +02:00
}
2017-09-21 03:08:45 -07:00
void RenderingManager : : configureFog ( float fogDepth , float underwaterFog , float dlFactor , float dlOffset , const osg : : Vec4f & color )
2015-04-14 15:55:56 +02:00
{
2020-06-07 10:35:33 +04:00
mFog - > configure ( mViewDistance , fogDepth , underwaterFog , dlFactor , dlOffset , color ) ;
2015-04-14 15:55:56 +02:00
}
SkyManager * RenderingManager : : getSkyManager ( )
{
return mSky . get ( ) ;
}
2015-04-19 01:57:52 +02:00
void RenderingManager : : update ( float dt , bool paused )
{
2017-02-22 02:18:18 +01:00
reportStats ( ) ;
2016-02-09 15:30:53 +01:00
mUnrefQueue - > flush ( mWorkQueue . get ( ) ) ;
2021-03-10 19:10:17 +04:00
float rainIntensity = mSky - > getPrecipitationAlpha ( ) ;
2021-03-08 10:58:51 +04:00
mWater - > setRainIntensity ( rainIntensity ) ;
2015-06-19 03:32:48 +02:00
if ( ! paused )
2015-06-20 17:08:46 +02:00
{
2015-06-19 03:32:48 +02:00
mEffectManager - > update ( dt ) ;
2015-06-20 17:08:46 +02:00
mSky - > update ( dt ) ;
2016-10-31 01:19:38 +09:00
mWater - > update ( dt ) ;
2020-01-12 11:42:47 +04:00
if ( mGroundcoverUpdater )
{
const MWWorld : : Ptr & player = mPlayerAnimation - > getPtr ( ) ;
osg : : Vec3f playerPos ( player . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
float windSpeed = mSky - > getBaseWindSpeed ( ) ;
mGroundcoverUpdater - > setWindSpeed ( windSpeed ) ;
mGroundcoverUpdater - > setPlayerPos ( playerPos ) ;
}
2015-06-20 17:08:46 +02:00
}
2015-06-19 03:32:48 +02:00
2019-01-20 19:27:52 +03:00
updateNavMesh ( ) ;
2019-11-27 23:45:01 +01:00
updateRecastMesh ( ) ;
2018-08-31 01:39:44 +03:00
2020-07-11 14:21:18 +02:00
if ( mViewOverShoulderController )
mViewOverShoulderController - > update ( ) ;
2015-05-22 01:43:16 +02:00
mCamera - > update ( dt , paused ) ;
2015-06-02 16:35:35 +02:00
2020-06-22 01:54:08 +02:00
osg : : Vec3d focal , cameraPos ;
2015-06-02 16:35:35 +02:00
mCamera - > getPosition ( focal , cameraPos ) ;
2015-12-03 15:16:20 +01:00
mCurrentCameraPos = cameraPos ;
2015-08-19 01:13:14 +02:00
2020-06-07 10:35:33 +04:00
bool isUnderwater = mWater - > isUnderwater ( cameraPos ) ;
mStateUpdater - > setFogStart ( mFog - > getFogStart ( isUnderwater ) ) ;
mStateUpdater - > setFogEnd ( mFog - > getFogEnd ( isUnderwater ) ) ;
setFogColor ( mFog - > getFogColor ( isUnderwater ) ) ;
2015-04-19 17:55:56 +02:00
}
2015-05-21 23:54:39 +02:00
void RenderingManager : : updatePlayerPtr ( const MWWorld : : Ptr & ptr )
{
if ( mPlayerAnimation . get ( ) )
2017-09-17 23:07:17 +00:00
{
setupPlayer ( ptr ) ;
2015-05-21 23:54:39 +02:00
mPlayerAnimation - > updatePtr ( ptr ) ;
2017-09-17 23:07:17 +00:00
}
2015-05-21 23:54:39 +02:00
mCamera - > attachTo ( ptr ) ;
}
2015-06-16 20:36:48 +02:00
void RenderingManager : : removePlayer ( const MWWorld : : Ptr & player )
{
mWater - > removeEmitter ( player ) ;
}
2015-04-23 23:50:46 +02:00
void RenderingManager : : rotateObject ( const MWWorld : : Ptr & ptr , const osg : : Quat & rot )
{
2015-05-21 23:54:39 +02:00
if ( ptr = = mCamera - > getTrackingPtr ( ) & &
! mCamera - > isVanityOrPreviewModeEnabled ( ) )
{
2020-07-18 11:48:46 +02:00
mCamera - > rotateCameraToTrackingPtr ( ) ;
2015-05-21 23:54:39 +02:00
}
2015-04-23 23:50:46 +02:00
ptr . getRefData ( ) . getBaseNode ( ) - > setAttitude ( rot ) ;
}
void RenderingManager : : moveObject ( const MWWorld : : Ptr & ptr , const osg : : Vec3f & pos )
{
ptr . getRefData ( ) . getBaseNode ( ) - > setPosition ( pos ) ;
}
void RenderingManager : : scaleObject ( const MWWorld : : Ptr & ptr , const osg : : Vec3f & scale )
{
ptr . getRefData ( ) . getBaseNode ( ) - > setScale ( scale ) ;
2015-11-09 17:30:11 +01:00
if ( ptr = = mCamera - > getTrackingPtr ( ) ) // update height of camera
mCamera - > processViewChange ( ) ;
2015-04-23 23:50:46 +02:00
}
2015-05-22 00:55:43 +02:00
void RenderingManager : : removeObject ( const MWWorld : : Ptr & ptr )
{
2018-07-21 13:37:02 +03:00
mActorsPaths - > remove ( ptr ) ;
2015-05-22 00:55:43 +02:00
mObjects - > removeObject ( ptr ) ;
2015-06-16 20:36:48 +02:00
mWater - > removeEmitter ( ptr ) ;
2015-05-22 00:55:43 +02:00
}
2015-06-02 16:35:35 +02:00
void RenderingManager : : setWaterEnabled ( bool enabled )
{
mWater - > setEnabled ( enabled ) ;
2015-11-03 23:15:43 +01:00
mSky - > setWaterEnabled ( enabled ) ;
2015-06-02 16:35:35 +02:00
}
void RenderingManager : : setWaterHeight ( float height )
{
2019-06-13 13:37:00 +00:00
mWater - > setCullCallback ( mTerrain - > getHeightCullCallback ( height , Mask_Water ) ) ;
2015-06-02 16:35:35 +02:00
mWater - > setHeight ( height ) ;
2015-11-03 23:15:43 +01:00
mSky - > setWaterHeight ( height ) ;
2015-06-02 16:35:35 +02:00
}
2021-01-08 19:33:51 +04:00
void RenderingManager : : screenshot ( osg : : Image * image , int w , int h )
2015-06-03 16:40:16 +02:00
{
2021-01-08 19:33:51 +04:00
mScreenshotManager - > screenshot ( image , w , h ) ;
}
2015-06-03 16:40:16 +02:00
2021-01-08 19:33:51 +04:00
bool RenderingManager : : screenshot360 ( osg : : Image * image )
2017-11-07 13:07:11 +01:00
{
2017-11-11 13:51:42 +01:00
if ( mCamera - > isVanityOrPreviewModeEnabled ( ) )
{
2018-08-14 23:05:43 +04:00
Log ( Debug : : Warning ) < < " Spherical screenshots are not allowed in preview mode. " ;
2017-11-11 13:51:42 +01:00
return false ;
}
2017-11-09 20:25:29 +01:00
2021-04-20 10:42:06 +04:00
unsigned int maskBackup = mPlayerAnimation - > getObjectRoot ( ) - > getNodeMask ( ) ;
2017-11-09 23:09:13 +01:00
2017-11-07 16:12:31 +01:00
if ( mCamera - > isFirstPerson ( ) )
2020-04-20 18:47:14 +02:00
mPlayerAnimation - > getObjectRoot ( ) - > setNodeMask ( 0 ) ;
2017-11-07 16:12:31 +01:00
2021-01-08 19:33:51 +04:00
mScreenshotManager - > screenshot360 ( image ) ;
2017-11-07 22:13:05 +01:00
2017-11-09 23:09:13 +01:00
mPlayerAnimation - > getObjectRoot ( ) - > setNodeMask ( maskBackup ) ;
2017-11-15 15:20:59 +01:00
2017-11-10 15:23:44 +01:00
return true ;
2017-11-07 13:07:11 +01:00
}
2020-05-11 13:37:00 +00:00
osg : : Vec4f RenderingManager : : getScreenBounds ( const osg : : BoundingBox & worldbb )
2015-05-24 03:36:34 +02:00
{
2020-05-11 13:37:00 +00:00
if ( ! worldbb . valid ( ) ) return osg : : Vec4f ( ) ;
2015-05-24 03:36:34 +02:00
osg : : Matrix viewProj = mViewer - > getCamera ( ) - > getViewMatrix ( ) * mViewer - > getCamera ( ) - > getProjectionMatrix ( ) ;
float min_x = 1.0f , max_x = 0.0f , min_y = 1.0f , max_y = 0.0f ;
for ( int i = 0 ; i < 8 ; + + i )
{
2020-05-11 13:37:00 +00:00
osg : : Vec3f corner = worldbb . corner ( i ) ;
2015-05-24 03:36:34 +02:00
corner = corner * viewProj ;
float x = ( corner . x ( ) + 1.f ) * 0.5f ;
float y = ( corner . y ( ) - 1.f ) * ( - 0.5f ) ;
if ( x < min_x )
min_x = x ;
if ( x > max_x )
max_x = x ;
if ( y < min_y )
min_y = y ;
if ( y > max_y )
max_y = y ;
}
return osg : : Vec4f ( min_x , min_y , max_x , max_y ) ;
}
2015-06-05 02:26:16 +02:00
RenderingManager : : RayResult getIntersectionResult ( osgUtil : : LineSegmentIntersector * intersector )
{
RenderingManager : : RayResult result ;
result . mHit = false ;
2021-04-19 16:01:41 +04:00
result . mHitRefnum . unset ( ) ;
2016-07-06 22:03:37 +09:00
result . mRatio = 0 ;
2015-06-05 02:26:16 +02:00
if ( intersector - > containsIntersections ( ) )
{
result . mHit = true ;
osgUtil : : LineSegmentIntersector : : Intersection intersection = intersector - > getFirstIntersection ( ) ;
result . mHitPointWorld = intersection . getWorldIntersectPoint ( ) ;
result . mHitNormalWorld = intersection . getWorldIntersectNormal ( ) ;
2016-07-06 22:03:37 +09:00
result . mRatio = intersection . ratio ;
2015-06-05 02:26:16 +02:00
2018-10-09 10:21:12 +04:00
PtrHolder * ptrHolder = nullptr ;
2020-05-11 13:37:00 +00:00
std : : vector < RefnumMarker * > refnumMarkers ;
2015-06-05 02:26:16 +02:00
for ( osg : : NodePath : : const_iterator it = intersection . nodePath . begin ( ) ; it ! = intersection . nodePath . end ( ) ; + + it )
{
osg : : UserDataContainer * userDataContainer = ( * it ) - > getUserDataContainer ( ) ;
if ( ! userDataContainer )
continue ;
for ( unsigned int i = 0 ; i < userDataContainer - > getNumUserObjects ( ) ; + + i )
{
if ( PtrHolder * p = dynamic_cast < PtrHolder * > ( userDataContainer - > getUserObject ( i ) ) )
ptrHolder = p ;
2020-05-11 13:37:00 +00:00
if ( RefnumMarker * r = dynamic_cast < RefnumMarker * > ( userDataContainer - > getUserObject ( i ) ) )
refnumMarkers . push_back ( r ) ;
2015-06-05 02:26:16 +02:00
}
}
if ( ptrHolder )
result . mHitObject = ptrHolder - > mPtr ;
2020-05-11 13:37:00 +00:00
unsigned int vertexCounter = 0 ;
for ( unsigned int i = 0 ; i < refnumMarkers . size ( ) ; + + i )
{
unsigned int intersectionIndex = intersection . indexList . empty ( ) ? 0 : intersection . indexList [ 0 ] ;
if ( ! refnumMarkers [ i ] - > mNumVertices | | ( intersectionIndex > = vertexCounter & & intersectionIndex < vertexCounter + refnumMarkers [ i ] - > mNumVertices ) )
{
result . mHitRefnum = refnumMarkers [ i ] - > mRefnum ;
break ;
}
vertexCounter + = refnumMarkers [ i ] - > mNumVertices ;
}
2015-06-05 02:26:16 +02:00
}
return result ;
}
2017-03-12 19:48:41 +01:00
osg : : ref_ptr < osgUtil : : IntersectionVisitor > RenderingManager : : getIntersectionVisitor ( osgUtil : : Intersector * intersector , bool ignorePlayer , bool ignoreActors )
2015-06-05 02:26:16 +02:00
{
2017-03-12 19:48:41 +01:00
if ( ! mIntersectionVisitor )
mIntersectionVisitor = new osgUtil : : IntersectionVisitor ;
mIntersectionVisitor - > setTraversalNumber ( mViewer - > getFrameStamp ( ) - > getFrameNumber ( ) ) ;
2020-05-12 13:37:00 +00:00
mIntersectionVisitor - > setFrameStamp ( mViewer - > getFrameStamp ( ) ) ;
2017-03-12 19:48:41 +01:00
mIntersectionVisitor - > setIntersector ( intersector ) ;
2021-04-20 10:42:06 +04:00
unsigned int mask = ~ 0u ;
2020-01-12 11:42:47 +04:00
mask & = ~ ( Mask_RenderToTexture | Mask_Sky | Mask_Debug | Mask_Effect | Mask_Water | Mask_SimpleWater | Mask_Groundcover ) ;
2015-06-05 02:26:16 +02:00
if ( ignorePlayer )
2020-04-20 18:47:14 +02:00
mask & = ~ ( Mask_Player ) ;
2015-06-05 02:26:16 +02:00
if ( ignoreActors )
2020-04-20 18:47:14 +02:00
mask & = ~ ( Mask_Actor | Mask_Player ) ;
2015-06-05 02:26:16 +02:00
2017-03-12 19:48:41 +01:00
mIntersectionVisitor - > setTraversalMask ( mask ) ;
return mIntersectionVisitor ;
2015-10-31 01:30:02 +01:00
}
RenderingManager : : RayResult RenderingManager : : castRay ( const osg : : Vec3f & origin , const osg : : Vec3f & dest , bool ignorePlayer , bool ignoreActors )
{
osg : : ref_ptr < osgUtil : : LineSegmentIntersector > intersector ( new osgUtil : : LineSegmentIntersector ( osgUtil : : LineSegmentIntersector : : MODEL ,
origin , dest ) ) ;
intersector - > setIntersectionLimit ( osgUtil : : LineSegmentIntersector : : LIMIT_NEAREST ) ;
2015-06-05 02:26:16 +02:00
2017-03-12 19:48:41 +01:00
mRootNode - > accept ( * getIntersectionVisitor ( intersector , ignorePlayer , ignoreActors ) ) ;
2015-06-05 02:26:16 +02:00
return getIntersectionResult ( intersector ) ;
}
RenderingManager : : RayResult RenderingManager : : castCameraToViewportRay ( const float nX , const float nY , float maxDistance , bool ignorePlayer , bool ignoreActors )
2015-05-24 03:36:34 +02:00
{
osg : : ref_ptr < osgUtil : : LineSegmentIntersector > intersector ( new osgUtil : : LineSegmentIntersector ( osgUtil : : LineSegmentIntersector : : PROJECTION ,
nX * 2.f - 1.f , nY * ( - 2.f ) + 1.f ) ) ;
osg : : Vec3d dist ( 0.f , 0.f , - maxDistance ) ;
dist = dist * mViewer - > getCamera ( ) - > getProjectionMatrix ( ) ;
osg : : Vec3d end = intersector - > getEnd ( ) ;
end . z ( ) = dist . z ( ) ;
intersector - > setEnd ( end ) ;
intersector - > setIntersectionLimit ( osgUtil : : LineSegmentIntersector : : LIMIT_NEAREST ) ;
2017-03-12 19:48:41 +01:00
mViewer - > getCamera ( ) - > accept ( * getIntersectionVisitor ( intersector , ignorePlayer , ignoreActors ) ) ;
2015-05-24 03:36:34 +02:00
2015-06-05 02:26:16 +02:00
return getIntersectionResult ( intersector ) ;
2015-05-24 03:36:34 +02:00
}
2015-05-14 17:34:55 +02:00
void RenderingManager : : updatePtr ( const MWWorld : : Ptr & old , const MWWorld : : Ptr & updated )
{
mObjects - > updatePtr ( old , updated ) ;
2018-07-21 13:37:02 +03:00
mActorsPaths - > updatePtr ( old , updated ) ;
2015-05-14 17:34:55 +02:00
}
2016-09-14 23:18:29 +09:00
void RenderingManager : : spawnEffect ( const std : : string & model , const std : : string & texture , const osg : : Vec3f & worldPosition , float scale , bool isMagicVFX )
2015-04-19 17:55:56 +02:00
{
2016-09-14 23:18:29 +09:00
mEffectManager - > addEffect ( model , texture , worldPosition , scale , isMagicVFX ) ;
2015-04-19 17:55:56 +02:00
}
void RenderingManager : : notifyWorldSpaceChanged ( )
{
mEffectManager - > clear ( ) ;
2020-02-19 23:26:42 +03:00
mWater - > clearRipples ( ) ;
2015-04-19 17:55:56 +02:00
}
void RenderingManager : : clear ( )
{
2015-06-16 20:56:48 +02:00
mSky - > setMoonColour ( false ) ;
2015-04-19 17:55:56 +02:00
notifyWorldSpaceChanged ( ) ;
2019-08-03 13:37:00 +00:00
if ( mObjectPaging )
mObjectPaging - > clear ( ) ;
2015-04-19 01:57:52 +02:00
}
2015-04-25 15:19:17 +02:00
MWRender : : Animation * RenderingManager : : getAnimation ( const MWWorld : : Ptr & ptr )
{
2015-05-22 04:36:17 +02:00
if ( mPlayerAnimation . get ( ) & & ptr = = mPlayerAnimation - > getPtr ( ) )
return mPlayerAnimation . get ( ) ;
2015-04-25 15:19:17 +02:00
return mObjects - > getAnimation ( ptr ) ;
}
2015-12-18 17:21:51 +01:00
const MWRender : : Animation * RenderingManager : : getAnimation ( const MWWorld : : ConstPtr & ptr ) const
{
if ( mPlayerAnimation . get ( ) & & ptr = = mPlayerAnimation - > getPtr ( ) )
return mPlayerAnimation . get ( ) ;
return mObjects - > getAnimation ( ptr ) ;
}
2015-05-01 18:21:50 +02:00
void RenderingManager : : setupPlayer ( const MWWorld : : Ptr & player )
{
if ( ! mPlayerNode )
{
2015-11-20 21:57:04 +01:00
mPlayerNode = new SceneUtil : : PositionAttitudeTransform ;
2020-04-20 18:47:14 +02:00
mPlayerNode - > setNodeMask ( Mask_Player ) ;
2017-02-02 21:46:25 +01:00
mPlayerNode - > setName ( " Player Root " ) ;
2016-02-08 16:45:56 +01:00
mSceneRoot - > addChild ( mPlayerNode ) ;
2015-05-01 18:21:50 +02:00
}
2015-05-24 04:00:35 +02:00
mPlayerNode - > setUserDataContainer ( new osg : : DefaultUserDataContainer ) ;
mPlayerNode - > getUserDataContainer ( ) - > addUserObject ( new PtrHolder ( player ) ) ;
2015-05-01 18:21:50 +02:00
2015-05-24 04:00:35 +02:00
player . getRefData ( ) . setBaseNode ( mPlayerNode ) ;
2015-06-16 20:36:48 +02:00
2017-09-17 23:07:17 +00:00
mWater - > removeEmitter ( player ) ;
2015-06-16 20:36:48 +02:00
mWater - > addEmitter ( player ) ;
2015-05-01 18:21:50 +02:00
}
void RenderingManager : : renderPlayer ( const MWWorld : : Ptr & player )
{
2017-02-22 15:22:40 +01:00
mPlayerAnimation = new NpcAnimation ( player , player . getRefData ( ) . getBaseNode ( ) , mResourceSystem , 0 , NpcAnimation : : VM_Normal ,
mFirstPersonFieldOfView ) ;
2015-05-01 18:21:50 +02:00
2015-05-21 23:54:39 +02:00
mCamera - > setAnimation ( mPlayerAnimation . get ( ) ) ;
mCamera - > attachTo ( player ) ;
2015-05-01 18:21:50 +02:00
}
2015-05-21 23:54:39 +02:00
void RenderingManager : : rebuildPtr ( const MWWorld : : Ptr & ptr )
{
2018-10-09 10:21:12 +04:00
NpcAnimation * anim = nullptr ;
2015-05-21 23:54:39 +02:00
if ( ptr = = mPlayerAnimation - > getPtr ( ) )
anim = mPlayerAnimation . get ( ) ;
else
anim = dynamic_cast < NpcAnimation * > ( mObjects - > getAnimation ( ptr ) ) ;
if ( anim )
{
anim - > rebuild ( ) ;
if ( mCamera - > getTrackingPtr ( ) = = ptr )
{
mCamera - > attachTo ( ptr ) ;
mCamera - > setAnimation ( anim ) ;
}
}
}
2015-06-16 20:36:48 +02:00
void RenderingManager : : addWaterRippleEmitter ( const MWWorld : : Ptr & ptr )
{
mWater - > addEmitter ( ptr ) ;
}
void RenderingManager : : removeWaterRippleEmitter ( const MWWorld : : Ptr & ptr )
{
mWater - > removeEmitter ( ptr ) ;
}
2015-12-04 23:28:11 +01:00
void RenderingManager : : emitWaterRipple ( const osg : : Vec3f & pos )
{
mWater - > emitRipple ( pos ) ;
}
2015-05-14 18:46:04 +02:00
void RenderingManager : : updateProjectionMatrix ( )
{
2015-05-26 16:40:44 +02:00
double aspect = mViewer - > getCamera ( ) - > getViewport ( ) - > aspectRatio ( ) ;
2015-12-07 16:11:47 +01:00
float fov = mFieldOfView ;
if ( mFieldOfViewOverridden )
fov = mFieldOfViewOverride ;
2021-06-01 12:15:25 -07:00
2021-07-09 10:05:27 -07:00
mViewer - > getCamera ( ) - > setProjectionMatrixAsPerspective ( fov , aspect , mNearClip , mViewDistance ) ;
2021-08-04 17:49:57 -07:00
if ( SceneUtil : : getReverseZ ( ) )
2021-06-01 12:15:25 -07:00
{
2021-07-09 10:05:27 -07:00
mSharedUniformStateUpdater - > setLinearFac ( - mNearClip / ( mViewDistance - mNearClip ) - 1.f ) ;
mSharedUniformStateUpdater - > setProjectionMatrix ( SceneUtil : : getReversedZProjectionMatrixAsPerspective ( fov , aspect , mNearClip , mViewDistance ) ) ;
2021-06-01 12:15:25 -07:00
}
else
2021-07-09 10:05:27 -07:00
mSharedUniformStateUpdater - > setProjectionMatrix ( mViewer - > getCamera ( ) - > getProjectionMatrix ( ) ) ;
2017-09-26 14:14:28 +02:00
2021-08-04 17:49:57 -07:00
mSharedUniformStateUpdater - > setNear ( mNearClip ) ;
mSharedUniformStateUpdater - > setFar ( mViewDistance ) ;
2019-02-20 13:37:00 +00:00
// Since our fog is not radial yet, we should take FOV in account, otherwise terrain near viewing distance may disappear.
2019-03-03 21:29:51 +04:00
// Limit FOV here just for sure, otherwise viewing distance can be too high.
fov = std : : min ( mFieldOfView , 140.f ) ;
float distanceMult = std : : cos ( osg : : DegreesToRadians ( fov ) / 2.f ) ;
2019-02-20 13:37:00 +00:00
mTerrain - > setViewDistance ( mViewDistance * ( distanceMult ? 1.f / distanceMult : 1.f ) ) ;
2015-05-14 21:42:04 +02:00
}
void RenderingManager : : updateTextureFiltering ( )
{
2016-02-14 23:14:52 +01:00
mViewer - > stopThreading ( ) ;
2016-02-05 23:20:13 +01:00
2016-02-05 18:51:51 +01:00
mResourceSystem - > getSceneManager ( ) - > setFilterSettings (
2015-12-13 17:05:19 -08:00
Settings : : Manager : : getString ( " texture mag filter " , " General " ) ,
Settings : : Manager : : getString ( " texture min filter " , " General " ) ,
2015-12-13 16:51:27 -08:00
Settings : : Manager : : getString ( " texture mipmap " , " General " ) ,
2016-02-14 23:14:52 +01:00
Settings : : Manager : : getInt ( " anisotropy " , " General " )
2015-12-13 16:51:27 -08:00
) ;
2016-02-14 23:14:52 +01:00
mTerrain - > updateTextureFiltering ( ) ;
mViewer - > startThreading ( ) ;
2015-05-14 18:46:04 +02:00
}
2015-06-11 23:16:05 +02:00
void RenderingManager : : updateAmbient ( )
{
osg : : Vec4f color = mAmbientColor ;
if ( mNightEyeFactor > 0.f )
color + = osg : : Vec4f ( 0.7 , 0.7 , 0.7 , 0.0 ) * mNightEyeFactor ;
mStateUpdater - > setAmbientColor ( color ) ;
}
2015-06-02 16:35:35 +02:00
void RenderingManager : : setFogColor ( const osg : : Vec4f & color )
{
mViewer - > getCamera ( ) - > setClearColor ( color ) ;
mStateUpdater - > setFogColor ( color ) ;
}
2017-03-07 04:02:06 +01:00
void RenderingManager : : reportStats ( ) const
2017-02-22 02:18:18 +01:00
{
osg : : Stats * stats = mViewer - > getViewerStats ( ) ;
unsigned int frameNumber = mViewer - > getFrameStamp ( ) - > getFrameNumber ( ) ;
if ( stats - > collectStats ( " resource " ) )
{
stats - > setAttribute ( frameNumber , " UnrefQueue " , mUnrefQueue - > getNumItems ( ) ) ;
2017-03-09 19:52:30 +01:00
mTerrain - > reportStats ( frameNumber , stats ) ;
2017-02-22 02:18:18 +01:00
}
}
2015-05-14 18:46:04 +02:00
void RenderingManager : : processChangedSettings ( const Settings : : CategorySettingVector & changed )
{
for ( Settings : : CategorySettingVector : : const_iterator it = changed . begin ( ) ; it ! = changed . end ( ) ; + + it )
{
2015-12-07 16:23:06 +01:00
if ( it - > first = = " Camera " & & it - > second = = " field of view " )
2015-05-14 18:46:04 +02:00
{
2015-12-07 16:23:06 +01:00
mFieldOfView = Settings : : Manager : : getFloat ( " field of view " , " Camera " ) ;
2015-05-14 18:46:04 +02:00
updateProjectionMatrix ( ) ;
}
2015-05-24 02:34:20 +02:00
else if ( it - > first = = " Camera " & & it - > second = = " viewing distance " )
2015-05-14 18:46:04 +02:00
{
2015-05-24 02:34:20 +02:00
mViewDistance = Settings : : Manager : : getFloat ( " viewing distance " , " Camera " ) ;
2020-06-07 10:35:33 +04:00
if ( ! Settings : : Manager : : getBool ( " use distant fog " , " Fog " ) )
2017-09-21 03:08:45 -07:00
mStateUpdater - > setFogEnd ( mViewDistance ) ;
2015-05-14 18:46:04 +02:00
updateProjectionMatrix ( ) ;
}
2015-12-13 16:02:09 -08:00
else if ( it - > first = = " General " & & ( it - > second = = " texture filter " | |
it - > second = = " texture mipmap " | |
2015-12-13 11:24:23 -08:00
it - > second = = " anisotropy " ) )
2021-04-03 22:25:13 -07:00
{
2015-05-14 21:42:04 +02:00
updateTextureFiltering ( ) ;
2021-04-03 22:25:13 -07:00
}
2015-10-28 21:22:14 +01:00
else if ( it - > first = = " Water " )
2021-04-03 22:25:13 -07:00
{
2015-10-28 21:22:14 +01:00
mWater - > processChangedSettings ( changed ) ;
2021-04-03 22:25:13 -07:00
}
else if ( it - > first = = " Shaders " & & it - > second = = " minimum interior brightness " )
{
mMinimumAmbientLuminance = std : : clamp ( Settings : : Manager : : getFloat ( " minimum interior brightness " , " Shaders " ) , 0.f , 1.f ) ;
2021-04-05 10:43:17 -07:00
if ( MWMechanics : : getPlayer ( ) . isInCell ( ) )
2021-04-03 22:25:13 -07:00
configureAmbient ( MWMechanics : : getPlayer ( ) . getCell ( ) - > getCell ( ) ) ;
}
else if ( it - > first = = " Shaders " & & ( it - > second = = " light bounds multiplier " | |
it - > second = = " maximum light distance " | |
2021-04-05 10:43:17 -07:00
it - > second = = " light fade start " | |
it - > second = = " max lights " ) )
2021-04-03 22:25:13 -07:00
{
2021-04-05 10:43:17 -07:00
auto * lightManager = static_cast < SceneUtil : : LightManager * > ( getLightRoot ( ) ) ;
lightManager - > processChangedSettings ( changed ) ;
if ( it - > second = = " max lights " & & ! lightManager - > usingFFP ( ) )
{
mViewer - > stopThreading ( ) ;
lightManager - > updateMaxLights ( ) ;
auto defines = mResourceSystem - > getSceneManager ( ) - > getShaderManager ( ) . getGlobalDefines ( ) ;
2021-04-13 11:00:48 -07:00
for ( const auto & [ name , key ] : lightManager - > getLightDefines ( ) )
defines [ name ] = key ;
2021-04-05 10:43:17 -07:00
mResourceSystem - > getSceneManager ( ) - > getShaderManager ( ) . setGlobalDefines ( defines ) ;
mSceneRoot - > removeUpdateCallback ( mStateUpdater ) ;
mStateUpdater = new StateUpdater ;
mSceneRoot - > addUpdateCallback ( mStateUpdater ) ;
mStateUpdater - > setFogEnd ( mViewDistance ) ;
updateAmbient ( ) ;
mViewer - > startThreading ( ) ;
}
2021-04-03 22:25:13 -07:00
}
2015-05-14 18:46:04 +02:00
}
}
2015-06-01 15:34:46 +02:00
float RenderingManager : : getNearClipDistance ( ) const
{
return mNearClip ;
}
2015-06-03 01:18:36 +02:00
float RenderingManager : : getTerrainHeightAt ( const osg : : Vec3f & pos )
{
return mTerrain - > getHeightAt ( pos ) ;
}
2015-12-07 16:11:47 +01:00
void RenderingManager : : overrideFieldOfView ( float val )
{
if ( mFieldOfViewOverridden ! = true | | mFieldOfViewOverride ! = val )
{
mFieldOfViewOverridden = true ;
mFieldOfViewOverride = val ;
updateProjectionMatrix ( ) ;
}
}
2018-06-16 10:18:04 +04:00
osg : : Vec3f RenderingManager : : getHalfExtents ( const MWWorld : : ConstPtr & object ) const
{
osg : : Vec3f halfExtents ( 0 , 0 , 0 ) ;
std : : string modelName = object . getClass ( ) . getModel ( object ) ;
if ( modelName . empty ( ) )
return halfExtents ;
osg : : ref_ptr < const osg : : Node > node = mResourceSystem - > getSceneManager ( ) - > getTemplate ( modelName ) ;
osg : : ComputeBoundsVisitor computeBoundsVisitor ;
2020-04-20 18:47:14 +02:00
computeBoundsVisitor . setTraversalMask ( ~ ( MWRender : : Mask_ParticleSystem | MWRender : : Mask_Effect ) ) ;
2018-06-16 10:18:04 +04:00
const_cast < osg : : Node * > ( node . get ( ) ) - > accept ( computeBoundsVisitor ) ;
osg : : BoundingBox bounds = computeBoundsVisitor . getBoundingBox ( ) ;
if ( bounds . valid ( ) )
{
halfExtents [ 0 ] = std : : abs ( bounds . xMax ( ) - bounds . xMin ( ) ) / 2.f ;
halfExtents [ 1 ] = std : : abs ( bounds . yMax ( ) - bounds . yMin ( ) ) / 2.f ;
halfExtents [ 2 ] = std : : abs ( bounds . zMax ( ) - bounds . zMin ( ) ) / 2.f ;
}
return halfExtents ;
}
2015-12-07 16:11:47 +01:00
void RenderingManager : : resetFieldOfView ( )
{
if ( mFieldOfViewOverridden = = true )
{
mFieldOfViewOverridden = false ;
2017-02-01 03:00:33 +01:00
2015-12-07 16:11:47 +01:00
updateProjectionMatrix ( ) ;
}
}
2017-02-01 03:00:33 +01:00
void RenderingManager : : exportSceneGraph ( const MWWorld : : Ptr & ptr , const std : : string & filename , const std : : string & format )
{
osg : : Node * node = mViewer - > getSceneData ( ) ;
if ( ! ptr . isEmpty ( ) )
node = ptr . getRefData ( ) . getBaseNode ( ) ;
SceneUtil : : writeScene ( node , filename , format ) ;
}
2015-12-07 16:11:47 +01:00
2017-03-06 19:04:17 +01:00
LandManager * RenderingManager : : getLandManager ( ) const
{
return mTerrainStorage - > getLandManager ( ) ;
}
2018-07-21 13:37:02 +03:00
void RenderingManager : : updateActorPath ( const MWWorld : : ConstPtr & actor , const std : : deque < osg : : Vec3f > & path ,
const osg : : Vec3f & halfExtents , const osg : : Vec3f & start , const osg : : Vec3f & end ) const
{
mActorsPaths - > update ( actor , path , halfExtents , start , end , mNavigator . getSettings ( ) ) ;
}
2017-03-06 19:04:17 +01:00
2018-07-21 13:37:02 +03:00
void RenderingManager : : removeActorPath ( const MWWorld : : ConstPtr & actor ) const
{
mActorsPaths - > remove ( actor ) ;
}
2017-03-06 19:04:17 +01:00
2018-08-31 01:39:44 +03:00
void RenderingManager : : setNavMeshNumber ( const std : : size_t value )
{
mNavMeshNumber = value ;
}
2019-01-20 19:27:52 +03:00
void RenderingManager : : updateNavMesh ( )
{
2019-01-20 19:30:26 +03:00
if ( ! mNavMesh - > isEnabled ( ) )
return ;
2019-01-20 19:27:52 +03:00
const auto navMeshes = mNavigator . getNavMeshes ( ) ;
auto it = navMeshes . begin ( ) ;
for ( std : : size_t i = 0 ; it ! = navMeshes . end ( ) & & i < mNavMeshNumber ; + + i )
+ + it ;
if ( it = = navMeshes . end ( ) )
{
mNavMesh - > reset ( ) ;
}
else
{
try
{
2019-03-10 16:49:56 +03:00
const auto locked = it - > second - > lockConst ( ) ;
2019-03-10 15:48:12 +03:00
mNavMesh - > update ( locked - > getImpl ( ) , mNavMeshNumber , locked - > getGeneration ( ) ,
2019-01-20 19:27:52 +03:00
locked - > getNavMeshRevision ( ) , mNavigator . getSettings ( ) ) ;
}
catch ( const std : : exception & e )
{
Log ( Debug : : Error ) < < " NavMesh render update exception: " < < e . what ( ) ;
}
}
}
2019-11-27 23:45:01 +01:00
void RenderingManager : : updateRecastMesh ( )
{
if ( ! mRecastMesh - > isEnabled ( ) )
return ;
mRecastMesh - > update ( mNavigator . getRecastMeshTiles ( ) , mNavigator . getSettings ( ) ) ;
}
2019-06-13 13:37:00 +00:00
void RenderingManager : : setActiveGrid ( const osg : : Vec4i & grid )
{
mTerrain - > setActiveGrid ( grid ) ;
}
2020-05-08 13:37:00 +00:00
bool RenderingManager : : pagingEnableObject ( int type , const MWWorld : : ConstPtr & ptr , bool enabled )
2019-08-03 13:37:00 +00:00
{
2020-05-11 13:37:00 +00:00
if ( ! ptr . isInCell ( ) | | ! ptr . getCell ( ) - > isExterior ( ) | | ! mObjectPaging )
2020-05-08 13:37:00 +00:00
return false ;
2020-07-10 00:44:54 +02:00
if ( mObjectPaging - > enableObject ( type , ptr . getCellRef ( ) . getRefNum ( ) , ptr . getCellRef ( ) . getPosition ( ) . asVec3 ( ) , osg : : Vec2i ( ptr . getCell ( ) - > getCell ( ) - > getGridX ( ) , ptr . getCell ( ) - > getCell ( ) - > getGridY ( ) ) , enabled ) )
2020-05-08 13:37:00 +00:00
{
mTerrain - > rebuildViews ( ) ;
return true ;
}
return false ;
2019-08-03 13:37:00 +00:00
}
2020-05-11 13:37:00 +00:00
void RenderingManager : : pagingBlacklistObject ( int type , const MWWorld : : ConstPtr & ptr )
{
if ( ! ptr . isInCell ( ) | | ! ptr . getCell ( ) - > isExterior ( ) | | ! mObjectPaging )
return ;
const ESM : : RefNum & refnum = ptr . getCellRef ( ) . getRefNum ( ) ;
if ( ! refnum . hasContentFile ( ) ) return ;
2020-07-09 23:17:01 +02:00
if ( mObjectPaging - > blacklistObject ( type , refnum , ptr . getCellRef ( ) . getPosition ( ) . asVec3 ( ) , osg : : Vec2i ( ptr . getCell ( ) - > getCell ( ) - > getGridX ( ) , ptr . getCell ( ) - > getCell ( ) - > getGridY ( ) ) ) )
2020-05-11 13:37:00 +00:00
mTerrain - > rebuildViews ( ) ;
}
2020-05-12 13:37:00 +00:00
bool RenderingManager : : pagingUnlockCache ( )
{
if ( mObjectPaging & & mObjectPaging - > unlockCache ( ) )
{
mTerrain - > rebuildViews ( ) ;
return true ;
}
return false ;
}
2020-05-10 13:37:00 +00:00
void RenderingManager : : getPagedRefnums ( const osg : : Vec4i & activeGrid , std : : set < ESM : : RefNum > & out )
{
if ( mObjectPaging )
mObjectPaging - > getPagedRefnums ( activeGrid , out ) ;
}
2012-05-27 21:39:18 +02:00
}