2010-06-28 09:36:54 +00:00
# include "sky.hpp"
2015-07-30 04:57:45 +00:00
# include <cmath>
2015-12-06 14:56:30 +00:00
# include <osg/ClipPlane>
# include <osg/Fog>
2015-04-14 13:55:56 +00:00
# include <osg/Transform>
# include <osg/Depth>
# include <osg/Geometry>
# include <osg/Material>
2015-04-15 16:50:50 +00:00
# include <osg/TexEnvCombine>
# include <osg/TexMat>
2015-09-19 16:10:02 +00:00
# include <osg/OcclusionQueryNode>
# include <osg/ColorMask>
2016-03-10 11:59:17 +00:00
# include <osg/PositionAttitudeTransform>
2015-09-21 14:03:30 +00:00
# include <osg/BlendFunc>
2015-09-21 15:09:24 +00:00
# include <osg/AlphaFunc>
2016-03-10 22:18:20 +00:00
# include <osg/PolygonOffset>
2020-02-13 18:32:17 +00:00
# include <osg/Version>
2016-01-25 20:03:33 +00:00
# include <osg/observer_ptr>
2015-04-15 16:50:50 +00:00
2019-10-14 11:40:35 +00:00
# include <osgParticle/BoxPlacer>
2019-10-12 15:35:06 +00:00
# include <osgParticle/ModularEmitter>
2015-06-25 15:23:01 +00:00
# include <osgParticle/ParticleSystem>
# include <osgParticle/ParticleSystemUpdater>
# include <osgParticle/ConstantRateCounter>
# include <osgParticle/RadialShooter>
2017-10-14 16:45:29 +00:00
# include <osgParticle/Operator>
# include <osgParticle/ModularProgram>
2015-04-22 15:58:55 +00:00
# include <components/misc/rng.hpp>
2015-03-15 01:07:47 +00:00
2014-08-12 10:18:38 +00:00
# include <components/misc/resourcehelpers.hpp>
2012-02-21 15:15:38 +00:00
2015-04-14 13:55:56 +00:00
# include <components/resource/scenemanager.hpp>
2016-02-05 22:03:53 +00:00
# include <components/resource/imagemanager.hpp>
2015-04-14 13:55:56 +00:00
# include <components/vfs/manager.hpp>
2016-01-06 11:46:06 +00:00
# include <components/fallback/fallback.hpp>
2015-04-14 13:55:56 +00:00
# include <components/sceneutil/util.hpp>
2015-04-14 15:29:12 +00:00
# include <components/sceneutil/statesetupdater.hpp>
2015-06-17 22:30:51 +00:00
# include <components/sceneutil/controller.hpp>
2015-11-02 22:49:22 +00:00
# include <components/sceneutil/visitor.hpp>
2017-11-08 01:44:49 +00:00
# include <components/sceneutil/shadow.hpp>
2012-07-13 07:13:12 +00:00
2021-04-08 16:32:38 +00:00
# include <components/settings/settings.hpp>
# include <components/misc/stringops.hpp>
2020-12-29 11:49:22 +00:00
# include <components/nifosg/particle.hpp>
2012-04-23 13:27:03 +00:00
# include "../mwbase/environment.hpp"
2012-07-03 10:30:50 +00:00
# include "../mwbase/world.hpp"
2020-04-20 16:47:14 +00:00
# include "vismask.hpp"
2015-09-19 16:19:36 +00:00
# include "renderbin.hpp"
2012-03-16 18:02:33 +00:00
2014-12-01 19:31:33 +00:00
namespace
{
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < osg : : Material > createAlphaTrackingUnlitMaterial ( )
2014-12-01 19:31:33 +00:00
{
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < osg : : Material > mat = new osg : : Material ;
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 1.f ) ) ;
mat - > setAmbient ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 1.f ) ) ;
mat - > setEmission ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 1.f , 1.f , 1.f , 1.f ) ) ;
2015-05-26 16:22:21 +00:00
mat - > setSpecular ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 0.f ) ) ;
2015-04-14 13:55:56 +00:00
mat - > setColorMode ( osg : : Material : : DIFFUSE ) ;
return mat ;
2014-12-01 19:31:33 +00:00
}
2012-07-13 07:13:12 +00:00
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < osg : : Material > createUnlitMaterial ( )
{
osg : : ref_ptr < osg : : Material > mat = new osg : : Material ;
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 1.f ) ) ;
mat - > setAmbient ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 1.f ) ) ;
mat - > setEmission ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 1.f , 1.f , 1.f , 1.f ) ) ;
2015-05-26 16:22:21 +00:00
mat - > setSpecular ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0.f , 0.f , 0.f , 0.f ) ) ;
2015-04-14 13:55:56 +00:00
mat - > setColorMode ( osg : : Material : : OFF ) ;
return mat ;
}
2012-07-13 07:13:12 +00:00
2015-04-15 16:50:50 +00:00
osg : : ref_ptr < osg : : Geometry > createTexturedQuad ( int numUvSets = 1 )
2015-04-14 13:55:56 +00:00
{
osg : : ref_ptr < osg : : Geometry > geom = new osg : : Geometry ;
2012-07-13 07:13:12 +00:00
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < osg : : Vec3Array > verts = new osg : : Vec3Array ;
verts - > push_back ( osg : : Vec3f ( - 0.5 , - 0.5 , 0 ) ) ;
verts - > push_back ( osg : : Vec3f ( - 0.5 , 0.5 , 0 ) ) ;
verts - > push_back ( osg : : Vec3f ( 0.5 , 0.5 , 0 ) ) ;
verts - > push_back ( osg : : Vec3f ( 0.5 , - 0.5 , 0 ) ) ;
2013-05-07 15:38:24 +00:00
2015-04-14 13:55:56 +00:00
geom - > setVertexArray ( verts ) ;
2013-05-07 15:38:24 +00:00
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < osg : : Vec2Array > texcoords = new osg : : Vec2Array ;
texcoords - > push_back ( osg : : Vec2f ( 0 , 0 ) ) ;
texcoords - > push_back ( osg : : Vec2f ( 0 , 1 ) ) ;
texcoords - > push_back ( osg : : Vec2f ( 1 , 1 ) ) ;
texcoords - > push_back ( osg : : Vec2f ( 1 , 0 ) ) ;
2012-07-13 07:13:12 +00:00
2015-05-29 00:26:58 +00:00
osg : : ref_ptr < osg : : Vec4Array > colors = new osg : : Vec4Array ;
colors - > push_back ( osg : : Vec4 ( 1.f , 1.f , 1.f , 1.f ) ) ;
geom - > setColorArray ( colors , osg : : Array : : BIND_OVERALL ) ;
2015-04-15 16:50:50 +00:00
for ( int i = 0 ; i < numUvSets ; + + i )
geom - > setTexCoordArray ( i , texcoords , osg : : Array : : BIND_PER_VERTEX ) ;
2012-07-20 15:08:15 +00:00
2015-04-14 13:55:56 +00:00
geom - > addPrimitiveSet ( new osg : : DrawArrays ( osg : : PrimitiveSet : : QUADS , 0 , 4 ) ) ;
2010-06-28 09:36:54 +00:00
2015-04-14 13:55:56 +00:00
return geom ;
}
2012-07-20 15:08:15 +00:00
}
2015-04-14 13:55:56 +00:00
namespace MWRender
2012-02-21 15:15:38 +00:00
{
2015-04-14 15:29:12 +00:00
class AtmosphereUpdater : public SceneUtil : : StateSetUpdater
2012-02-26 19:46:09 +00:00
{
2015-04-14 13:55:56 +00:00
public :
2015-06-19 18:55:04 +00:00
void setEmissionColor ( const osg : : Vec4f & emissionColor )
2015-04-14 13:55:56 +00:00
{
mEmissionColor = emissionColor ;
}
2012-02-26 19:46:09 +00:00
2015-04-14 13:55:56 +00:00
protected :
2020-10-16 18:18:54 +00:00
void setDefaults ( osg : : StateSet * stateset ) override
2012-07-13 07:13:12 +00:00
{
2015-06-18 21:15:13 +00:00
stateset - > setAttributeAndModes ( createAlphaTrackingUnlitMaterial ( ) , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2012-07-13 07:13:12 +00:00
}
2012-02-26 19:46:09 +00:00
2020-10-16 18:18:54 +00:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * /*nv*/ ) override
2015-04-14 13:55:56 +00:00
{
osg : : Material * mat = static_cast < osg : : Material * > ( stateset - > getAttribute ( osg : : StateAttribute : : MATERIAL ) ) ;
mat - > setEmission ( osg : : Material : : FRONT_AND_BACK , mEmissionColor ) ;
}
2012-02-21 15:15:38 +00:00
2015-04-14 13:55:56 +00:00
private :
osg : : Vec4f mEmissionColor ;
} ;
2012-02-26 19:46:09 +00:00
2015-06-19 18:55:04 +00:00
class AtmosphereNightUpdater : public SceneUtil : : StateSetUpdater
{
public :
2016-02-05 22:10:27 +00:00
AtmosphereNightUpdater ( Resource : : ImageManager * imageManager )
2015-06-19 18:55:04 +00:00
{
// we just need a texture, its contents don't really matter
2016-02-05 22:10:27 +00:00
mTexture = new osg : : Texture2D ( imageManager - > getWarningImage ( ) ) ;
2015-06-19 18:55:04 +00:00
}
void setFade ( const float fade )
{
mColor . a ( ) = fade ;
}
protected :
2020-10-16 18:18:54 +00:00
void setDefaults ( osg : : StateSet * stateset ) override
2015-06-19 18:55:04 +00:00
{
osg : : ref_ptr < osg : : TexEnvCombine > texEnv ( new osg : : TexEnvCombine ) ;
texEnv - > setCombine_Alpha ( osg : : TexEnvCombine : : MODULATE ) ;
texEnv - > setSource0_Alpha ( osg : : TexEnvCombine : : PREVIOUS ) ;
texEnv - > setSource1_Alpha ( osg : : TexEnvCombine : : CONSTANT ) ;
texEnv - > setCombine_RGB ( osg : : TexEnvCombine : : REPLACE ) ;
texEnv - > setSource0_RGB ( osg : : TexEnvCombine : : PREVIOUS ) ;
stateset - > setTextureAttributeAndModes ( 1 , mTexture , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
stateset - > setTextureAttributeAndModes ( 1 , texEnv , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
}
2020-10-16 18:18:54 +00:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * /*nv*/ ) override
2015-06-19 18:55:04 +00:00
{
osg : : TexEnvCombine * texEnv = static_cast < osg : : TexEnvCombine * > ( stateset - > getTextureAttribute ( 1 , osg : : StateAttribute : : TEXENV ) ) ;
texEnv - > setConstantColor ( mColor ) ;
}
osg : : ref_ptr < osg : : Texture2D > mTexture ;
osg : : Vec4f mColor ;
} ;
2015-04-15 16:50:50 +00:00
class CloudUpdater : public SceneUtil : : StateSetUpdater
{
public :
2015-07-01 01:42:04 +00:00
CloudUpdater ( )
: mAnimationTimer ( 0.f )
, mOpacity ( 0.f )
{
}
2015-06-04 18:08:44 +00:00
void setAnimationTimer ( float timer )
{
mAnimationTimer = timer ;
}
2015-04-15 16:50:50 +00:00
void setTexture ( osg : : ref_ptr < osg : : Texture2D > texture )
{
mTexture = texture ;
}
2015-06-19 18:55:04 +00:00
void setEmissionColor ( const osg : : Vec4f & emissionColor )
2015-04-15 16:50:50 +00:00
{
mEmissionColor = emissionColor ;
}
void setOpacity ( float opacity )
{
mOpacity = opacity ;
}
protected :
2020-10-16 18:18:54 +00:00
void setDefaults ( osg : : StateSet * stateset ) override
2015-04-15 16:50:50 +00:00
{
2015-06-18 23:03:12 +00:00
osg : : ref_ptr < osg : : TexMat > texmat ( new osg : : TexMat ) ;
stateset - > setTextureAttributeAndModes ( 0 , texmat , osg : : StateAttribute : : ON ) ;
stateset - > setTextureAttributeAndModes ( 1 , texmat , osg : : StateAttribute : : ON ) ;
2015-04-15 16:50:50 +00:00
stateset - > setAttribute ( createAlphaTrackingUnlitMaterial ( ) , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2015-06-18 23:03:12 +00:00
// need to set opacity on a separate texture unit, diffuse alpha is used by the vertex colors already
osg : : ref_ptr < osg : : TexEnvCombine > texEnvCombine ( new osg : : TexEnvCombine ) ;
texEnvCombine - > setSource0_RGB ( osg : : TexEnvCombine : : PREVIOUS ) ;
texEnvCombine - > setSource0_Alpha ( osg : : TexEnvCombine : : PREVIOUS ) ;
texEnvCombine - > setSource1_Alpha ( osg : : TexEnvCombine : : CONSTANT ) ;
texEnvCombine - > setConstantColor ( osg : : Vec4f ( 1 , 1 , 1 , 1 ) ) ;
texEnvCombine - > setCombine_Alpha ( osg : : TexEnvCombine : : MODULATE ) ;
texEnvCombine - > setCombine_RGB ( osg : : TexEnvCombine : : REPLACE ) ;
stateset - > setTextureAttributeAndModes ( 1 , texEnvCombine , osg : : StateAttribute : : ON ) ;
stateset - > setTextureMode ( 0 , GL_TEXTURE_2D , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
stateset - > setTextureMode ( 1 , GL_TEXTURE_2D , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2015-04-15 16:50:50 +00:00
}
2020-10-16 18:18:54 +00:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * nv ) override
2015-04-15 16:50:50 +00:00
{
osg : : TexMat * texMat = static_cast < osg : : TexMat * > ( stateset - > getTextureAttribute ( 0 , osg : : StateAttribute : : TEXMAT ) ) ;
2019-10-19 19:18:58 +00:00
texMat - > setMatrix ( osg : : Matrix : : translate ( osg : : Vec3f ( 0 , - mAnimationTimer , 0.f ) ) ) ;
2015-04-15 16:50:50 +00:00
2015-06-18 23:03:12 +00:00
stateset - > setTextureAttribute ( 0 , mTexture , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
stateset - > setTextureAttribute ( 1 , mTexture , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2015-04-15 16:50:50 +00:00
osg : : Material * mat = static_cast < osg : : Material * > ( stateset - > getAttribute ( osg : : StateAttribute : : MATERIAL ) ) ;
mat - > setEmission ( osg : : Material : : FRONT_AND_BACK , mEmissionColor ) ;
2015-06-18 23:03:12 +00:00
osg : : TexEnvCombine * texEnvCombine = static_cast < osg : : TexEnvCombine * > ( stateset - > getTextureAttribute ( 1 , osg : : StateAttribute : : TEXENV ) ) ;
texEnvCombine - > setConstantColor ( osg : : Vec4f ( 1 , 1 , 1 , mOpacity ) ) ;
2015-04-15 16:50:50 +00:00
}
private :
float mAnimationTimer ;
osg : : ref_ptr < osg : : Texture2D > mTexture ;
osg : : Vec4f mEmissionColor ;
float mOpacity ;
} ;
2015-04-14 13:55:56 +00:00
/// Transform that removes the eyepoint of the modelview matrix,
/// i.e. its children are positioned relative to the camera.
class CameraRelativeTransform : public osg : : Transform
2012-04-06 13:17:54 +00:00
{
2015-04-14 13:55:56 +00:00
public :
CameraRelativeTransform ( )
{
2015-04-15 16:50:50 +00:00
// Culling works in node-local space, not in camera space, so we can't cull this node correctly
// That's not a problem though, children of this node can be culled just fine
// Just make sure you do not place a CameraRelativeTransform deep in the scene graph
setCullingActive ( false ) ;
2015-10-26 20:36:19 +00:00
addCullCallback ( new CullCallback ) ;
2015-04-14 13:55:56 +00:00
}
2012-04-06 13:17:54 +00:00
2015-04-14 13:55:56 +00:00
CameraRelativeTransform ( const CameraRelativeTransform & copy , const osg : : CopyOp & copyop )
: osg : : Transform ( copy , copyop )
2012-07-13 07:13:12 +00:00
{
}
2012-02-22 18:17:37 +00:00
2015-04-14 13:55:56 +00:00
META_Node ( MWRender , CameraRelativeTransform )
2012-02-23 20:44:56 +00:00
2019-04-12 13:35:44 +00:00
const osg : : Vec3f & getLastViewPoint ( ) const
2015-04-14 13:55:56 +00:00
{
2019-04-12 13:35:44 +00:00
return mViewPoint ;
2015-11-03 22:15:43 +00:00
}
2020-10-16 18:18:54 +00:00
bool computeLocalToWorldMatrix ( osg : : Matrix & matrix , osg : : NodeVisitor * nv ) const override
2015-11-03 22:15:43 +00:00
{
if ( nv - > getVisitorType ( ) = = osg : : NodeVisitor : : CULL_VISITOR )
{
2019-04-12 13:35:44 +00:00
mViewPoint = static_cast < osgUtil : : CullVisitor * > ( nv ) - > getViewPoint ( ) ;
2015-11-03 22:15:43 +00:00
}
2015-04-14 13:55:56 +00:00
if ( _referenceFrame = = RELATIVE_RF )
{
matrix . setTrans ( osg : : Vec3f ( 0.f , 0.f , 0.f ) ) ;
return false ;
}
else // absolute
{
matrix . makeIdentity ( ) ;
return true ;
}
}
2012-02-24 15:12:43 +00:00
2020-10-16 18:18:54 +00:00
osg : : BoundingSphere computeBound ( ) const override
2015-04-14 13:55:56 +00:00
{
return osg : : BoundingSphere ( osg : : Vec3f ( 0 , 0 , 0 ) , 0 ) ;
}
2015-10-26 20:36:19 +00:00
class CullCallback : public osg : : NodeCallback
{
public :
2020-10-16 18:18:54 +00:00
void operator ( ) ( osg : : Node * node , osg : : NodeVisitor * nv ) override
2015-10-26 20:36:19 +00:00
{
osgUtil : : CullVisitor * cv = static_cast < osgUtil : : CullVisitor * > ( nv ) ;
// XXX have to remove unwanted culling plane of the water reflection camera
// Remove all planes that aren't from the standard frustum
unsigned int numPlanes = 4 ;
if ( cv - > getCullingMode ( ) & osg : : CullSettings : : NEAR_PLANE_CULLING )
+ + numPlanes ;
if ( cv - > getCullingMode ( ) & osg : : CullSettings : : FAR_PLANE_CULLING )
+ + numPlanes ;
int mask = 0x1 ;
int resultMask = cv - > getProjectionCullingStack ( ) . back ( ) . getFrustum ( ) . getResultMask ( ) ;
for ( unsigned int i = 0 ; i < cv - > getProjectionCullingStack ( ) . back ( ) . getFrustum ( ) . getPlaneList ( ) . size ( ) ; + + i )
{
if ( i > = numPlanes )
{
// turn off this culling plane
resultMask & = ( ~ mask ) ;
}
mask < < = 1 ;
}
cv - > getProjectionCullingStack ( ) . back ( ) . getFrustum ( ) . setResultMask ( resultMask ) ;
cv - > getCurrentCullingSet ( ) . getFrustum ( ) . setResultMask ( resultMask ) ;
cv - > getProjectionCullingStack ( ) . back ( ) . pushCurrentMask ( ) ;
cv - > getCurrentCullingSet ( ) . pushCurrentMask ( ) ;
traverse ( node , nv ) ;
cv - > getProjectionCullingStack ( ) . back ( ) . popCurrentMask ( ) ;
cv - > getCurrentCullingSet ( ) . popCurrentMask ( ) ;
}
} ;
2015-11-03 22:15:43 +00:00
private :
2019-04-12 13:35:44 +00:00
// viewPoint for the current frame
mutable osg : : Vec3f mViewPoint ;
2015-04-14 13:55:56 +00:00
} ;
2012-02-29 09:13:25 +00:00
2015-04-14 13:55:56 +00:00
class ModVertexAlphaVisitor : public osg : : NodeVisitor
2012-02-21 16:38:15 +00:00
{
2015-04-14 13:55:56 +00:00
public :
ModVertexAlphaVisitor ( int meshType )
: osg : : NodeVisitor ( TRAVERSE_ALL_CHILDREN )
, mMeshType ( meshType )
{
}
2012-02-21 16:38:15 +00:00
2020-10-16 18:18:54 +00:00
void apply ( osg : : Drawable & drw ) override
2015-11-10 17:21:56 +00:00
{
osg : : Geometry * geom = drw . asGeometry ( ) ;
if ( ! geom )
return ;
2012-04-23 13:27:03 +00:00
2015-11-10 17:21:56 +00:00
osg : : ref_ptr < osg : : Vec4Array > colors = new osg : : Vec4Array ( geom - > getVertexArray ( ) - > getNumElements ( ) ) ;
for ( unsigned int i = 0 ; i < colors - > size ( ) ; + + i )
{
float alpha = 1.f ;
2016-01-03 17:20:34 +00:00
if ( mMeshType = = 0 ) alpha = ( i % 2 ) ? 0.f : 1.f ; // this is a cylinder, so every second vertex belongs to the bottom-most row
2015-11-10 17:21:56 +00:00
else if ( mMeshType = = 1 )
2015-04-14 13:55:56 +00:00
{
2015-11-10 17:21:56 +00:00
if ( i > = 49 & & i < = 64 ) alpha = 0.f ; // bottom-most row
else if ( i > = 33 & & i < = 48 ) alpha = 0.25098 ; // second row
else alpha = 1.f ;
}
else if ( mMeshType = = 2 )
{
2016-01-25 17:52:20 +00:00
if ( geom - > getColorArray ( ) )
{
osg : : Vec4Array * origColors = static_cast < osg : : Vec4Array * > ( geom - > getColorArray ( ) ) ;
alpha = ( ( * origColors ) [ i ] . x ( ) = = 1.f ) ? 1.f : 0.f ;
}
else
alpha = 1.f ;
2015-04-14 13:55:56 +00:00
}
2012-04-23 13:27:03 +00:00
2015-11-10 17:21:56 +00:00
( * colors ) [ i ] = osg : : Vec4f ( 0.f , 0.f , 0.f , alpha ) ;
2015-04-14 13:55:56 +00:00
}
2015-11-10 17:21:56 +00:00
geom - > setColorArray ( colors , osg : : Array : : BIND_PER_VERTEX ) ;
2015-01-29 23:51:12 +00:00
}
2012-02-29 09:13:25 +00:00
2015-04-14 13:55:56 +00:00
private :
int mMeshType ;
} ;
2012-02-21 15:15:38 +00:00
2015-11-03 22:15:43 +00:00
/// @brief Hides the node subgraph if the eye point is below water.
/// @note Must be added as cull callback.
/// @note Meant to be used on a node that is child of a CameraRelativeTransform.
2019-04-12 13:35:44 +00:00
/// The current view point must be retrieved by the CameraRelativeTransform since we can't get it anymore once we are in camera-relative space.
2015-11-03 22:15:43 +00:00
class UnderwaterSwitchCallback : public osg : : NodeCallback
{
public :
UnderwaterSwitchCallback ( CameraRelativeTransform * cameraRelativeTransform )
: mCameraRelativeTransform ( cameraRelativeTransform )
, mEnabled ( true )
, mWaterLevel ( 0.f )
{
}
2018-11-04 11:42:34 +00:00
bool isUnderwater ( )
2015-11-03 22:15:43 +00:00
{
2019-04-12 13:35:44 +00:00
osg : : Vec3f viewPoint = mCameraRelativeTransform - > getLastViewPoint ( ) ;
return mEnabled & & viewPoint . z ( ) < mWaterLevel ;
2018-11-04 11:42:34 +00:00
}
2015-11-03 22:15:43 +00:00
2020-10-16 18:18:54 +00:00
void operator ( ) ( osg : : Node * node , osg : : NodeVisitor * nv ) override
2018-11-04 11:42:34 +00:00
{
if ( isUnderwater ( ) )
2015-11-03 22:15:43 +00:00
return ;
traverse ( node , nv ) ;
}
void setEnabled ( bool enabled )
{
mEnabled = enabled ;
}
void setWaterLevel ( float waterLevel )
{
mWaterLevel = waterLevel ;
}
private :
osg : : ref_ptr < CameraRelativeTransform > mCameraRelativeTransform ;
bool mEnabled ;
float mWaterLevel ;
} ;
2015-08-06 02:03:21 +00:00
/// A base class for the sun and moons.
2015-08-07 05:08:18 +00:00
class CelestialBody
2015-04-15 16:50:50 +00:00
{
public :
2017-03-14 00:01:50 +00:00
CelestialBody ( osg : : Group * parentNode , float scaleFactor , int numUvSets , unsigned int visibleMask = ~ 0 )
: mVisibleMask ( visibleMask )
2015-04-15 16:50:50 +00:00
{
2016-03-10 12:17:01 +00:00
mGeom = createTexturedQuad ( numUvSets ) ;
2015-04-15 16:50:50 +00:00
mTransform = new osg : : PositionAttitudeTransform ;
2017-03-14 00:01:50 +00:00
mTransform - > setNodeMask ( mVisibleMask ) ;
2015-04-15 16:50:50 +00:00
mTransform - > setScale ( osg : : Vec3f ( 450 , 450 , 450 ) * scaleFactor ) ;
2016-03-10 12:17:01 +00:00
mTransform - > addChild ( mGeom ) ;
2015-04-15 16:50:50 +00:00
parentNode - > addChild ( mTransform ) ;
}
2015-08-07 05:08:18 +00:00
virtual ~ CelestialBody ( ) { }
2015-04-15 16:50:50 +00:00
2015-08-05 02:07:42 +00:00
virtual void adjustTransparency ( const float ratio ) = 0 ;
2015-04-15 16:50:50 +00:00
void setVisible ( bool visible )
{
2020-04-20 16:47:14 +00:00
mTransform - > setNodeMask ( visible ? mVisibleMask : 0 ) ;
2015-04-15 16:50:50 +00:00
}
protected :
2017-03-14 00:01:50 +00:00
unsigned int mVisibleMask ;
2015-08-05 02:07:42 +00:00
static const float mDistance ;
2015-04-15 16:50:50 +00:00
osg : : ref_ptr < osg : : PositionAttitudeTransform > mTransform ;
2016-03-10 12:17:01 +00:00
osg : : ref_ptr < osg : : Geometry > mGeom ;
2015-04-15 16:50:50 +00:00
} ;
2015-08-05 02:07:42 +00:00
const float CelestialBody : : mDistance = 1000.0f ;
2015-04-15 16:50:50 +00:00
class Sun : public CelestialBody
{
public :
2016-02-05 22:10:27 +00:00
Sun ( osg : : Group * parentNode , Resource : : ImageManager & imageManager )
2020-04-20 16:47:14 +00:00
: CelestialBody ( parentNode , 1.0f , 1 , Mask_Sun )
2015-09-19 16:08:31 +00:00
, mUpdater ( new Updater )
2015-04-15 16:50:50 +00:00
{
2015-09-19 16:08:31 +00:00
mTransform - > addUpdateCallback ( mUpdater ) ;
2016-02-05 22:10:27 +00:00
osg : : ref_ptr < osg : : Texture2D > sunTex ( new osg : : Texture2D ( imageManager . getImage ( " textures/tx_sun_05.dds " ) ) ) ;
2016-02-05 21:58:02 +00:00
sunTex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : CLAMP_TO_EDGE ) ;
sunTex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : CLAMP_TO_EDGE ) ;
2015-09-19 16:08:31 +00:00
2016-03-10 12:17:01 +00:00
mGeom - > getOrCreateStateSet ( ) - > setTextureAttributeAndModes ( 0 , sunTex , osg : : StateAttribute : : ON ) ;
2015-09-19 16:10:02 +00:00
2015-09-21 15:09:24 +00:00
osg : : ref_ptr < osg : : Group > queryNode ( new osg : : Group ) ;
2015-09-19 16:10:02 +00:00
// Need to render after the world geometry so we can correctly test for occlusions
2016-03-10 22:18:20 +00:00
osg : : StateSet * stateset = queryNode - > getOrCreateStateSet ( ) ;
stateset - > setRenderBinDetails ( RenderBin_OcclusionQuery , " RenderBin " ) ;
stateset - > setNestRenderBins ( false ) ;
2015-09-21 15:09:24 +00:00
// Set up alpha testing on the occlusion testing subgraph, that way we can get the occlusion tested fragments to match the circular shape of the sun
osg : : ref_ptr < osg : : AlphaFunc > alphaFunc ( new osg : : AlphaFunc ) ;
alphaFunc - > setFunction ( osg : : AlphaFunc : : GREATER , 0.8 ) ;
2016-03-10 22:18:20 +00:00
stateset - > setAttributeAndModes ( alphaFunc , osg : : StateAttribute : : ON ) ;
stateset - > setTextureAttributeAndModes ( 0 , sunTex , osg : : StateAttribute : : ON ) ;
stateset - > setAttributeAndModes ( createUnlitMaterial ( ) , osg : : StateAttribute : : ON ) ;
// Disable writing to the color buffer. We are using this geometry for visibility tests only.
osg : : ref_ptr < osg : : ColorMask > colormask ( new osg : : ColorMask ( 0 , 0 , 0 , 0 ) ) ;
stateset - > setAttributeAndModes ( colormask , osg : : StateAttribute : : ON ) ;
2015-09-19 16:10:02 +00:00
2015-09-21 15:09:24 +00:00
mTransform - > addChild ( queryNode ) ;
2015-09-19 16:10:02 +00:00
2015-09-21 15:09:24 +00:00
mOcclusionQueryVisiblePixels = createOcclusionQueryNode ( queryNode , true ) ;
mOcclusionQueryTotalPixels = createOcclusionQueryNode ( queryNode , false ) ;
2015-09-19 16:22:16 +00:00
2016-02-05 22:10:27 +00:00
createSunFlash ( imageManager ) ;
2015-09-21 14:03:30 +00:00
createSunGlare ( ) ;
2015-04-15 16:50:50 +00:00
}
2015-08-05 02:07:42 +00:00
2015-08-07 05:08:18 +00:00
~ Sun ( )
2015-08-05 02:07:42 +00:00
{
2015-09-19 16:08:31 +00:00
mTransform - > removeUpdateCallback ( mUpdater ) ;
2015-09-21 14:03:30 +00:00
destroySunFlash ( ) ;
destroySunGlare ( ) ;
2015-08-05 02:07:42 +00:00
}
2015-09-21 17:43:48 +00:00
void setColor ( const osg : : Vec4f & color )
{
mUpdater - > mColor . r ( ) = color . r ( ) ;
mUpdater - > mColor . g ( ) = color . g ( ) ;
mUpdater - > mColor . b ( ) = color . b ( ) ;
}
2020-10-16 18:18:54 +00:00
void adjustTransparency ( const float ratio ) override
2015-08-05 02:07:42 +00:00
{
2015-08-07 05:08:18 +00:00
mUpdater - > mColor . a ( ) = ratio ;
2015-09-21 14:03:30 +00:00
if ( mSunGlareCallback )
mSunGlareCallback - > setGlareView ( ratio ) ;
2015-09-21 15:32:57 +00:00
if ( mSunFlashCallback )
mSunFlashCallback - > setGlareView ( ratio ) ;
2015-08-05 02:07:42 +00:00
}
void setDirection ( const osg : : Vec3f & direction )
{
osg : : Vec3f normalizedDirection = direction / direction . length ( ) ;
mTransform - > setPosition ( normalizedDirection * mDistance ) ;
osg : : Quat quat ;
quat . makeRotate ( osg : : Vec3f ( 0.0f , 0.0f , 1.0f ) , normalizedDirection ) ;
mTransform - > setAttitude ( quat ) ;
}
2015-09-21 14:03:30 +00:00
void setGlareTimeOfDayFade ( float val )
{
if ( mSunGlareCallback )
mSunGlareCallback - > setTimeOfDayFade ( val ) ;
}
2015-08-05 02:07:42 +00:00
private :
2016-03-10 23:09:30 +00:00
class DummyComputeBoundCallback : public osg : : Node : : ComputeBoundingSphereCallback
{
public :
2020-10-16 18:18:54 +00:00
osg : : BoundingSphere computeBound ( const osg : : Node & node ) const override { return osg : : BoundingSphere ( ) ; }
2016-03-10 23:09:30 +00:00
} ;
2015-09-19 16:10:02 +00:00
/// @param queryVisible If true, queries the amount of visible pixels. If false, queries the total amount of pixels.
osg : : ref_ptr < osg : : OcclusionQueryNode > createOcclusionQueryNode ( osg : : Group * parent , bool queryVisible )
{
osg : : ref_ptr < osg : : OcclusionQueryNode > oqn = new osg : : OcclusionQueryNode ;
oqn - > setQueriesEnabled ( true ) ;
2020-02-13 18:15:33 +00:00
# if OSG_VERSION_GREATER_OR_EQUAL(3, 6, 5)
// With OSG 3.6.5, the method of providing user defined query geometry has been completely replaced
osg : : ref_ptr < osg : : QueryGeometry > queryGeom = new osg : : QueryGeometry ( oqn - > getName ( ) ) ;
# else
2020-02-21 06:51:50 +00:00
osg : : ref_ptr < osg : : QueryGeometry > queryGeom = oqn - > getQueryGeometry ( ) ;
2020-02-13 18:15:33 +00:00
# endif
2015-09-19 16:10:02 +00:00
// Make it fast! A DYNAMIC query geometry means we can't break frame until the flare is rendered (which is rendered after all the other geometry,
// so that would be pretty bad). STATIC should be safe, since our node's local bounds are static, thus computeBounds() which modifies the queryGeometry
// is only called once.
// Note the debug geometry setDebugDisplay(true) is always DYNAMIC and that can't be changed, not a big deal.
2020-02-13 18:15:33 +00:00
queryGeom - > setDataVariance ( osg : : Object : : STATIC ) ;
2015-09-19 16:10:02 +00:00
2016-03-10 23:09:30 +00:00
// Set up the query geometry to match the actual sun's rendering shape. osg::OcclusionQueryNode wasn't originally intended to allow this,
// normally it would automatically adjust the query geometry to match the sub graph's bounding box. The below hack is needed to
// circumvent this.
queryGeom - > setVertexArray ( mGeom - > getVertexArray ( ) ) ;
queryGeom - > setTexCoordArray ( 0 , mGeom - > getTexCoordArray ( 0 ) , osg : : Array : : BIND_PER_VERTEX ) ;
2020-02-15 02:12:36 +00:00
queryGeom - > removePrimitiveSet ( 0 , queryGeom - > getNumPrimitiveSets ( ) ) ;
2016-03-10 23:09:30 +00:00
queryGeom - > addPrimitiveSet ( mGeom - > getPrimitiveSet ( 0 ) ) ;
// Hack to disable unwanted awful code inside OcclusionQueryNode::computeBound.
oqn - > setComputeBoundingSphereCallback ( new DummyComputeBoundCallback ) ;
// Still need a proper bounding sphere.
oqn - > setInitialBound ( queryGeom - > getBound ( ) ) ;
2015-09-21 15:09:24 +00:00
2020-02-13 18:15:33 +00:00
# if OSG_VERSION_GREATER_OR_EQUAL(3, 6, 5)
oqn - > setQueryGeometry ( queryGeom . release ( ) ) ;
# endif
2016-03-10 22:18:20 +00:00
osg : : StateSet * queryStateSet = new osg : : StateSet ;
2015-09-19 16:10:02 +00:00
if ( queryVisible )
{
osg : : ref_ptr < osg : : Depth > depth ( new osg : : Depth ) ;
2021-03-17 01:46:04 +00:00
depth - > setFunction ( osg : : Depth : : LEQUAL ) ;
2015-09-19 16:10:02 +00:00
// This is a trick to make fragments written by the query always use the maximum depth value,
// without having to retrieve the current far clipping distance.
// We want the sun glare to be "infinitely" far away.
depth - > setZNear ( 1.0 ) ;
depth - > setZFar ( 1.0 ) ;
2016-03-10 22:18:20 +00:00
depth - > setWriteMask ( false ) ;
queryStateSet - > setAttributeAndModes ( depth , osg : : StateAttribute : : ON ) ;
2015-09-19 16:10:02 +00:00
}
else
{
2016-03-10 22:18:20 +00:00
queryStateSet - > setMode ( GL_DEPTH_TEST , osg : : StateAttribute : : OFF ) ;
2015-09-19 16:10:02 +00:00
}
2016-03-10 22:18:20 +00:00
oqn - > setQueryStateSet ( queryStateSet ) ;
2015-09-19 16:10:02 +00:00
parent - > addChild ( oqn ) ;
return oqn ;
}
2016-02-05 22:10:27 +00:00
void createSunFlash ( Resource : : ImageManager & imageManager )
2015-09-19 16:22:16 +00:00
{
2016-02-05 22:10:27 +00:00
osg : : ref_ptr < osg : : Texture2D > tex ( new osg : : Texture2D ( imageManager . getImage ( " textures/tx_sun_flash_grey_05.dds " ) ) ) ;
2016-02-05 21:58:02 +00:00
tex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : CLAMP_TO_EDGE ) ;
tex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : CLAMP_TO_EDGE ) ;
2015-09-19 16:22:16 +00:00
osg : : ref_ptr < osg : : PositionAttitudeTransform > transform ( new osg : : PositionAttitudeTransform ) ;
2015-09-21 15:50:36 +00:00
const float scale = 2.6f ;
transform - > setScale ( osg : : Vec3f ( scale , scale , scale ) ) ;
2015-09-19 16:22:16 +00:00
mTransform - > addChild ( transform ) ;
2016-03-10 12:17:01 +00:00
osg : : ref_ptr < osg : : Geometry > geom = createTexturedQuad ( ) ;
transform - > addChild ( geom ) ;
2015-09-19 16:22:16 +00:00
2016-03-10 12:17:01 +00:00
osg : : StateSet * stateset = geom - > getOrCreateStateSet ( ) ;
2015-09-19 16:22:16 +00:00
stateset - > setTextureAttributeAndModes ( 0 , tex , osg : : StateAttribute : : ON ) ;
stateset - > setMode ( GL_DEPTH_TEST , osg : : StateAttribute : : OFF ) ;
stateset - > setRenderBinDetails ( RenderBin_SunGlare , " RenderBin " ) ;
stateset - > setNestRenderBins ( false ) ;
2015-09-21 14:34:30 +00:00
mSunFlashNode = transform ;
mSunFlashCallback = new SunFlashCallback ( mOcclusionQueryVisiblePixels , mOcclusionQueryTotalPixels ) ;
mSunFlashNode - > addCullCallback ( mSunFlashCallback ) ;
2015-09-21 14:03:30 +00:00
}
void destroySunFlash ( )
{
2015-09-21 14:34:30 +00:00
if ( mSunFlashNode )
{
mSunFlashNode - > removeCullCallback ( mSunFlashCallback ) ;
2018-10-09 06:21:12 +00:00
mSunFlashCallback = nullptr ;
2015-09-21 14:34:30 +00:00
}
2015-09-21 14:03:30 +00:00
}
void createSunGlare ( )
{
osg : : ref_ptr < osg : : Camera > camera ( new osg : : Camera ) ;
camera - > setProjectionMatrix ( osg : : Matrix : : identity ( ) ) ;
camera - > setReferenceFrame ( osg : : Transform : : ABSOLUTE_RF ) ; // add to skyRoot instead?
camera - > setViewMatrix ( osg : : Matrix : : identity ( ) ) ;
2020-04-20 16:47:14 +00:00
camera - > setClearMask ( 0 ) ;
2015-09-21 14:03:30 +00:00
camera - > setRenderOrder ( osg : : Camera : : NESTED_RENDER ) ;
camera - > setAllowEventFocus ( false ) ;
osg : : ref_ptr < osg : : Geometry > geom = osg : : createTexturedQuadGeometry ( osg : : Vec3f ( - 1 , - 1 , 0 ) , osg : : Vec3f ( 2 , 0 , 0 ) , osg : : Vec3f ( 0 , 2 , 0 ) ) ;
2016-03-10 12:17:01 +00:00
camera - > addChild ( geom ) ;
2015-09-21 14:03:30 +00:00
osg : : StateSet * stateset = geom - > getOrCreateStateSet ( ) ;
2015-09-19 16:22:16 +00:00
2015-09-21 14:03:30 +00:00
stateset - > setRenderBinDetails ( RenderBin_SunGlare , " RenderBin " ) ;
stateset - > setNestRenderBins ( false ) ;
stateset - > setMode ( GL_DEPTH_TEST , osg : : StateAttribute : : OFF ) ;
// set up additive blending
osg : : ref_ptr < osg : : BlendFunc > blendFunc ( new osg : : BlendFunc ) ;
blendFunc - > setSource ( osg : : BlendFunc : : SRC_ALPHA ) ;
blendFunc - > setDestination ( osg : : BlendFunc : : ONE ) ;
stateset - > setAttributeAndModes ( blendFunc , osg : : StateAttribute : : ON ) ;
mSunGlareCallback = new SunGlareCallback ( mOcclusionQueryVisiblePixels , mOcclusionQueryTotalPixels , mTransform ) ;
mSunGlareNode = camera ;
mSunGlareNode - > addCullCallback ( mSunGlareCallback ) ;
mTransform - > addChild ( camera ) ;
}
void destroySunGlare ( )
{
2015-09-21 14:34:30 +00:00
if ( mSunGlareNode )
{
mSunGlareNode - > removeCullCallback ( mSunGlareCallback ) ;
2018-10-09 06:21:12 +00:00
mSunGlareCallback = nullptr ;
2015-09-21 14:34:30 +00:00
}
2015-09-19 16:22:16 +00:00
}
2015-09-21 14:03:30 +00:00
class Updater : public SceneUtil : : StateSetUpdater
2015-08-07 05:08:18 +00:00
{
2015-09-21 14:03:30 +00:00
public :
2015-08-07 05:08:18 +00:00
osg : : Vec4f mColor ;
2015-09-19 16:08:31 +00:00
Updater ( )
2015-09-21 17:43:48 +00:00
: mColor ( 1.f , 1.f , 1.f , 1.f )
2015-08-07 05:08:18 +00:00
{
}
2020-10-16 18:18:54 +00:00
void setDefaults ( osg : : StateSet * stateset ) override
2015-08-07 05:08:18 +00:00
{
2015-09-21 14:03:30 +00:00
stateset - > setAttributeAndModes ( createUnlitMaterial ( ) , osg : : StateAttribute : : ON ) ;
2015-08-07 05:08:18 +00:00
}
2020-10-16 18:18:54 +00:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * ) override
2015-08-07 05:08:18 +00:00
{
osg : : Material * mat = static_cast < osg : : Material * > ( stateset - > getAttribute ( osg : : StateAttribute : : MATERIAL ) ) ;
2015-09-21 17:43:48 +00:00
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0 , 0 , 0 , mColor . a ( ) ) ) ;
mat - > setEmission ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( mColor . r ( ) , mColor . g ( ) , mColor . b ( ) , 1 ) ) ;
2015-08-07 05:08:18 +00:00
}
} ;
2015-09-21 14:03:30 +00:00
class OcclusionCallback : public osg : : NodeCallback
{
public :
OcclusionCallback ( osg : : ref_ptr < osg : : OcclusionQueryNode > oqnVisible , osg : : ref_ptr < osg : : OcclusionQueryNode > oqnTotal )
: mOcclusionQueryVisiblePixels ( oqnVisible )
, mOcclusionQueryTotalPixels ( oqnTotal )
{
}
protected :
float getVisibleRatio ( osg : : Camera * camera )
{
int visible = mOcclusionQueryVisiblePixels - > getQueryGeometry ( ) - > getNumPixels ( camera ) ;
int total = mOcclusionQueryTotalPixels - > getQueryGeometry ( ) - > getNumPixels ( camera ) ;
float visibleRatio = 0.f ;
if ( total > 0 )
visibleRatio = static_cast < float > ( visible ) / static_cast < float > ( total ) ;
float dt = MWBase : : Environment : : get ( ) . getFrameDuration ( ) ;
float lastRatio = mLastRatio [ osg : : observer_ptr < osg : : Camera > ( camera ) ] ;
float change = dt * 10 ;
if ( visibleRatio > lastRatio )
visibleRatio = std : : min ( visibleRatio , lastRatio + change ) ;
else
visibleRatio = std : : max ( visibleRatio , lastRatio - change ) ;
mLastRatio [ osg : : observer_ptr < osg : : Camera > ( camera ) ] = visibleRatio ;
return visibleRatio ;
}
private :
osg : : ref_ptr < osg : : OcclusionQueryNode > mOcclusionQueryVisiblePixels ;
osg : : ref_ptr < osg : : OcclusionQueryNode > mOcclusionQueryTotalPixels ;
std : : map < osg : : observer_ptr < osg : : Camera > , float > mLastRatio ;
} ;
2015-09-21 14:34:30 +00:00
/// SunFlashCallback handles fading/scaling of a node depending on occlusion query result. Must be attached as a cull callback.
2015-09-21 14:03:30 +00:00
class SunFlashCallback : public OcclusionCallback
{
public :
2015-09-21 14:34:30 +00:00
SunFlashCallback ( osg : : ref_ptr < osg : : OcclusionQueryNode > oqnVisible , osg : : ref_ptr < osg : : OcclusionQueryNode > oqnTotal )
2015-09-21 14:03:30 +00:00
: OcclusionCallback ( oqnVisible , oqnTotal )
2015-09-21 15:32:57 +00:00
, mGlareView ( 1.f )
2015-09-21 14:03:30 +00:00
{
}
2020-10-16 18:18:54 +00:00
void operator ( ) ( osg : : Node * node , osg : : NodeVisitor * nv ) override
2015-09-21 14:03:30 +00:00
{
osgUtil : : CullVisitor * cv = static_cast < osgUtil : : CullVisitor * > ( nv ) ;
float visibleRatio = getVisibleRatio ( cv - > getCurrentCamera ( ) ) ;
2015-09-21 15:32:57 +00:00
osg : : ref_ptr < osg : : StateSet > stateset ;
2015-09-21 14:34:30 +00:00
if ( visibleRatio > 0.f )
{
2015-09-21 15:32:57 +00:00
const float fadeThreshold = 0.1 ;
if ( visibleRatio < fadeThreshold )
{
float fade = 1.f - ( fadeThreshold - visibleRatio ) / fadeThreshold ;
osg : : ref_ptr < osg : : Material > mat ( createUnlitMaterial ( ) ) ;
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0 , 0 , 0 , fade * mGlareView ) ) ;
stateset = new osg : : StateSet ;
stateset - > setAttributeAndModes ( mat , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
}
2015-09-21 15:09:24 +00:00
const float threshold = 0.6 ;
2015-09-21 14:34:30 +00:00
visibleRatio = visibleRatio * ( 1.f - threshold ) + threshold ;
}
2015-09-21 14:03:30 +00:00
2015-09-21 14:34:30 +00:00
float scale = visibleRatio ;
2015-09-21 14:03:30 +00:00
2015-09-21 14:34:30 +00:00
if ( scale = = 0.f )
{
// no traverse
return ;
}
else
{
osg : : Matrix modelView = * cv - > getModelViewMatrix ( ) ;
2015-09-21 14:03:30 +00:00
2015-09-21 14:34:30 +00:00
modelView . preMultScale ( osg : : Vec3f ( visibleRatio , visibleRatio , visibleRatio ) ) ;
2015-09-21 15:32:57 +00:00
if ( stateset )
cv - > pushStateSet ( stateset ) ;
2015-09-21 14:34:30 +00:00
cv - > pushModelViewMatrix ( new osg : : RefMatrix ( modelView ) , osg : : Transform : : RELATIVE_RF ) ;
traverse ( node , nv ) ;
cv - > popModelViewMatrix ( ) ;
2015-09-21 15:32:57 +00:00
if ( stateset )
cv - > popStateSet ( ) ;
2015-09-21 14:34:30 +00:00
}
}
2015-09-21 15:32:57 +00:00
void setGlareView ( float value )
{
mGlareView = value ;
}
private :
float mGlareView ;
2015-09-21 14:03:30 +00:00
} ;
/// SunGlareCallback controls a full-screen glare effect depending on occlusion query result and the angle between sun and camera.
/// Must be attached as a cull callback to the node above the glare node.
class SunGlareCallback : public OcclusionCallback
{
public :
SunGlareCallback ( osg : : ref_ptr < osg : : OcclusionQueryNode > oqnVisible , osg : : ref_ptr < osg : : OcclusionQueryNode > oqnTotal ,
osg : : ref_ptr < osg : : PositionAttitudeTransform > sunTransform )
: OcclusionCallback ( oqnVisible , oqnTotal )
, mSunTransform ( sunTransform )
, mTimeOfDayFade ( 1.f )
, mGlareView ( 1.f )
{
2019-01-22 06:08:48 +00:00
mColor = Fallback : : Map : : getColour ( " Weather_Sun_Glare_Fader_Color " ) ;
mSunGlareFaderMax = Fallback : : Map : : getFloat ( " Weather_Sun_Glare_Fader_Max " ) ;
mSunGlareFaderAngleMax = Fallback : : Map : : getFloat ( " Weather_Sun_Glare_Fader_Angle_Max " ) ;
2015-10-13 14:35:31 +00:00
// Replicating a design flaw in MW. The color was being set on both ambient and emissive properties, which multiplies the result by two,
// then finally gets clamped by the fixed function pipeline. With the default INI settings, only the red component gets clamped,
// so the resulting color looks more orange than red.
mColor * = 2 ;
for ( int i = 0 ; i < 3 ; + + i )
mColor [ i ] = std : : min ( 1.f , mColor [ i ] ) ;
2015-09-21 14:03:30 +00:00
}
2020-10-16 18:18:54 +00:00
void operator ( ) ( osg : : Node * node , osg : : NodeVisitor * nv ) override
2015-09-21 14:03:30 +00:00
{
osgUtil : : CullVisitor * cv = static_cast < osgUtil : : CullVisitor * > ( nv ) ;
2015-10-24 13:46:15 +00:00
float angleRadians = getAngleToSunInRadians ( * cv - > getCurrentRenderStage ( ) - > getInitialViewMatrix ( ) ) ;
2015-09-21 14:03:30 +00:00
float visibleRatio = getVisibleRatio ( cv - > getCurrentCamera ( ) ) ;
2015-10-13 14:35:31 +00:00
const float angleMaxRadians = osg : : DegreesToRadians ( mSunGlareFaderAngleMax ) ;
2015-09-21 14:03:30 +00:00
float value = 1.f - std : : min ( 1.f , angleRadians / angleMaxRadians ) ;
2015-10-13 14:35:31 +00:00
float fade = value * mSunGlareFaderMax ;
2015-09-21 14:03:30 +00:00
fade * = mTimeOfDayFade * mGlareView * visibleRatio ;
if ( fade = = 0.f )
{
// no traverse
return ;
}
else
{
osg : : ref_ptr < osg : : StateSet > stateset ( new osg : : StateSet ) ;
osg : : ref_ptr < osg : : Material > mat ( createUnlitMaterial ( ) ) ;
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0 , 0 , 0 , fade ) ) ;
2015-10-13 14:35:31 +00:00
mat - > setEmission ( osg : : Material : : FRONT_AND_BACK , mColor ) ;
2015-09-21 14:03:30 +00:00
stateset - > setAttributeAndModes ( mat , osg : : StateAttribute : : ON ) ;
cv - > pushStateSet ( stateset ) ;
traverse ( node , nv ) ;
cv - > popStateSet ( ) ;
}
}
void setTimeOfDayFade ( float val )
{
mTimeOfDayFade = val ;
}
void setGlareView ( float glareView )
{
mGlareView = glareView ;
}
private :
2015-10-24 13:46:15 +00:00
float getAngleToSunInRadians ( const osg : : Matrix & viewMatrix ) const
2015-09-21 14:03:30 +00:00
{
osg : : Vec3d eye , center , up ;
2015-10-24 13:46:15 +00:00
viewMatrix . getLookAt ( eye , center , up ) ;
2015-09-21 14:03:30 +00:00
osg : : Vec3d forward = center - eye ;
osg : : Vec3d sun = mSunTransform - > getPosition ( ) ;
forward . normalize ( ) ;
sun . normalize ( ) ;
float angleRadians = std : : acos ( forward * sun ) ;
return angleRadians ;
}
osg : : ref_ptr < osg : : PositionAttitudeTransform > mSunTransform ;
float mTimeOfDayFade ;
float mGlareView ;
2015-10-13 14:35:31 +00:00
osg : : Vec4f mColor ;
float mSunGlareFaderMax ;
float mSunGlareFaderAngleMax ;
2015-09-21 14:03:30 +00:00
} ;
2015-08-07 05:08:18 +00:00
osg : : ref_ptr < Updater > mUpdater ;
2015-09-21 14:03:30 +00:00
osg : : ref_ptr < SunFlashCallback > mSunFlashCallback ;
2015-09-21 14:34:30 +00:00
osg : : ref_ptr < osg : : Node > mSunFlashNode ;
2015-09-21 14:03:30 +00:00
osg : : ref_ptr < SunGlareCallback > mSunGlareCallback ;
osg : : ref_ptr < osg : : Node > mSunGlareNode ;
2015-09-19 16:10:02 +00:00
osg : : ref_ptr < osg : : OcclusionQueryNode > mOcclusionQueryVisiblePixels ;
osg : : ref_ptr < osg : : OcclusionQueryNode > mOcclusionQueryTotalPixels ;
2015-04-15 16:50:50 +00:00
} ;
class Moon : public CelestialBody
{
public :
enum Type
{
Type_Masser = 0 ,
Type_Secunda
} ;
2016-02-05 22:10:27 +00:00
Moon ( osg : : Group * parentNode , Resource : : ImageManager & imageManager , float scaleFactor , Type type )
2015-08-05 02:07:42 +00:00
: CelestialBody ( parentNode , scaleFactor , 2 )
2015-04-15 16:50:50 +00:00
, mType ( type )
2020-10-26 21:16:31 +00:00
, mPhase ( MoonState : : Phase : : Unspecified )
2016-02-05 22:10:27 +00:00
, mUpdater ( new Updater ( imageManager ) )
2015-04-15 16:50:50 +00:00
{
2020-10-26 21:16:31 +00:00
setPhase ( MoonState : : Phase : : Full ) ;
2015-07-30 04:57:45 +00:00
setVisible ( true ) ;
2015-04-15 16:50:50 +00:00
2016-03-10 12:17:01 +00:00
mGeom - > addUpdateCallback ( mUpdater ) ;
2015-08-05 02:07:42 +00:00
}
2015-08-07 05:08:18 +00:00
~ Moon ( )
2015-08-05 02:07:42 +00:00
{
2016-03-10 12:17:01 +00:00
mGeom - > removeUpdateCallback ( mUpdater ) ;
2015-08-05 02:07:42 +00:00
}
2020-10-16 18:18:54 +00:00
void adjustTransparency ( const float ratio ) override
2015-08-05 02:07:42 +00:00
{
2015-08-07 05:08:18 +00:00
mUpdater - > mTransparency * = ratio ;
2015-08-05 02:07:42 +00:00
}
2015-07-30 04:57:45 +00:00
void setState ( const MoonState & state )
{
2016-12-21 15:49:37 +00:00
float radsX = ( ( state . mRotationFromHorizon ) * static_cast < float > ( osg : : PI ) ) / 180.0f ;
float radsZ = ( ( state . mRotationFromNorth ) * static_cast < float > ( osg : : PI ) ) / 180.0f ;
2015-07-30 04:57:45 +00:00
2015-08-01 12:57:05 +00:00
osg : : Quat rotX ( radsX , osg : : Vec3f ( 1.0f , 0.0f , 0.0f ) ) ;
osg : : Quat rotZ ( radsZ , osg : : Vec3f ( 0.0f , 0.0f , 1.0f ) ) ;
osg : : Vec3f direction = rotX * rotZ * osg : : Vec3f ( 0.0f , 1.0f , 0.0f ) ;
2015-08-05 02:07:42 +00:00
mTransform - > setPosition ( direction * mDistance ) ;
2015-08-01 12:57:05 +00:00
// The moon quad is initially oriented facing down, so we need to offset its X-axis
// rotation to rotate it to face the camera when sitting at the horizon.
2016-12-21 15:49:37 +00:00
osg : : Quat attX ( ( - static_cast < float > ( osg : : PI ) / 2.0f ) + radsX , osg : : Vec3f ( 1.0f , 0.0f , 0.0f ) ) ;
2015-08-01 12:57:05 +00:00
mTransform - > setAttitude ( attX * rotZ ) ;
2015-07-30 04:57:45 +00:00
setPhase ( state . mPhase ) ;
2015-08-07 05:08:18 +00:00
mUpdater - > mTransparency = state . mMoonAlpha ;
mUpdater - > mShadowBlend = state . mShadowBlend ;
2015-07-30 04:57:45 +00:00
}
2015-04-15 16:50:50 +00:00
2015-08-05 02:07:42 +00:00
void setAtmosphereColor ( const osg : : Vec4f & color )
2015-04-15 16:50:50 +00:00
{
2015-08-07 05:08:18 +00:00
mUpdater - > mAtmosphereColor = color ;
2015-08-05 02:07:42 +00:00
}
2015-04-15 16:50:50 +00:00
2015-08-05 02:07:42 +00:00
void setColor ( const osg : : Vec4f & color )
{
2015-08-07 05:08:18 +00:00
mUpdater - > mMoonColor = color ;
2015-08-05 02:07:42 +00:00
}
2015-04-15 16:50:50 +00:00
2015-08-05 02:07:42 +00:00
unsigned int getPhaseInt ( ) const
{
2020-10-26 21:16:31 +00:00
if ( mPhase = = MoonState : : Phase : : New ) return 0 ;
else if ( mPhase = = MoonState : : Phase : : WaxingCrescent ) return 1 ;
else if ( mPhase = = MoonState : : Phase : : WaningCrescent ) return 1 ;
else if ( mPhase = = MoonState : : Phase : : FirstQuarter ) return 2 ;
else if ( mPhase = = MoonState : : Phase : : ThirdQuarter ) return 2 ;
else if ( mPhase = = MoonState : : Phase : : WaxingGibbous ) return 3 ;
else if ( mPhase = = MoonState : : Phase : : WaningGibbous ) return 3 ;
else if ( mPhase = = MoonState : : Phase : : Full ) return 4 ;
2015-08-05 02:07:42 +00:00
return 0 ;
}
private :
2015-08-07 05:08:18 +00:00
struct Updater : public SceneUtil : : StateSetUpdater
{
2016-02-05 22:10:27 +00:00
Resource : : ImageManager & mImageManager ;
2015-08-07 05:08:18 +00:00
osg : : ref_ptr < osg : : Texture2D > mPhaseTex ;
osg : : ref_ptr < osg : : Texture2D > mCircleTex ;
float mTransparency ;
float mShadowBlend ;
osg : : Vec4f mAtmosphereColor ;
osg : : Vec4f mMoonColor ;
2016-02-05 22:10:27 +00:00
Updater ( Resource : : ImageManager & imageManager )
: mImageManager ( imageManager )
2015-08-07 05:08:18 +00:00
, mPhaseTex ( )
, mCircleTex ( )
, mTransparency ( 1.0f )
, mShadowBlend ( 1.0f )
, mAtmosphereColor ( 1.0f , 1.0f , 1.0f , 1.0f )
, mMoonColor ( 1.0f , 1.0f , 1.0f , 1.0f )
{
}
2015-08-05 02:07:42 +00:00
2020-10-16 18:18:54 +00:00
void setDefaults ( osg : : StateSet * stateset ) override
2015-08-07 05:08:18 +00:00
{
stateset - > setTextureAttributeAndModes ( 0 , mPhaseTex , osg : : StateAttribute : : ON ) ;
osg : : ref_ptr < osg : : TexEnvCombine > texEnv = new osg : : TexEnvCombine ;
texEnv - > setCombine_RGB ( osg : : TexEnvCombine : : MODULATE ) ;
texEnv - > setSource0_RGB ( osg : : TexEnvCombine : : CONSTANT ) ;
texEnv - > setSource1_RGB ( osg : : TexEnvCombine : : TEXTURE ) ;
texEnv - > setConstantColor ( osg : : Vec4f ( 1.f , 0.f , 0.f , 1.f ) ) ; // mShadowBlend * mMoonColor
stateset - > setTextureAttributeAndModes ( 0 , texEnv , osg : : StateAttribute : : ON ) ;
stateset - > setTextureAttributeAndModes ( 1 , mCircleTex , osg : : StateAttribute : : ON ) ;
osg : : ref_ptr < osg : : TexEnvCombine > texEnv2 = new osg : : TexEnvCombine ;
texEnv2 - > setCombine_RGB ( osg : : TexEnvCombine : : ADD ) ;
texEnv2 - > setCombine_Alpha ( osg : : TexEnvCombine : : MODULATE ) ;
texEnv2 - > setSource0_Alpha ( osg : : TexEnvCombine : : TEXTURE ) ;
texEnv2 - > setSource1_Alpha ( osg : : TexEnvCombine : : CONSTANT ) ;
texEnv2 - > setSource0_RGB ( osg : : TexEnvCombine : : PREVIOUS ) ;
texEnv2 - > setSource1_RGB ( osg : : TexEnvCombine : : CONSTANT ) ;
texEnv2 - > setConstantColor ( osg : : Vec4f ( 0.f , 0.f , 0.f , 1.f ) ) ; // mAtmosphereColor.rgb, mTransparency
stateset - > setTextureAttributeAndModes ( 1 , texEnv2 , osg : : StateAttribute : : ON ) ;
stateset - > setAttributeAndModes ( createUnlitMaterial ( ) , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
}
2015-08-05 02:07:42 +00:00
2020-10-16 18:18:54 +00:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * ) override
2015-08-07 05:08:18 +00:00
{
osg : : TexEnvCombine * texEnv = static_cast < osg : : TexEnvCombine * > ( stateset - > getTextureAttribute ( 0 , osg : : StateAttribute : : TEXENV ) ) ;
texEnv - > setConstantColor ( mMoonColor * mShadowBlend ) ;
osg : : TexEnvCombine * texEnv2 = static_cast < osg : : TexEnvCombine * > ( stateset - > getTextureAttribute ( 1 , osg : : StateAttribute : : TEXENV ) ) ;
texEnv2 - > setConstantColor ( osg : : Vec4f ( mAtmosphereColor . x ( ) , mAtmosphereColor . y ( ) , mAtmosphereColor . z ( ) , mTransparency ) ) ;
}
void setTextures ( const std : : string & phaseTex , const std : : string & circleTex )
{
2016-02-05 22:10:27 +00:00
mPhaseTex = new osg : : Texture2D ( mImageManager . getImage ( phaseTex ) ) ;
2016-02-05 21:58:02 +00:00
mPhaseTex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : CLAMP_TO_EDGE ) ;
mPhaseTex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : CLAMP_TO_EDGE ) ;
2016-02-05 22:10:27 +00:00
mCircleTex = new osg : : Texture2D ( mImageManager . getImage ( circleTex ) ) ;
2016-02-05 21:58:02 +00:00
mCircleTex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : CLAMP_TO_EDGE ) ;
mCircleTex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : CLAMP_TO_EDGE ) ;
2015-08-07 05:08:18 +00:00
reset ( ) ;
}
} ;
Type mType ;
MoonState : : Phase mPhase ;
osg : : ref_ptr < Updater > mUpdater ;
2015-04-15 16:50:50 +00:00
2015-07-30 04:57:45 +00:00
void setPhase ( const MoonState : : Phase & phase )
2015-04-15 16:50:50 +00:00
{
2015-08-07 05:08:18 +00:00
if ( mPhase = = phase )
2015-04-15 16:50:50 +00:00
return ;
2015-08-07 05:08:18 +00:00
2015-04-15 16:50:50 +00:00
mPhase = phase ;
std : : string textureName = " textures/tx_ " ;
2015-08-07 05:08:18 +00:00
if ( mType = = Moon : : Type_Secunda )
textureName + = " secunda_ " ;
else
textureName + = " masser_ " ;
2015-04-15 16:50:50 +00:00
2020-10-26 21:16:31 +00:00
if ( phase = = MoonState : : Phase : : New ) textureName + = " new " ;
else if ( phase = = MoonState : : Phase : : WaxingCrescent ) textureName + = " one_wax " ;
else if ( phase = = MoonState : : Phase : : FirstQuarter ) textureName + = " half_wax " ;
else if ( phase = = MoonState : : Phase : : WaxingGibbous ) textureName + = " three_wax " ;
else if ( phase = = MoonState : : Phase : : WaningCrescent ) textureName + = " one_wan " ;
else if ( phase = = MoonState : : Phase : : ThirdQuarter ) textureName + = " half_wan " ;
else if ( phase = = MoonState : : Phase : : WaningGibbous ) textureName + = " three_wan " ;
else if ( phase = = MoonState : : Phase : : Full ) textureName + = " full " ;
2015-04-15 16:50:50 +00:00
textureName + = " .dds " ;
if ( mType = = Moon : : Type_Secunda )
2015-08-07 05:08:18 +00:00
mUpdater - > setTextures ( textureName , " textures/tx_mooncircle_full_s.dds " ) ;
2015-04-15 16:50:50 +00:00
else
2015-08-07 05:08:18 +00:00
mUpdater - > setTextures ( textureName , " textures/tx_mooncircle_full_m.dds " ) ;
2015-04-15 16:50:50 +00:00
}
} ;
2017-10-05 10:58:56 +00:00
SkyManager : : SkyManager ( osg : : Group * parentNode , Resource : : SceneManager * sceneManager )
2015-04-14 13:55:56 +00:00
: mSceneManager ( sceneManager )
2018-10-09 06:21:12 +00:00
, mCamera ( nullptr )
2015-06-16 14:33:31 +00:00
, mAtmosphereNightRoll ( 0.f )
2015-05-04 00:41:50 +00:00
, mCreated ( false )
2015-04-25 18:37:42 +00:00
, mIsStorm ( false )
2012-03-26 22:34:06 +00:00
, mDay ( 0 )
, mMonth ( 0 )
2015-04-25 18:37:42 +00:00
, mCloudAnimationTimer ( 0.f )
2015-05-04 00:41:50 +00:00
, mRainTimer ( 0.f )
2019-10-19 19:18:58 +00:00
, mStormDirection ( 0 , 1 , 0 )
2012-03-26 22:34:06 +00:00
, mClouds ( )
, mNextClouds ( )
, mCloudBlendFactor ( 0.0f )
, mCloudSpeed ( 0.0f )
, mStarsOpacity ( 0.0f )
, mRemainingTransitionTime ( 0.0f )
2015-04-25 18:37:42 +00:00
, mRainEnabled ( false )
, mRainSpeed ( 0 )
2019-10-12 15:35:06 +00:00
, mRainDiameter ( 0 )
, mRainMinHeight ( 0 )
, mRainMaxHeight ( 0 )
, mRainEntranceSpeed ( 1 )
, mRainMaxRaindrops ( 0 )
2015-06-25 15:23:01 +00:00
, mWindSpeed ( 0.f )
2021-01-13 09:41:02 +00:00
, mBaseWindSpeed ( 0.f )
2012-03-26 22:34:06 +00:00
, mEnabled ( true )
, mSunEnabled ( true )
2021-03-10 15:10:17 +00:00
, mPrecipitationAlpha ( 0.f )
2012-02-21 15:15:38 +00:00
{
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < CameraRelativeTransform > skyroot ( new CameraRelativeTransform ) ;
2017-02-02 20:46:25 +00:00
skyroot - > setName ( " Sky Root " ) ;
2016-02-16 21:28:06 +00:00
// Assign empty program to specify we don't want shaders
// The shaders generated by the SceneManager can't handle everything we need
2017-11-01 18:22:59 +00:00
skyroot - > getOrCreateStateSet ( ) - > setAttributeAndModes ( new osg : : Program ( ) , osg : : StateAttribute : : OVERRIDE | osg : : StateAttribute : : PROTECTED | osg : : StateAttribute : : ON ) ;
2018-02-26 14:29:31 +00:00
SceneUtil : : ShadowManager : : disableShadowsForStateSet ( skyroot - > getOrCreateStateSet ( ) ) ;
2015-10-26 20:36:19 +00:00
2020-04-20 16:47:14 +00:00
skyroot - > setNodeMask ( Mask_Sky ) ;
2015-04-14 13:55:56 +00:00
parentNode - > addChild ( skyroot ) ;
mRootNode = skyroot ;
2015-11-03 00:53:38 +00:00
mEarlyRenderBinRoot = new osg : : Group ;
// render before the world is rendered
mEarlyRenderBinRoot - > getOrCreateStateSet ( ) - > setRenderBinDetails ( RenderBin_Sky , " RenderBin " ) ;
2015-10-26 20:36:19 +00:00
// Prevent unwanted clipping by water reflection camera's clipping plane
2015-11-03 00:53:38 +00:00
mEarlyRenderBinRoot - > getOrCreateStateSet ( ) - > setMode ( GL_CLIP_PLANE0 , osg : : StateAttribute : : OFF ) ;
mRootNode - > addChild ( mEarlyRenderBinRoot ) ;
2015-11-03 22:15:43 +00:00
mUnderwaterSwitch = new UnderwaterSwitchCallback ( skyroot ) ;
2012-04-01 13:07:41 +00:00
}
2012-02-29 09:13:25 +00:00
2012-04-01 13:07:41 +00:00
void SkyManager : : create ( )
{
2012-07-23 16:19:34 +00:00
assert ( ! mCreated ) ;
2021-04-08 16:32:38 +00:00
mAtmosphereDay = mSceneManager - > getInstance ( Settings : : Manager : : getString ( " skyatmosphere " , " Models " ) , mEarlyRenderBinRoot ) ;
2015-04-14 13:55:56 +00:00
ModVertexAlphaVisitor modAtmosphere ( 0 ) ;
mAtmosphereDay - > accept ( modAtmosphere ) ;
2012-02-29 09:13:25 +00:00
2015-04-14 13:55:56 +00:00
mAtmosphereUpdater = new AtmosphereUpdater ;
mAtmosphereDay - > addUpdateCallback ( mAtmosphereUpdater ) ;
2013-02-25 14:12:03 +00:00
2015-06-16 14:33:31 +00:00
mAtmosphereNightNode = new osg : : PositionAttitudeTransform ;
2020-04-20 16:47:14 +00:00
mAtmosphereNightNode - > setNodeMask ( 0 ) ;
2015-11-03 00:53:38 +00:00
mEarlyRenderBinRoot - > addChild ( mAtmosphereNightNode ) ;
2015-06-16 14:33:31 +00:00
osg : : ref_ptr < osg : : Node > atmosphereNight ;
2021-04-08 16:32:38 +00:00
if ( mSceneManager - > getVFS ( ) - > exists ( Settings : : Manager : : getString ( " skynight02 " , " Models " ) ) )
atmosphereNight = mSceneManager - > getInstance ( Settings : : Manager : : getString ( " skynight02 " , " Models " ) , mAtmosphereNightNode ) ;
2015-04-14 13:55:56 +00:00
else
2021-04-08 16:32:38 +00:00
atmosphereNight = mSceneManager - > getInstance ( Settings : : Manager : : getString ( " skynight01 " , " Models " ) , mAtmosphereNightNode ) ;
2015-06-16 14:33:31 +00:00
atmosphereNight - > getOrCreateStateSet ( ) - > setAttributeAndModes ( createAlphaTrackingUnlitMaterial ( ) , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2015-04-14 13:55:56 +00:00
ModVertexAlphaVisitor modStars ( 2 ) ;
2015-06-16 14:33:31 +00:00
atmosphereNight - > accept ( modStars ) ;
2016-02-05 22:10:27 +00:00
mAtmosphereNightUpdater = new AtmosphereNightUpdater ( mSceneManager - > getImageManager ( ) ) ;
2015-06-19 18:55:04 +00:00
atmosphereNight - > addUpdateCallback ( mAtmosphereNightUpdater ) ;
2015-04-14 13:55:56 +00:00
2016-02-05 22:10:27 +00:00
mSun . reset ( new Sun ( mEarlyRenderBinRoot , * mSceneManager - > getImageManager ( ) ) ) ;
2015-04-14 13:55:56 +00:00
2019-01-22 06:08:48 +00:00
mMasser . reset ( new Moon ( mEarlyRenderBinRoot , * mSceneManager - > getImageManager ( ) , Fallback : : Map : : getFloat ( " Moons_Masser_Size " ) / 125 , Moon : : Type_Masser ) ) ;
mSecunda . reset ( new Moon ( mEarlyRenderBinRoot , * mSceneManager - > getImageManager ( ) , Fallback : : Map : : getFloat ( " Moons_Secunda_Size " ) / 125 , Moon : : Type_Secunda ) ) ;
2015-04-14 13:55:56 +00:00
2015-06-16 14:33:31 +00:00
mCloudNode = new osg : : PositionAttitudeTransform ;
2015-11-03 00:53:38 +00:00
mEarlyRenderBinRoot - > addChild ( mCloudNode ) ;
2021-04-08 16:32:38 +00:00
mCloudMesh = mSceneManager - > getInstance ( Settings : : Manager : : getString ( " skyclouds " , " Models " ) , mCloudNode ) ;
2015-04-14 13:55:56 +00:00
ModVertexAlphaVisitor modClouds ( 1 ) ;
2015-06-18 23:03:12 +00:00
mCloudMesh - > accept ( modClouds ) ;
2015-04-15 16:50:50 +00:00
mCloudUpdater = new CloudUpdater ;
2015-08-07 13:34:01 +00:00
mCloudUpdater - > setOpacity ( 1.f ) ;
2015-06-18 23:03:12 +00:00
mCloudMesh - > addUpdateCallback ( mCloudUpdater ) ;
2015-04-14 13:55:56 +00:00
2021-04-08 16:32:38 +00:00
mCloudMesh2 = mSceneManager - > getInstance ( Settings : : Manager : : getString ( " skyclouds " , " Models " ) , mCloudNode ) ;
2015-06-18 23:03:12 +00:00
mCloudMesh2 - > accept ( modClouds ) ;
mCloudUpdater2 = new CloudUpdater ;
2015-08-07 13:34:01 +00:00
mCloudUpdater2 - > setOpacity ( 0.f ) ;
2015-06-18 23:03:12 +00:00
mCloudMesh2 - > addUpdateCallback ( mCloudUpdater2 ) ;
2020-04-20 16:47:14 +00:00
mCloudMesh2 - > setNodeMask ( 0 ) ;
2015-04-14 13:55:56 +00:00
osg : : ref_ptr < osg : : Depth > depth = new osg : : Depth ;
depth - > setWriteMask ( false ) ;
2015-11-03 00:53:38 +00:00
mEarlyRenderBinRoot - > getOrCreateStateSet ( ) - > setAttributeAndModes ( depth , osg : : StateAttribute : : ON ) ;
mEarlyRenderBinRoot - > getOrCreateStateSet ( ) - > setMode ( GL_BLEND , osg : : StateAttribute : : ON ) ;
mEarlyRenderBinRoot - > getOrCreateStateSet ( ) - > setMode ( GL_FOG , osg : : StateAttribute : : OFF ) ;
2012-04-01 13:07:41 +00:00
2019-01-22 06:08:48 +00:00
mMoonScriptColor = Fallback : : Map : : getColour ( " Moons_Script_Color " ) ;
2015-06-19 00:51:01 +00:00
2012-04-01 13:07:41 +00:00
mCreated = true ;
2012-02-21 15:15:38 +00:00
}
2018-02-10 23:14:41 +00:00
class RainCounter : public osgParticle : : ConstantRateCounter
{
public :
2020-10-16 18:18:54 +00:00
int numParticlesToCreate ( double dt ) const override
2018-02-10 23:14:41 +00:00
{
// limit dt to avoid large particle emissions if there are jumps in the simulation time
// 0.2 seconds is the same cap as used in Engine's frame loop
dt = std : : min ( dt , 0.2 ) ;
return ConstantRateCounter : : numParticlesToCreate ( dt ) ;
}
} ;
2015-06-25 15:23:01 +00:00
class RainShooter : public osgParticle : : Shooter
{
public :
RainShooter ( )
: mAngle ( 0.f )
{
}
2020-10-16 18:18:54 +00:00
void shoot ( osgParticle : : Particle * particle ) const override
2015-06-25 15:23:01 +00:00
{
particle - > setVelocity ( mVelocity ) ;
particle - > setAngle ( osg : : Vec3f ( - mAngle , 0 , ( Misc : : Rng : : rollProbability ( ) * 2 - 1 ) * osg : : PI ) ) ;
}
void setVelocity ( const osg : : Vec3f & velocity )
{
mVelocity = velocity ;
}
void setAngle ( float angle )
{
mAngle = angle ;
}
2020-10-16 18:18:54 +00:00
osg : : Object * cloneType ( ) const override
2015-06-25 15:23:01 +00:00
{
return new RainShooter ;
}
2020-10-16 18:18:54 +00:00
osg : : Object * clone ( const osg : : CopyOp & ) const override
2015-06-25 15:23:01 +00:00
{
return new RainShooter ( * this ) ;
}
private :
osg : : Vec3f mVelocity ;
float mAngle ;
} ;
2015-06-29 18:19:46 +00:00
// Updater for alpha value on a node's StateSet. Assumes the node has an existing Material StateAttribute.
class AlphaFader : public SceneUtil : : StateSetUpdater
2015-06-25 15:38:08 +00:00
{
public :
2020-12-29 11:49:22 +00:00
/// @param alpha the variable alpha value is recovered from
AlphaFader ( float & alpha )
: mAlpha ( alpha )
2015-06-25 15:38:08 +00:00
{
}
2020-10-16 18:18:54 +00:00
void setDefaults ( osg : : StateSet * stateset ) override
2015-07-30 22:03:01 +00:00
{
// need to create a deep copy of StateAttributes we will modify
osg : : Material * mat = static_cast < osg : : Material * > ( stateset - > getAttribute ( osg : : StateAttribute : : MATERIAL ) ) ;
stateset - > setAttribute ( osg : : clone ( mat , osg : : CopyOp : : DEEP_COPY_ALL ) , osg : : StateAttribute : : ON ) ;
}
2020-10-16 18:18:54 +00:00
void apply ( osg : : StateSet * stateset , osg : : NodeVisitor * nv ) override
2015-06-29 18:19:46 +00:00
{
osg : : Material * mat = static_cast < osg : : Material * > ( stateset - > getAttribute ( osg : : StateAttribute : : MATERIAL ) ) ;
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 0 , 0 , 0 , mAlpha ) ) ;
}
2015-11-02 22:38:34 +00:00
// Helper for adding AlphaFaders to a subgraph
2015-06-29 18:19:46 +00:00
class SetupVisitor : public osg : : NodeVisitor
{
public :
2020-12-29 11:49:22 +00:00
SetupVisitor ( float & alpha )
2015-06-29 18:19:46 +00:00
: osg : : NodeVisitor ( TRAVERSE_ALL_CHILDREN )
2020-12-29 11:49:22 +00:00
, mAlpha ( alpha )
2015-06-29 18:19:46 +00:00
{
}
2020-10-16 18:18:54 +00:00
void apply ( osg : : Node & node ) override
2015-06-29 18:19:46 +00:00
{
if ( osg : : StateSet * stateset = node . getStateSet ( ) )
{
if ( stateset - > getAttribute ( osg : : StateAttribute : : MATERIAL ) )
{
2018-10-09 06:21:12 +00:00
SceneUtil : : CompositeStateSetUpdater * composite = nullptr ;
2015-07-05 19:44:17 +00:00
osg : : Callback * callback = node . getUpdateCallback ( ) ;
2017-10-16 18:23:56 +00:00
2015-06-29 18:19:46 +00:00
while ( callback )
{
if ( ( composite = dynamic_cast < SceneUtil : : CompositeStateSetUpdater * > ( callback ) ) )
break ;
2017-10-16 18:23:56 +00:00
2015-06-29 18:19:46 +00:00
callback = callback - > getNestedCallback ( ) ;
}
2020-12-29 11:49:22 +00:00
osg : : ref_ptr < AlphaFader > alphaFader ( new AlphaFader ( mAlpha ) ) ;
2015-11-02 22:38:34 +00:00
2015-06-29 18:19:46 +00:00
if ( composite )
2015-11-02 22:38:34 +00:00
composite - > addController ( alphaFader ) ;
2015-06-29 18:19:46 +00:00
else
2015-11-02 22:38:34 +00:00
node . addUpdateCallback ( alphaFader ) ;
2015-06-29 18:19:46 +00:00
}
}
2017-10-16 21:56:03 +00:00
2015-06-29 18:19:46 +00:00
traverse ( node ) ;
}
private :
2020-12-29 11:49:22 +00:00
float & mAlpha ;
2015-06-29 18:19:46 +00:00
} ;
2017-09-28 17:04:31 +00:00
protected :
2020-12-29 11:49:22 +00:00
float & mAlpha ;
2015-06-25 15:38:08 +00:00
} ;
2017-10-14 16:45:29 +00:00
void SkyManager : : setCamera ( osg : : Camera * camera )
{
mCamera = camera ;
}
class WrapAroundOperator : public osgParticle : : Operator
2017-10-04 14:41:55 +00:00
{
public :
2017-10-14 16:45:29 +00:00
WrapAroundOperator ( osg : : Camera * camera , const osg : : Vec3 & wrapRange ) : osgParticle : : Operator ( )
2017-10-04 14:41:55 +00:00
{
2017-10-14 16:45:29 +00:00
mCamera = camera ;
2017-10-12 10:56:03 +00:00
mWrapRange = wrapRange ;
2017-10-14 16:45:29 +00:00
mHalfWrapRange = mWrapRange / 2.0 ;
mPreviousCameraPosition = getCameraPosition ( ) ;
2017-10-04 14:41:55 +00:00
}
2020-10-16 18:18:54 +00:00
osg : : Object * cloneType ( ) const override
2017-10-04 14:41:55 +00:00
{
2018-10-09 06:21:12 +00:00
return nullptr ;
2017-10-14 16:45:29 +00:00
}
2017-10-12 13:28:54 +00:00
2020-10-16 18:18:54 +00:00
osg : : Object * clone ( const osg : : CopyOp & op ) const override
2017-10-14 16:45:29 +00:00
{
2018-10-09 06:21:12 +00:00
return nullptr ;
2017-10-14 16:45:29 +00:00
}
2017-10-05 10:58:56 +00:00
2020-10-16 18:18:54 +00:00
void operate ( osgParticle : : Particle * P , double dt ) override
2017-10-14 16:45:29 +00:00
{
}
2017-10-04 14:41:55 +00:00
2020-10-16 18:18:54 +00:00
void operateParticles ( osgParticle : : ParticleSystem * ps , double dt ) override
2017-10-14 16:45:29 +00:00
{
osg : : Vec3 position = getCameraPosition ( ) ;
osg : : Vec3 positionDifference = position - mPreviousCameraPosition ;
2017-10-04 14:41:55 +00:00
2017-10-08 18:56:36 +00:00
osg : : Matrix toWorld , toLocal ;
2017-10-04 14:41:55 +00:00
2017-10-14 16:45:29 +00:00
std : : vector < osg : : Matrix > worldMatrices = ps - > getWorldMatrices ( ) ;
2017-10-08 18:56:36 +00:00
if ( ! worldMatrices . empty ( ) )
2017-10-12 13:28:54 +00:00
{
toWorld = worldMatrices [ 0 ] ;
toLocal . invert ( toWorld ) ;
}
2017-10-05 10:58:56 +00:00
2017-10-14 16:45:29 +00:00
for ( int i = 0 ; i < ps - > numParticles ( ) ; + + i )
2017-10-12 13:28:54 +00:00
{
2017-10-14 16:45:29 +00:00
osgParticle : : Particle * p = ps - > getParticle ( i ) ;
p - > setPosition ( toWorld . preMult ( p - > getPosition ( ) ) ) ;
p - > setPosition ( p - > getPosition ( ) - positionDifference ) ;
2017-10-12 13:28:54 +00:00
2017-10-14 16:45:29 +00:00
for ( int j = 0 ; j < 3 ; + + j ) // wrap-around in all 3 dimensions
{
osg : : Vec3 pos = p - > getPosition ( ) ;
2017-10-12 13:28:54 +00:00
2017-10-14 16:45:29 +00:00
if ( pos [ j ] < - mHalfWrapRange [ j ] )
pos [ j ] = mHalfWrapRange [ j ] + fmod ( pos [ j ] - mHalfWrapRange [ j ] , mWrapRange [ j ] ) ;
else if ( pos [ j ] > mHalfWrapRange [ j ] )
pos [ j ] = fmod ( pos [ j ] + mHalfWrapRange [ j ] , mWrapRange [ j ] ) - mHalfWrapRange [ j ] ;
2017-10-12 13:28:54 +00:00
2017-10-14 16:45:29 +00:00
p - > setPosition ( pos ) ;
}
2017-10-05 10:58:56 +00:00
2017-10-14 16:45:29 +00:00
p - > setPosition ( toLocal . preMult ( p - > getPosition ( ) ) ) ;
2017-10-12 13:28:54 +00:00
}
2017-10-14 16:45:29 +00:00
mPreviousCameraPosition = position ;
2017-10-04 14:41:55 +00:00
}
protected :
2017-10-14 16:45:29 +00:00
osg : : Camera * mCamera ;
osg : : Vec3 mPreviousCameraPosition ;
2017-10-12 10:56:03 +00:00
osg : : Vec3 mWrapRange ;
2017-10-14 16:45:29 +00:00
osg : : Vec3 mHalfWrapRange ;
osg : : Vec3 getCameraPosition ( )
{
return mCamera - > getInverseViewMatrix ( ) . getTrans ( ) ;
}
2017-10-04 14:41:55 +00:00
} ;
2020-12-29 11:49:22 +00:00
class WeatherAlphaOperator : public osgParticle : : Operator
{
public :
WeatherAlphaOperator ( float & alpha , bool rain )
: mAlpha ( alpha )
, mIsRain ( rain )
{
}
osg : : Object * cloneType ( ) const override
{
return nullptr ;
}
osg : : Object * clone ( const osg : : CopyOp & op ) const override
{
return nullptr ;
}
void operate ( osgParticle : : Particle * particle , double dt ) override
{
constexpr float rainThreshold = 0.6f ; // Rain_Threshold?
const float alpha = mIsRain ? mAlpha * rainThreshold : mAlpha ;
particle - > setAlphaRange ( osgParticle : : rangef ( alpha , alpha ) ) ;
}
private :
float & mAlpha ;
bool mIsRain ;
} ;
2015-06-25 15:23:01 +00:00
void SkyManager : : createRain ( )
{
if ( mRainNode )
return ;
mRainNode = new osg : : Group ;
2020-12-29 11:49:22 +00:00
mRainParticleSystem = new NifOsg : : ParticleSystem ;
2019-10-12 15:35:06 +00:00
osg : : Vec3 rainRange = osg : : Vec3 ( mRainDiameter , mRainDiameter , ( mRainMinHeight + mRainMaxHeight ) / 2.f ) ;
2017-10-04 14:41:55 +00:00
2015-06-25 15:23:01 +00:00
mRainParticleSystem - > setParticleAlignment ( osgParticle : : ParticleSystem : : FIXED ) ;
mRainParticleSystem - > setAlignVectorX ( osg : : Vec3f ( 0.1 , 0 , 0 ) ) ;
2016-11-17 16:51:55 +00:00
mRainParticleSystem - > setAlignVectorY ( osg : : Vec3f ( 0 , 0 , 1 ) ) ;
2015-06-25 15:23:01 +00:00
osg : : ref_ptr < osg : : StateSet > stateset ( mRainParticleSystem - > getOrCreateStateSet ( ) ) ;
2016-02-05 21:58:02 +00:00
2016-02-05 22:10:27 +00:00
osg : : ref_ptr < osg : : Texture2D > raindropTex ( new osg : : Texture2D ( mSceneManager - > getImageManager ( ) - > getImage ( " textures/tx_raindrop_01.dds " ) ) ) ;
2016-02-05 21:58:02 +00:00
raindropTex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : CLAMP_TO_EDGE ) ;
raindropTex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : CLAMP_TO_EDGE ) ;
stateset - > setTextureAttributeAndModes ( 0 , raindropTex , osg : : StateAttribute : : ON ) ;
2015-06-25 15:23:01 +00:00
stateset - > setNestRenderBins ( false ) ;
stateset - > setRenderingHint ( osg : : StateSet : : TRANSPARENT_BIN ) ;
stateset - > setMode ( GL_CULL_FACE , osg : : StateAttribute : : OFF ) ;
2015-11-03 00:53:38 +00:00
stateset - > setMode ( GL_BLEND , osg : : StateAttribute : : ON ) ;
2015-06-25 15:23:01 +00:00
2020-12-29 11:49:22 +00:00
osg : : ref_ptr < osg : : Material > mat ( new osg : : Material ) ;
mat - > setAmbient ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 1 , 1 , 1 , 1 ) ) ;
mat - > setDiffuse ( osg : : Material : : FRONT_AND_BACK , osg : : Vec4f ( 1 , 1 , 1 , 1 ) ) ;
mat - > setColorMode ( osg : : Material : : AMBIENT_AND_DIFFUSE ) ;
stateset - > setAttributeAndModes ( mat , osg : : StateAttribute : : ON ) ;
2015-06-25 15:23:01 +00:00
osgParticle : : Particle & particleTemplate = mRainParticleSystem - > getDefaultParticleTemplate ( ) ;
particleTemplate . setSizeRange ( osgParticle : : rangef ( 5.f , 15.f ) ) ;
particleTemplate . setAlphaRange ( osgParticle : : rangef ( 1.f , 1.f ) ) ;
particleTemplate . setLifeTime ( 1 ) ;
osg : : ref_ptr < osgParticle : : ModularEmitter > emitter ( new osgParticle : : ModularEmitter ) ;
emitter - > setParticleSystem ( mRainParticleSystem ) ;
osg : : ref_ptr < osgParticle : : BoxPlacer > placer ( new osgParticle : : BoxPlacer ) ;
2019-10-12 15:35:06 +00:00
placer - > setXRange ( - rainRange . x ( ) / 2 , rainRange . x ( ) / 2 ) ;
2017-10-12 13:28:54 +00:00
placer - > setYRange ( - rainRange . y ( ) / 2 , rainRange . y ( ) / 2 ) ;
2019-10-12 15:35:06 +00:00
placer - > setZRange ( - rainRange . z ( ) / 2 , rainRange . z ( ) / 2 ) ;
2015-06-25 15:23:01 +00:00
emitter - > setPlacer ( placer ) ;
2019-10-12 15:35:06 +00:00
mPlacer = placer ;
2015-06-25 15:23:01 +00:00
2019-10-12 15:35:06 +00:00
// FIXME: vanilla engine does not use a particle system to handle rain, it uses a NIF-file with 20 raindrops in it.
// It spawns the (maxRaindrops-getParticleSystem()->numParticles())*dt/rainEntranceSpeed batches every frame (near 1-2).
// Since the rain is a regular geometry, it produces water ripples, also in theory it can be removed if collides with something.
2018-02-10 23:14:41 +00:00
osg : : ref_ptr < RainCounter > counter ( new RainCounter ) ;
2019-10-12 15:35:06 +00:00
counter - > setNumberOfParticlesPerSecondToCreate ( mRainMaxRaindrops / mRainEntranceSpeed * 20 ) ;
2015-06-25 15:23:01 +00:00
emitter - > setCounter ( counter ) ;
2019-10-12 15:35:06 +00:00
mCounter = counter ;
2015-06-25 15:23:01 +00:00
osg : : ref_ptr < RainShooter > shooter ( new RainShooter ) ;
mRainShooter = shooter ;
emitter - > setShooter ( shooter ) ;
osg : : ref_ptr < osgParticle : : ParticleSystemUpdater > updater ( new osgParticle : : ParticleSystemUpdater ) ;
updater - > addParticleSystem ( mRainParticleSystem ) ;
2017-10-14 16:45:29 +00:00
osg : : ref_ptr < osgParticle : : ModularProgram > program ( new osgParticle : : ModularProgram ) ;
program - > addOperator ( new WrapAroundOperator ( mCamera , rainRange ) ) ;
2021-03-10 15:10:17 +00:00
program - > addOperator ( new WeatherAlphaOperator ( mPrecipitationAlpha , true ) ) ;
2017-10-14 16:45:29 +00:00
program - > setParticleSystem ( mRainParticleSystem ) ;
mRainNode - > addChild ( program ) ;
2015-06-25 15:23:01 +00:00
mRainNode - > addChild ( emitter ) ;
2016-03-10 12:17:01 +00:00
mRainNode - > addChild ( mRainParticleSystem ) ;
2015-06-25 15:23:01 +00:00
mRainNode - > addChild ( updater ) ;
2020-12-29 11:49:22 +00:00
// Note: if we ever switch to regular geometry rain, it'll need to use an AlphaFader.
2015-11-03 22:15:43 +00:00
mRainNode - > addCullCallback ( mUnderwaterSwitch ) ;
2020-04-20 16:47:14 +00:00
mRainNode - > setNodeMask ( Mask_WeatherParticles ) ;
2015-06-25 15:38:08 +00:00
2015-06-25 15:23:01 +00:00
mRootNode - > addChild ( mRainNode ) ;
}
void SkyManager : : destroyRain ( )
{
if ( ! mRainNode )
return ;
mRootNode - > removeChild ( mRainNode ) ;
2018-10-09 06:21:12 +00:00
mRainNode = nullptr ;
2019-10-12 15:35:06 +00:00
mPlacer = nullptr ;
mCounter = nullptr ;
2018-10-09 06:21:12 +00:00
mRainParticleSystem = nullptr ;
mRainShooter = nullptr ;
2015-06-25 15:23:01 +00:00
}
2012-02-21 15:15:38 +00:00
SkyManager : : ~ SkyManager ( )
{
2015-04-30 21:21:25 +00:00
if ( mRootNode )
{
mRootNode - > getParent ( 0 ) - > removeChild ( mRootNode ) ;
2018-10-09 06:21:12 +00:00
mRootNode = nullptr ;
2015-04-30 21:21:25 +00:00
}
2012-02-21 15:15:38 +00:00
}
2012-02-21 16:38:15 +00:00
int SkyManager : : getMasserPhase ( ) const
{
2012-04-01 13:07:41 +00:00
if ( ! mCreated ) return 0 ;
2015-04-15 20:38:43 +00:00
return mMasser - > getPhaseInt ( ) ;
2012-02-21 16:38:15 +00:00
}
int SkyManager : : getSecundaPhase ( ) const
{
2012-04-01 13:07:41 +00:00
if ( ! mCreated ) return 0 ;
2015-04-15 16:50:50 +00:00
return mSecunda - > getPhaseInt ( ) ;
2012-02-21 16:38:15 +00:00
}
2017-09-29 17:56:16 +00:00
bool SkyManager : : isEnabled ( )
{
return mEnabled ;
}
2021-03-08 06:58:51 +00:00
bool SkyManager : : hasRain ( ) const
2017-09-29 17:56:16 +00:00
{
2018-10-09 06:21:12 +00:00
return mRainNode ! = nullptr ;
2017-09-29 17:56:16 +00:00
}
2021-03-10 15:10:17 +00:00
float SkyManager : : getPrecipitationAlpha ( ) const
2012-02-26 17:21:11 +00:00
{
2021-03-08 06:58:51 +00:00
if ( mEnabled & & ! mIsStorm & & ( hasRain ( ) | | mParticleNode ) )
2021-03-10 15:10:17 +00:00
return mPrecipitationAlpha ;
2020-12-29 11:49:22 +00:00
2021-03-08 06:58:51 +00:00
return 0.f ;
}
2012-08-15 11:17:35 +00:00
2021-03-08 06:58:51 +00:00
void SkyManager : : update ( float duration )
{
2021-03-11 07:46:44 +00:00
if ( ! mEnabled )
return ;
2018-11-04 11:42:34 +00:00
switchUnderwaterRain ( ) ;
2015-06-16 14:33:31 +00:00
if ( mIsStorm )
{
osg : : Quat quat ;
quat . makeRotate ( osg : : Vec3f ( 0 , 1 , 0 ) , mStormDirection ) ;
2019-10-19 19:18:58 +00:00
mCloudNode - > setAttitude ( quat ) ;
2017-10-08 18:56:36 +00:00
if ( mParticleNode )
2019-10-19 19:18:58 +00:00
{
// Morrowind deliberately rotates the blizzard mesh, so so should we.
2021-04-08 16:32:38 +00:00
if ( mCurrentParticleEffect = = Settings : : Manager : : getString ( " weatherblizzard " , " Models " ) )
2019-10-19 19:18:58 +00:00
quat . makeRotate ( osg : : Vec3f ( - 1 , 0 , 0 ) , mStormDirection ) ;
2017-10-08 18:56:36 +00:00
mParticleNode - > setAttitude ( quat ) ;
2019-10-19 19:18:58 +00:00
}
2015-06-16 14:33:31 +00:00
}
else
mCloudNode - > setAttitude ( osg : : Quat ( ) ) ;
2014-06-26 17:01:49 +00:00
2012-02-21 15:15:38 +00:00
// UV Scroll the clouds
2015-06-04 18:08:44 +00:00
mCloudAnimationTimer + = duration * mCloudSpeed * 0.003 ;
mCloudUpdater - > setAnimationTimer ( mCloudAnimationTimer ) ;
2015-06-18 23:03:12 +00:00
mCloudUpdater2 - > setAnimationTimer ( mCloudAnimationTimer ) ;
2012-02-29 09:13:25 +00:00
2012-03-16 18:02:33 +00:00
// rotate the stars by 360 degrees every 4 days
2015-06-16 14:33:31 +00:00
mAtmosphereNightRoll + = MWBase : : Environment : : get ( ) . getWorld ( ) - > getTimeScaleFactor ( ) * duration * osg : : DegreesToRadians ( 360.f ) / ( 3600 * 96.f ) ;
if ( mAtmosphereNightNode - > getNodeMask ( ) ! = 0 )
mAtmosphereNightNode - > setAttitude ( osg : : Quat ( mAtmosphereNightRoll , osg : : Vec3f ( 0 , 0 , 1 ) ) ) ;
2012-02-21 15:15:38 +00:00
}
2015-04-14 13:55:56 +00:00
void SkyManager : : setEnabled ( bool enabled )
2012-02-21 15:15:38 +00:00
{
2015-04-14 13:55:56 +00:00
if ( enabled & & ! mCreated )
2012-04-01 13:07:41 +00:00
create ( ) ;
2020-04-20 16:47:14 +00:00
mRootNode - > setNodeMask ( enabled ? Mask_Sky : 0 ) ;
2012-02-21 15:15:38 +00:00
2015-04-14 13:55:56 +00:00
mEnabled = enabled ;
2012-02-21 15:15:38 +00:00
}
void SkyManager : : setMoonColour ( bool red )
{
2015-06-19 16:19:52 +00:00
if ( ! mCreated ) return ;
mSecunda - > setColor ( red ? mMoonScriptColor : osg : : Vec4f ( 1 , 1 , 1 , 1 ) ) ;
2012-02-21 21:11:41 +00:00
}
2012-02-22 18:17:37 +00:00
2015-06-25 15:23:01 +00:00
void SkyManager : : updateRainParameters ( )
{
if ( mRainShooter )
{
2019-10-19 19:18:58 +00:00
float angle = - std : : atan ( mWindSpeed / 50.f ) ;
2019-10-12 15:35:06 +00:00
mRainShooter - > setVelocity ( osg : : Vec3f ( 0 , mRainSpeed * std : : sin ( angle ) , - mRainSpeed / std : : cos ( angle ) ) ) ;
2015-06-25 15:23:01 +00:00
mRainShooter - > setAngle ( angle ) ;
2019-10-12 15:35:06 +00:00
osg : : Vec3 rainRange = osg : : Vec3 ( mRainDiameter , mRainDiameter , ( mRainMinHeight + mRainMaxHeight ) / 2.f ) ;
mPlacer - > setXRange ( - rainRange . x ( ) / 2 , rainRange . x ( ) / 2 ) ;
mPlacer - > setYRange ( - rainRange . y ( ) / 2 , rainRange . y ( ) / 2 ) ;
mPlacer - > setZRange ( - rainRange . z ( ) / 2 , rainRange . z ( ) / 2 ) ;
mCounter - > setNumberOfParticlesPerSecondToCreate ( mRainMaxRaindrops / mRainEntranceSpeed * 20 ) ;
2015-06-25 15:23:01 +00:00
}
}
2018-11-04 11:42:34 +00:00
void SkyManager : : switchUnderwaterRain ( )
{
if ( ! mRainParticleSystem )
return ;
bool freeze = mUnderwaterSwitch - > isUnderwater ( ) ;
mRainParticleSystem - > setFrozen ( freeze ) ;
}
2015-08-05 02:07:42 +00:00
void SkyManager : : setWeather ( const WeatherResult & weather )
2012-02-22 18:17:37 +00:00
{
2012-04-01 13:07:41 +00:00
if ( ! mCreated ) return ;
2012-07-06 08:31:48 +00:00
2019-10-12 15:35:06 +00:00
mRainEntranceSpeed = weather . mRainEntranceSpeed ;
mRainMaxRaindrops = weather . mRainMaxRaindrops ;
mRainDiameter = weather . mRainDiameter ;
mRainMinHeight = weather . mRainMinHeight ;
mRainMaxHeight = weather . mRainMaxHeight ;
mRainSpeed = weather . mRainSpeed ;
mWindSpeed = weather . mWindSpeed ;
2021-01-13 09:41:02 +00:00
mBaseWindSpeed = weather . mBaseWindSpeed ;
2019-10-12 15:35:06 +00:00
2015-06-25 15:23:01 +00:00
if ( mRainEffect ! = weather . mRainEffect )
{
mRainEffect = weather . mRainEffect ;
if ( ! mRainEffect . empty ( ) )
{
createRain ( ) ;
}
else
{
destroyRain ( ) ;
}
}
updateRainParameters ( ) ;
2014-06-26 17:01:49 +00:00
mIsStorm = weather . mIsStorm ;
2014-06-25 16:20:21 +00:00
2014-06-24 13:00:15 +00:00
if ( mCurrentParticleEffect ! = weather . mParticleEffect )
{
mCurrentParticleEffect = weather . mParticleEffect ;
2015-11-02 20:35:03 +00:00
// cleanup old particles
if ( mParticleEffect )
{
mParticleNode - > removeChild ( mParticleEffect ) ;
2018-10-09 06:21:12 +00:00
mParticleEffect = nullptr ;
2015-11-02 20:35:03 +00:00
}
2014-06-24 13:00:15 +00:00
if ( mCurrentParticleEffect . empty ( ) )
{
2015-06-16 14:33:31 +00:00
if ( mParticleNode )
2015-06-26 00:30:39 +00:00
{
2015-06-16 14:33:31 +00:00
mRootNode - > removeChild ( mParticleNode ) ;
2018-10-09 06:21:12 +00:00
mParticleNode = nullptr ;
2015-06-26 00:30:39 +00:00
}
2014-06-24 13:00:15 +00:00
}
else
{
2015-06-16 14:33:31 +00:00
if ( ! mParticleNode )
{
2017-10-05 10:58:56 +00:00
mParticleNode = new osg : : PositionAttitudeTransform ;
2015-11-03 22:15:43 +00:00
mParticleNode - > addCullCallback ( mUnderwaterSwitch ) ;
2020-04-20 16:47:14 +00:00
mParticleNode - > setNodeMask ( Mask_WeatherParticles ) ;
2015-06-16 14:33:31 +00:00
mRootNode - > addChild ( mParticleNode ) ;
}
2017-10-16 18:23:56 +00:00
2016-02-09 17:33:02 +00:00
mParticleEffect = mSceneManager - > getInstance ( mCurrentParticleEffect , mParticleNode ) ;
2015-04-14 13:55:56 +00:00
2017-05-05 17:26:09 +00:00
SceneUtil : : AssignControllerSourcesVisitor assignVisitor ( std : : shared_ptr < SceneUtil : : ControllerSource > ( new SceneUtil : : FrameTimeSource ) ) ;
2015-04-19 18:07:18 +00:00
mParticleEffect - > accept ( assignVisitor ) ;
2015-06-29 18:19:46 +00:00
2021-03-10 15:10:17 +00:00
AlphaFader : : SetupVisitor alphaFaderSetupVisitor ( mPrecipitationAlpha ) ;
2017-10-16 18:23:56 +00:00
2015-06-29 18:19:46 +00:00
mParticleEffect - > accept ( alphaFaderSetupVisitor ) ;
2015-11-02 22:49:22 +00:00
SceneUtil : : DisableFreezeOnCullVisitor disableFreezeOnCullVisitor ;
mParticleEffect - > accept ( disableFreezeOnCullVisitor ) ;
2017-10-04 14:41:55 +00:00
2020-12-29 11:49:22 +00:00
SceneUtil : : FindByClassVisitor findPSVisitor ( std : : string ( " ParticleSystem " ) ) ;
mParticleEffect - > accept ( findPSVisitor ) ;
2017-10-14 16:45:29 +00:00
2020-12-29 11:49:22 +00:00
for ( unsigned int i = 0 ; i < findPSVisitor . mFoundNodes . size ( ) ; + + i )
{
osgParticle : : ParticleSystem * ps = static_cast < osgParticle : : ParticleSystem * > ( findPSVisitor . mFoundNodes [ i ] ) ;
2017-10-14 16:45:29 +00:00
2020-12-29 11:49:22 +00:00
osg : : ref_ptr < osgParticle : : ModularProgram > program ( new osgParticle : : ModularProgram ) ;
if ( ! mIsStorm )
2017-10-14 16:45:29 +00:00
program - > addOperator ( new WrapAroundOperator ( mCamera , osg : : Vec3 ( 1024 , 1024 , 800 ) ) ) ;
2021-03-10 15:10:17 +00:00
program - > addOperator ( new WeatherAlphaOperator ( mPrecipitationAlpha , false ) ) ;
2020-12-29 11:49:22 +00:00
program - > setParticleSystem ( ps ) ;
mParticleNode - > addChild ( program ) ;
2017-10-12 10:56:03 +00:00
}
2014-06-24 13:00:15 +00:00
}
}
2012-02-22 18:17:37 +00:00
if ( mClouds ! = weather . mCloudTexture )
{
mClouds = weather . mCloudTexture ;
2015-04-15 16:50:50 +00:00
std : : string texture = Misc : : ResourceHelpers : : correctTexturePath ( mClouds , mSceneManager - > getVFS ( ) ) ;
2016-02-05 22:10:27 +00:00
osg : : ref_ptr < osg : : Texture2D > cloudTex ( new osg : : Texture2D ( mSceneManager - > getImageManager ( ) - > getImage ( texture ) ) ) ;
2016-02-05 21:58:02 +00:00
cloudTex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : REPEAT ) ;
cloudTex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : REPEAT ) ;
mCloudUpdater - > setTexture ( cloudTex ) ;
2012-02-22 18:17:37 +00:00
}
2012-02-29 09:13:25 +00:00
2012-02-22 18:17:37 +00:00
if ( mNextClouds ! = weather . mNextCloudTexture )
{
mNextClouds = weather . mNextCloudTexture ;
2012-02-29 09:13:25 +00:00
2015-11-28 05:00:18 +00:00
if ( ! mNextClouds . empty ( ) )
{
std : : string texture = Misc : : ResourceHelpers : : correctTexturePath ( mNextClouds , mSceneManager - > getVFS ( ) ) ;
2015-06-18 23:03:12 +00:00
2016-02-05 22:10:27 +00:00
osg : : ref_ptr < osg : : Texture2D > cloudTex ( new osg : : Texture2D ( mSceneManager - > getImageManager ( ) - > getImage ( texture ) ) ) ;
2016-02-05 21:58:02 +00:00
cloudTex - > setWrap ( osg : : Texture : : WRAP_S , osg : : Texture : : REPEAT ) ;
cloudTex - > setWrap ( osg : : Texture : : WRAP_T , osg : : Texture : : REPEAT ) ;
mCloudUpdater2 - > setTexture ( cloudTex ) ;
2015-11-28 05:00:18 +00:00
}
2012-02-22 18:17:37 +00:00
}
2012-02-29 09:13:25 +00:00
2015-08-07 13:34:01 +00:00
if ( mCloudBlendFactor ! = weather . mCloudBlendFactor )
2012-02-22 18:17:37 +00:00
{
2015-06-18 23:03:12 +00:00
mCloudBlendFactor = weather . mCloudBlendFactor ;
2015-04-15 16:50:50 +00:00
2015-08-07 13:34:01 +00:00
mCloudUpdater - > setOpacity ( ( 1.f - mCloudBlendFactor ) ) ;
mCloudUpdater2 - > setOpacity ( mCloudBlendFactor ) ;
2020-04-20 16:47:14 +00:00
mCloudMesh2 - > setNodeMask ( mCloudBlendFactor > 0.f ? ~ 0 : 0 ) ;
2012-02-22 18:17:37 +00:00
}
2012-02-29 09:13:25 +00:00
2015-11-09 01:22:40 +00:00
if ( mCloudColour ! = weather . mFogColor )
2012-02-23 20:12:06 +00:00
{
2015-11-09 01:22:40 +00:00
osg : : Vec4f clr ( weather . mFogColor ) ;
clr + = osg : : Vec4f ( 0.13f , 0.13f , 0.13f , 0.f ) ;
2015-04-15 16:50:50 +00:00
mCloudUpdater - > setEmissionColor ( clr ) ;
2015-06-18 23:03:12 +00:00
mCloudUpdater2 - > setEmissionColor ( clr ) ;
2012-07-11 07:08:55 +00:00
2015-11-09 01:22:40 +00:00
mCloudColour = weather . mFogColor ;
2012-02-23 18:49:56 +00:00
}
2012-02-29 09:13:25 +00:00
2012-02-23 18:49:56 +00:00
if ( mSkyColour ! = weather . mSkyColor )
2012-02-29 09:13:25 +00:00
{
2012-02-23 18:49:56 +00:00
mSkyColour = weather . mSkyColor ;
2015-04-14 13:55:56 +00:00
mAtmosphereUpdater - > setEmissionColor ( mSkyColour ) ;
2015-06-19 00:51:01 +00:00
mMasser - > setAtmosphereColor ( mSkyColour ) ;
mSecunda - > setAtmosphereColor ( mSkyColour ) ;
2012-02-23 18:49:56 +00:00
}
2012-02-29 09:13:25 +00:00
2013-02-25 17:29:11 +00:00
if ( mFogColour ! = weather . mFogColor )
{
mFogColour = weather . mFogColor ;
}
2012-07-11 07:08:55 +00:00
mCloudSpeed = weather . mCloudSpeed ;
2012-02-29 09:13:25 +00:00
2015-08-07 05:08:18 +00:00
mMasser - > adjustTransparency ( weather . mGlareView ) ;
mSecunda - > adjustTransparency ( weather . mGlareView ) ;
2015-09-21 15:18:27 +00:00
2015-09-21 17:43:48 +00:00
mSun - > setColor ( weather . mSunDiscColor ) ;
mSun - > adjustTransparency ( weather . mGlareView * weather . mSunDiscColor . a ( ) ) ;
2015-08-05 02:07:42 +00:00
2015-08-07 05:08:18 +00:00
float nextStarsOpacity = weather . mNightFade * weather . mGlareView ;
2017-10-16 17:47:08 +00:00
if ( weather . mNight & & mStarsOpacity ! = nextStarsOpacity )
2012-02-24 22:26:38 +00:00
{
2015-08-05 02:07:42 +00:00
mStarsOpacity = nextStarsOpacity ;
2012-07-11 07:08:55 +00:00
2015-06-19 18:55:04 +00:00
mAtmosphereNightUpdater - > setFade ( mStarsOpacity ) ;
2012-02-24 22:26:38 +00:00
}
2012-02-26 19:46:09 +00:00
2020-04-20 16:47:14 +00:00
mAtmosphereNightNode - > setNodeMask ( weather . mNight ? ~ 0 : 0 ) ;
2012-02-26 19:46:09 +00:00
2021-03-10 15:10:17 +00:00
mPrecipitationAlpha = weather . mPrecipitationAlpha ;
2012-02-22 18:17:37 +00:00
}
2012-02-24 15:12:43 +00:00
2021-01-13 09:41:02 +00:00
float SkyManager : : getBaseWindSpeed ( ) const
{
if ( ! mCreated ) return 0.f ;
return mBaseWindSpeed ;
}
2012-02-24 17:30:16 +00:00
void SkyManager : : sunEnable ( )
{
2015-04-15 16:50:50 +00:00
if ( ! mCreated ) return ;
mSun - > setVisible ( true ) ;
2012-02-24 17:30:16 +00:00
}
void SkyManager : : sunDisable ( )
{
2015-04-15 16:50:50 +00:00
if ( ! mCreated ) return ;
mSun - > setVisible ( false ) ;
2012-02-24 17:30:16 +00:00
}
2015-05-21 23:54:06 +00:00
void SkyManager : : setStormDirection ( const osg : : Vec3f & direction )
2014-06-26 17:01:49 +00:00
{
mStormDirection = direction ;
}
2015-04-14 13:55:56 +00:00
void SkyManager : : setSunDirection ( const osg : : Vec3f & direction )
2012-02-24 17:30:16 +00:00
{
2012-04-01 13:07:41 +00:00
if ( ! mCreated ) return ;
2012-07-19 20:23:07 +00:00
2015-04-15 16:50:50 +00:00
mSun - > setDirection ( direction ) ;
2012-02-24 17:30:16 +00:00
}
2012-02-25 12:46:17 +00:00
2015-07-30 04:57:45 +00:00
void SkyManager : : setMasserState ( const MoonState & state )
2012-02-25 15:36:45 +00:00
{
2015-07-30 04:57:45 +00:00
if ( ! mCreated ) return ;
2015-04-15 16:50:50 +00:00
2015-07-30 04:57:45 +00:00
mMasser - > setState ( state ) ;
2012-02-25 15:36:45 +00:00
}
2015-07-30 04:57:45 +00:00
void SkyManager : : setSecundaState ( const MoonState & state )
2012-02-25 15:36:45 +00:00
{
2015-07-30 04:57:45 +00:00
if ( ! mCreated ) return ;
2015-04-15 16:50:50 +00:00
2015-07-30 04:57:45 +00:00
mSecunda - > setState ( state ) ;
2012-02-25 15:36:45 +00:00
}
2012-02-25 20:06:03 +00:00
void SkyManager : : setDate ( int day , int month )
{
mDay = day ;
mMonth = month ;
}
2012-03-24 16:59:26 +00:00
2015-09-21 14:03:30 +00:00
void SkyManager : : setGlareTimeOfDayFade ( float val )
2012-07-19 20:23:07 +00:00
{
2015-09-21 14:03:30 +00:00
mSun - > setGlareTimeOfDayFade ( val ) ;
2012-07-19 20:23:07 +00:00
}
2014-10-01 16:55:35 +00:00
2015-11-03 22:15:43 +00:00
void SkyManager : : setWaterHeight ( float height )
{
mUnderwaterSwitch - > setWaterLevel ( height ) ;
}
2016-02-09 00:17:02 +00:00
void SkyManager : : listAssetsToPreload ( std : : vector < std : : string > & models , std : : vector < std : : string > & textures )
{
2021-04-08 16:32:38 +00:00
models . emplace_back ( Settings : : Manager : : getString ( " skyatmosphere " , " Models " ) ) ;
if ( mSceneManager - > getVFS ( ) - > exists ( Settings : : Manager : : getString ( " skynight02 " , " Models " ) ) )
models . emplace_back ( Settings : : Manager : : getString ( " skynight02 " , " Models " ) ) ;
models . emplace_back ( Settings : : Manager : : getString ( " skynight01 " , " Models " ) ) ;
models . emplace_back ( Settings : : Manager : : getString ( " skyclouds " , " Models " ) ) ;
models . emplace_back ( Settings : : Manager : : getString ( " weatherashcloud " , " Models " ) ) ;
models . emplace_back ( Settings : : Manager : : getString ( " weatherblightcloud " , " Models " ) ) ;
models . emplace_back ( Settings : : Manager : : getString ( " weathersnow " , " Models " ) ) ;
models . emplace_back ( Settings : : Manager : : getString ( " weatherblizzard " , " Models " ) ) ;
2020-10-17 08:26:35 +00:00
textures . emplace_back ( " textures/tx_mooncircle_full_s.dds " ) ;
textures . emplace_back ( " textures/tx_mooncircle_full_m.dds " ) ;
textures . emplace_back ( " textures/tx_masser_new.dds " ) ;
textures . emplace_back ( " textures/tx_masser_one_wax.dds " ) ;
textures . emplace_back ( " textures/tx_masser_half_wax.dds " ) ;
textures . emplace_back ( " textures/tx_masser_three_wax.dds " ) ;
textures . emplace_back ( " textures/tx_masser_one_wan.dds " ) ;
textures . emplace_back ( " textures/tx_masser_half_wan.dds " ) ;
textures . emplace_back ( " textures/tx_masser_three_wan.dds " ) ;
textures . emplace_back ( " textures/tx_masser_full.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_new.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_one_wax.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_half_wax.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_three_wax.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_one_wan.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_half_wan.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_three_wan.dds " ) ;
textures . emplace_back ( " textures/tx_secunda_full.dds " ) ;
textures . emplace_back ( " textures/tx_sun_05.dds " ) ;
textures . emplace_back ( " textures/tx_sun_flash_grey_05.dds " ) ;
textures . emplace_back ( " textures/tx_raindrop_01.dds " ) ;
2016-02-09 00:17:02 +00:00
}
2015-11-03 22:15:43 +00:00
void SkyManager : : setWaterEnabled ( bool enabled )
{
mUnderwaterSwitch - > setEnabled ( enabled ) ;
}
2014-10-01 16:55:35 +00:00
}