2016-02-16 18:18:48 +01:00
# include "shadervisitor.hpp"
2021-09-29 13:40:37 +00:00
# include <unordered_set>
2021-11-20 18:39:20 -08:00
# include <unordered_map>
2021-09-29 13:40:37 +00:00
# include <set>
2020-12-16 23:44:15 +00:00
# include <osg/AlphaFunc>
2022-06-21 22:28:17 +02:00
# include <osg/BlendFunc>
2016-02-16 18:18:48 +01:00
# include <osg/Geometry>
2021-01-07 18:13:51 +00:00
# include <osg/GLExtensions>
2020-04-10 15:45:37 +01:00
# include <osg/Material>
2020-12-26 22:45:53 +00:00
# include <osg/Multisample>
2020-04-10 15:45:37 +01:00
# include <osg/Texture>
2021-03-24 16:32:15 +04:00
# include <osg/ValueObject>
2022-05-13 18:58:00 -07:00
# include <osg/Capability>
2016-02-16 18:18:48 +01:00
2021-10-20 09:42:18 -07:00
# include <osgParticle/ParticleSystem>
2016-02-16 18:18:48 +01:00
# include <osgUtil/TangentSpaceGenerator>
2018-08-14 19:42:41 +04:00
# include <components/debug/debuglog.hpp>
2020-01-09 15:57:05 +04:00
# include <components/misc/stringops.hpp>
2022-04-04 22:51:23 +02:00
# include <components/stereo/stereomanager.hpp>
2016-02-20 17:57:19 +01:00
# include <components/resource/imagemanager.hpp>
# include <components/vfs/manager.hpp>
2016-03-23 16:48:41 +01:00
# include <components/sceneutil/riggeometry.hpp>
2017-09-01 22:56:09 +02:00
# include <components/sceneutil/morphgeometry.hpp>
2021-11-21 02:25:05 +00:00
# include <components/sceneutil/depth.hpp>
2022-06-21 15:27:34 +00:00
# include <components/sceneutil/riggeometryosgaextension.hpp>
2016-02-16 18:18:48 +01:00
2020-12-16 23:44:15 +00:00
# include "removedalphafunc.hpp"
2016-02-16 18:18:48 +01:00
# include "shadermanager.hpp"
2022-05-13 18:58:00 -07:00
namespace
{
class OpaqueDepthAttribute : public osg : : StateAttribute
{
public :
OpaqueDepthAttribute ( ) = default ;
OpaqueDepthAttribute ( const OpaqueDepthAttribute & copy , const osg : : CopyOp & copyop = osg : : CopyOp : : SHALLOW_COPY )
: osg : : StateAttribute ( copy , copyop ) , mTextures ( copy . mTextures ) , mUnit ( copy . mUnit ) { }
2022-06-21 15:55:06 +00:00
void setTexturesAndUnit ( const std : : array < osg : : ref_ptr < osg : : Texture > , 2 > & textures , int unit )
2022-05-13 18:58:00 -07:00
{
mTextures = textures ;
mUnit = unit ;
}
META_StateAttribute ( Shader , OpaqueDepthAttribute , osg : : StateAttribute : : TEXTURE )
int compare ( const StateAttribute & sa ) const override
{
COMPARE_StateAttribute_Types ( OpaqueDepthAttribute , sa ) ;
COMPARE_StateAttribute_Parameter ( mTextures ) ;
return 0 ;
}
void apply ( osg : : State & state ) const override
{
auto index = state . getFrameStamp ( ) - > getFrameNumber ( ) % 2 ;
if ( ! mTextures [ index ] )
return ;
state . setActiveTextureUnit ( mUnit ) ;
state . applyTextureAttribute ( mUnit , mTextures [ index ] ) ;
}
private :
2022-06-21 15:55:06 +00:00
mutable std : : array < osg : : ref_ptr < osg : : Texture > , 2 > mTextures ;
2022-05-13 18:58:00 -07:00
int mUnit ;
} ;
}
2016-02-16 18:18:48 +01:00
namespace Shader
{
2021-12-08 20:44:10 +00:00
/**
* Miniature version of osg : : StateSet used to track state added by the shader visitor which should be ignored when
* it ' s applied a second time , and removed when shaders are removed .
* Actual StateAttributes aren ' t kept as they ' re recoverable from the StateSet this is attached to - we just want
* the TypeMemberPair as that uniquely identifies which of those StateAttributes it was we ' re tracking .
* Not all StateSet features have been added yet - we implement an equivalently - named method to each of the StateSet
* methods called in createProgram , and implement new ones as they ' re needed .
* When expanding tracking to cover new things , ensure they ' re accounted for in ensureFFP .
*/
2021-07-03 21:04:17 +01:00
class AddedState : public osg : : Object
{
public :
AddedState ( ) = default ;
AddedState ( const AddedState & rhs , const osg : : CopyOp & copyOp )
: osg : : Object ( rhs , copyOp )
, mUniforms ( rhs . mUniforms )
, mModes ( rhs . mModes )
, mAttributes ( rhs . mAttributes )
2021-11-20 18:39:20 -08:00
, mTextureModes ( rhs . mTextureModes )
2021-07-03 21:04:17 +01:00
{
}
void addUniform ( const std : : string & name ) { mUniforms . emplace ( name ) ; }
void setMode ( osg : : StateAttribute : : GLMode mode ) { mModes . emplace ( mode ) ; }
void setAttribute ( osg : : StateAttribute : : TypeMemberPair typeMemberPair ) { mAttributes . emplace ( typeMemberPair ) ; }
void setAttribute ( const osg : : StateAttribute * attribute )
{
mAttributes . emplace ( attribute - > getTypeMemberPair ( ) ) ;
}
template < typename T >
void setAttribute ( osg : : ref_ptr < T > attribute ) { setAttribute ( attribute . get ( ) ) ; }
void setAttributeAndModes ( const osg : : StateAttribute * attribute )
{
setAttribute ( attribute ) ;
InterrogateModesHelper helper ( this ) ;
attribute - > getModeUsage ( helper ) ;
}
template < typename T >
void setAttributeAndModes ( osg : : ref_ptr < T > attribute ) { setAttributeAndModes ( attribute . get ( ) ) ; }
2021-12-08 20:44:10 +00:00
void setTextureMode ( unsigned int unit , osg : : StateAttribute : : GLMode mode ) { mTextureModes [ unit ] . emplace ( mode ) ; }
void setTextureAttribute ( int unit , osg : : StateAttribute : : TypeMemberPair typeMemberPair ) { mTextureAttributes [ unit ] . emplace ( typeMemberPair ) ; }
void setTextureAttribute ( unsigned int unit , const osg : : StateAttribute * attribute )
{
mTextureAttributes [ unit ] . emplace ( attribute - > getTypeMemberPair ( ) ) ;
}
template < typename T >
void setTextureAttribute ( unsigned int unit , osg : : ref_ptr < T > attribute ) { setTextureAttribute ( unit , attribute . get ( ) ) ; }
void setTextureAttributeAndModes ( unsigned int unit , const osg : : StateAttribute * attribute )
{
setTextureAttribute ( unit , attribute ) ;
InterrogateModesHelper helper ( this , unit ) ;
attribute - > getModeUsage ( helper ) ;
}
template < typename T >
void setTextureAttributeAndModes ( unsigned int unit , osg : : ref_ptr < T > attribute ) { setTextureAttributeAndModes ( unit , attribute . get ( ) ) ; }
2021-07-03 21:04:17 +01:00
bool hasUniform ( const std : : string & name ) { return mUniforms . count ( name ) ; }
bool hasMode ( osg : : StateAttribute : : GLMode mode ) { return mModes . count ( mode ) ; }
2021-08-15 19:50:28 +02:00
bool hasAttribute ( const osg : : StateAttribute : : TypeMemberPair & typeMemberPair ) { return mAttributes . count ( typeMemberPair ) ; }
2021-07-03 21:04:17 +01:00
bool hasAttribute ( osg : : StateAttribute : : Type type , unsigned int member ) { return hasAttribute ( osg : : StateAttribute : : TypeMemberPair ( type , member ) ) ; }
2021-11-20 18:39:20 -08:00
bool hasTextureMode ( int unit , osg : : StateAttribute : : GLMode mode )
{
auto it = mTextureModes . find ( unit ) ;
if ( it = = mTextureModes . cend ( ) )
return false ;
return it - > second . count ( mode ) ;
}
2021-07-03 21:04:17 +01:00
const std : : set < osg : : StateAttribute : : TypeMemberPair > & getAttributes ( ) { return mAttributes ; }
2021-12-08 20:44:10 +00:00
const std : : unordered_map < unsigned int , std : : set < osg : : StateAttribute : : TypeMemberPair > > & getTextureAttributes ( ) { return mTextureAttributes ; }
2021-07-03 21:04:17 +01:00
bool empty ( )
{
2021-12-08 20:44:10 +00:00
return mUniforms . empty ( ) & & mModes . empty ( ) & & mAttributes . empty ( ) & & mTextureModes . empty ( ) & & mTextureAttributes . empty ( ) ;
2021-07-03 21:04:17 +01:00
}
META_Object ( Shader , AddedState )
private :
class InterrogateModesHelper : public osg : : StateAttribute : : ModeUsage
{
public :
2021-12-08 20:44:10 +00:00
InterrogateModesHelper ( AddedState * tracker , unsigned int textureUnit = 0 )
: mTracker ( tracker )
, mTextureUnit ( textureUnit )
{ }
2021-07-03 21:04:17 +01:00
void usesMode ( osg : : StateAttribute : : GLMode mode ) override { mTracker - > setMode ( mode ) ; }
2021-12-08 20:44:10 +00:00
void usesTextureMode ( osg : : StateAttribute : : GLMode mode ) override { mTracker - > setTextureMode ( mTextureUnit , mode ) ; }
2021-07-03 21:04:17 +01:00
private :
AddedState * mTracker ;
2021-12-08 20:44:10 +00:00
unsigned int mTextureUnit ;
2021-07-03 21:04:17 +01:00
} ;
2021-11-20 18:39:20 -08:00
using ModeSet = std : : unordered_set < osg : : StateAttribute : : GLMode > ;
2021-12-08 20:44:10 +00:00
using AttributeSet = std : : set < osg : : StateAttribute : : TypeMemberPair > ;
2021-11-20 18:39:20 -08:00
2021-07-03 21:04:17 +01:00
std : : unordered_set < std : : string > mUniforms ;
2021-11-20 18:39:20 -08:00
ModeSet mModes ;
2021-12-08 20:44:10 +00:00
AttributeSet mAttributes ;
std : : unordered_map < unsigned int , ModeSet > mTextureModes ;
std : : unordered_map < unsigned int , AttributeSet > mTextureAttributes ;
2021-07-03 21:04:17 +01:00
} ;
2016-02-16 18:18:48 +01:00
ShaderVisitor : : ShaderRequirements : : ShaderRequirements ( )
2016-02-20 19:02:11 +01:00
: mShaderRequired ( false )
2018-12-03 18:56:37 +00:00
, mColorMode ( 0 )
2016-02-18 22:48:53 +01:00
, mMaterialOverridden ( false )
2020-12-16 23:44:15 +00:00
, mAlphaTestOverridden ( false )
2021-01-02 19:09:06 +00:00
, mAlphaBlendOverridden ( false )
2020-12-16 23:44:15 +00:00
, mAlphaFunc ( GL_ALWAYS )
, mAlphaRef ( 1.0 )
2021-01-02 19:09:06 +00:00
, mAlphaBlend ( false )
2022-06-21 22:28:17 +02:00
, mBlendFuncOverridden ( false )
, mAdditiveBlending ( false )
2016-03-22 21:22:59 +01:00
, mNormalHeight ( false )
2016-02-16 18:18:48 +01:00
, mTexStageRequiringTangents ( - 1 )
2021-11-20 18:39:20 -08:00
, mSoftParticles ( false )
, mSoftParticleSize ( 0.f )
2018-10-09 10:21:12 +04:00
, mNode ( nullptr )
2016-02-16 18:18:48 +01:00
{
}
2020-12-17 00:46:09 +03:00
ShaderVisitor : : ShaderVisitor ( ShaderManager & shaderManager , Resource : : ImageManager & imageManager , const std : : string & defaultShaderPrefix )
2016-02-16 18:18:48 +01:00
: osg : : NodeVisitor ( TRAVERSE_ALL_CHILDREN )
2016-02-18 17:08:18 +01:00
, mForceShaders ( false )
2016-02-18 22:48:53 +01:00
, mAllowedToModifyStateSets ( true )
2016-02-20 17:57:19 +01:00
, mAutoUseNormalMaps ( false )
2016-02-20 19:02:11 +01:00
, mAutoUseSpecularMaps ( false )
2020-11-09 13:53:58 +03:00
, mApplyLightingToEnvMaps ( false )
2021-04-09 23:08:51 +00:00
, mConvertAlphaTestToAlphaToCoverage ( false )
2022-05-13 18:58:00 -07:00
, mSupportsNormalsRT ( false )
2016-02-16 18:18:48 +01:00
, mShaderManager ( shaderManager )
2016-02-20 17:57:19 +01:00
, mImageManager ( imageManager )
2020-12-17 00:46:09 +03:00
, mDefaultShaderPrefix ( defaultShaderPrefix )
2016-02-16 18:18:48 +01:00
{
}
2016-02-18 17:08:18 +01:00
void ShaderVisitor : : setForceShaders ( bool force )
{
mForceShaders = force ;
}
2016-02-16 18:18:48 +01:00
void ShaderVisitor : : apply ( osg : : Node & node )
{
if ( node . getStateSet ( ) )
{
2017-08-31 21:39:21 +00:00
pushRequirements ( node ) ;
2016-02-18 22:48:53 +01:00
applyStateSet ( node . getStateSet ( ) , node ) ;
2016-02-16 18:18:48 +01:00
traverse ( node ) ;
popRequirements ( ) ;
}
else
traverse ( node ) ;
}
2016-02-18 22:48:53 +01:00
osg : : StateSet * getWritableStateSet ( osg : : Node & node )
2016-02-16 18:18:48 +01:00
{
2016-02-18 22:48:53 +01:00
if ( ! node . getStateSet ( ) )
return node . getOrCreateStateSet ( ) ;
2017-02-09 04:50:51 +01:00
osg : : ref_ptr < osg : : StateSet > newStateSet = new osg : : StateSet ( * node . getStateSet ( ) , osg : : CopyOp : : SHALLOW_COPY ) ;
2016-02-18 22:48:53 +01:00
node . setStateSet ( newStateSet ) ;
return newStateSet . get ( ) ;
}
2020-12-16 23:44:15 +00:00
osg : : UserDataContainer * getWritableUserDataContainer ( osg : : Object & object )
{
if ( ! object . getUserDataContainer ( ) )
return object . getOrCreateUserDataContainer ( ) ;
osg : : ref_ptr < osg : : UserDataContainer > newUserData = static_cast < osg : : UserDataContainer * > ( object . getUserDataContainer ( ) - > clone ( osg : : CopyOp : : SHALLOW_COPY ) ) ;
object . setUserDataContainer ( newUserData ) ;
return newUserData . get ( ) ;
}
osg : : StateSet * getRemovedState ( osg : : StateSet & stateSet )
{
if ( ! stateSet . getUserDataContainer ( ) )
return nullptr ;
return static_cast < osg : : StateSet * > ( stateSet . getUserDataContainer ( ) - > getUserObject ( " removedState " ) ) ;
}
2021-07-03 21:04:17 +01:00
void updateRemovedState ( osg : : UserDataContainer & userData , osg : : StateSet * removedState )
2020-12-16 23:44:15 +00:00
{
unsigned int index = userData . getUserObjectIndex ( " removedState " ) ;
if ( index < userData . getNumUserObjects ( ) )
2021-07-03 21:04:17 +01:00
userData . setUserObject ( index , removedState ) ;
2020-12-16 23:44:15 +00:00
else
2021-07-03 21:04:17 +01:00
userData . addUserObject ( removedState ) ;
removedState - > setName ( " removedState " ) ;
}
AddedState * getAddedState ( osg : : StateSet & stateSet )
{
if ( ! stateSet . getUserDataContainer ( ) )
return nullptr ;
return static_cast < AddedState * > ( stateSet . getUserDataContainer ( ) - > getUserObject ( " addedState " ) ) ;
}
void updateAddedState ( osg : : UserDataContainer & userData , AddedState * addedState )
{
unsigned int index = userData . getUserObjectIndex ( " addedState " ) ;
if ( index < userData . getNumUserObjects ( ) )
userData . setUserObject ( index , addedState ) ;
else
userData . addUserObject ( addedState ) ;
addedState - > setName ( " addedState " ) ;
2020-12-16 23:44:15 +00:00
}
2022-01-08 05:53:10 +03:00
const char * defaultTextures [ ] = { " diffuseMap " , " normalMap " , " emissiveMap " , " darkMap " , " detailMap " , " envMap " , " specularMap " , " decalMap " , " bumpMap " , " glossMap " } ;
2022-05-21 22:08:20 +02:00
bool isTextureNameRecognized ( std : : string_view name )
2016-02-20 23:43:05 +01:00
{
2022-05-21 22:08:20 +02:00
return std : : find ( std : : begin ( defaultTextures ) , std : : end ( defaultTextures ) , name ) ! = std : : end ( defaultTextures ) ;
2016-02-20 23:43:05 +01:00
}
2016-02-18 22:48:53 +01:00
void ShaderVisitor : : applyStateSet ( osg : : ref_ptr < osg : : StateSet > stateset , osg : : Node & node )
{
2018-10-09 10:21:12 +04:00
osg : : StateSet * writableStateSet = nullptr ;
2016-02-18 22:48:53 +01:00
if ( mAllowedToModifyStateSets )
writableStateSet = node . getStateSet ( ) ;
2016-02-16 18:18:48 +01:00
const osg : : StateSet : : TextureAttributeList & texAttributes = stateset - > getTextureAttributeList ( ) ;
2020-12-17 00:46:09 +03:00
bool shaderRequired = false ;
if ( node . getUserValue ( " shaderRequired " , shaderRequired ) & & shaderRequired )
mRequirements . back ( ) . mShaderRequired = true ;
2021-12-08 20:44:10 +00:00
// Make sure to disregard any state that came from a previous call to createProgram
2021-11-20 18:39:20 -08:00
osg : : ref_ptr < AddedState > addedState = getAddedState ( * stateset ) ;
2016-02-22 19:06:12 +01:00
if ( ! texAttributes . empty ( ) )
2016-02-16 18:18:48 +01:00
{
2018-10-09 10:21:12 +04:00
const osg : : Texture * diffuseMap = nullptr ;
const osg : : Texture * normalMap = nullptr ;
const osg : : Texture * specularMap = nullptr ;
2020-03-05 13:15:18 +03:00
const osg : : Texture * bumpMap = nullptr ;
2016-02-20 17:57:19 +01:00
for ( unsigned int unit = 0 ; unit < texAttributes . size ( ) ; + + unit )
2016-02-16 18:18:48 +01:00
{
2016-02-20 17:57:19 +01:00
const osg : : StateAttribute * attr = stateset - > getTextureAttribute ( unit , osg : : StateAttribute : : TEXTURE ) ;
if ( attr )
2016-02-16 18:18:48 +01:00
{
2021-12-08 20:44:10 +00:00
// If textures ever get removed in createProgram, expand this to check we're operating on main texture attribute list
// rather than the removed list
2021-11-20 18:39:20 -08:00
if ( addedState & & addedState - > hasTextureMode ( unit , GL_TEXTURE_2D ) )
continue ;
2016-02-20 17:57:19 +01:00
const osg : : Texture * texture = attr - > asTexture ( ) ;
if ( texture )
2016-02-16 18:18:48 +01:00
{
2016-02-20 23:43:05 +01:00
std : : string texName = texture - > getName ( ) ;
if ( ( texName . empty ( ) | | ! isTextureNameRecognized ( texName ) ) & & unit = = 0 )
texName = " diffuseMap " ;
2016-03-22 21:22:59 +01:00
if ( texName = = " normalHeightMap " )
{
mRequirements . back ( ) . mNormalHeight = true ;
texName = " normalMap " ;
}
2016-02-20 23:43:05 +01:00
if ( ! texName . empty ( ) )
2016-02-16 18:18:48 +01:00
{
2016-02-20 23:43:05 +01:00
mRequirements . back ( ) . mTextures [ unit ] = texName ;
if ( texName = = " normalMap " )
2016-02-20 17:57:19 +01:00
{
mRequirements . back ( ) . mTexStageRequiringTangents = unit ;
2016-02-20 19:02:11 +01:00
mRequirements . back ( ) . mShaderRequired = true ;
2016-02-20 17:57:19 +01:00
if ( ! writableStateSet )
writableStateSet = getWritableStateSet ( node ) ;
// normal maps are by default off since the FFP can't render them, now that we'll use shaders switch to On
writableStateSet - > setTextureMode ( unit , GL_TEXTURE_2D , osg : : StateAttribute : : ON ) ;
normalMap = texture ;
}
2016-03-22 21:22:59 +01:00
else if ( texName = = " diffuseMap " )
2016-02-20 17:57:19 +01:00
diffuseMap = texture ;
2016-03-22 21:22:59 +01:00
else if ( texName = = " specularMap " )
2016-02-20 19:02:11 +01:00
specularMap = texture ;
2020-03-02 04:03:36 +03:00
else if ( texName = = " bumpMap " )
{
2020-03-05 13:15:18 +03:00
bumpMap = texture ;
2020-03-02 04:03:36 +03:00
mRequirements . back ( ) . mShaderRequired = true ;
if ( ! writableStateSet )
writableStateSet = getWritableStateSet ( node ) ;
// Bump maps are off by default as well
writableStateSet - > setTextureMode ( unit , GL_TEXTURE_2D , osg : : StateAttribute : : ON ) ;
}
2020-11-09 13:53:58 +03:00
else if ( texName = = " envMap " & & mApplyLightingToEnvMaps )
2020-03-04 14:06:22 +03:00
{
2020-11-09 13:53:58 +03:00
mRequirements . back ( ) . mShaderRequired = true ;
2020-03-04 14:06:22 +03:00
}
2022-01-08 05:53:10 +03:00
else if ( texName = = " glossMap " )
{
mRequirements . back ( ) . mShaderRequired = true ;
if ( ! writableStateSet )
writableStateSet = getWritableStateSet ( node ) ;
// As well as gloss maps
writableStateSet - > setTextureMode ( unit , GL_TEXTURE_2D , osg : : StateAttribute : : ON ) ;
}
2016-02-16 18:18:48 +01:00
}
2021-10-05 12:37:08 +00:00
else
2018-08-14 19:42:41 +04:00
Log ( Debug : : Error ) < < " ShaderVisitor encountered unknown texture " < < texture ;
2016-02-16 18:18:48 +01:00
}
2016-02-20 17:57:19 +01:00
}
2016-02-16 18:18:48 +01:00
}
2016-02-20 17:57:19 +01:00
2018-10-09 10:21:12 +04:00
if ( mAutoUseNormalMaps & & diffuseMap ! = nullptr & & normalMap = = nullptr & & diffuseMap - > getImage ( 0 ) )
2016-02-18 22:48:53 +01:00
{
2016-08-29 12:20:00 +02:00
std : : string normalMapFileName = diffuseMap - > getImage ( 0 ) - > getFileName ( ) ;
2016-03-22 21:22:59 +01:00
osg : : ref_ptr < osg : : Image > image ;
bool normalHeight = false ;
2016-08-29 12:20:00 +02:00
std : : string normalHeightMap = normalMapFileName ;
2020-01-09 15:57:05 +04:00
Misc : : StringUtils : : replaceLast ( normalHeightMap , " . " , mNormalHeightMapPattern + " . " ) ;
2016-03-22 21:22:59 +01:00
if ( mImageManager . getVFS ( ) - > exists ( normalHeightMap ) )
{
image = mImageManager . getImage ( normalHeightMap ) ;
normalHeight = true ;
}
else
{
2020-01-09 15:57:05 +04:00
Misc : : StringUtils : : replaceLast ( normalMapFileName , " . " , mNormalMapPattern + " . " ) ;
2016-08-29 12:20:00 +02:00
if ( mImageManager . getVFS ( ) - > exists ( normalMapFileName ) )
2016-03-22 21:22:59 +01:00
{
2016-08-29 12:20:00 +02:00
image = mImageManager . getImage ( normalMapFileName ) ;
2016-03-22 21:22:59 +01:00
}
}
2020-03-05 13:15:18 +03:00
// Avoid using the auto-detected normal map if it's already being used as a bump map.
// It's probably not an actual normal map.
bool hasNamesakeBumpMap = image & & bumpMap & & bumpMap - > getImage ( 0 ) & & image - > getFileName ( ) = = bumpMap - > getImage ( 0 ) - > getFileName ( ) ;
2016-03-22 21:22:59 +01:00
2020-03-05 13:15:18 +03:00
if ( ! hasNamesakeBumpMap & & image )
2016-02-20 17:57:19 +01:00
{
2016-03-22 21:22:59 +01:00
osg : : ref_ptr < osg : : Texture2D > normalMapTex ( new osg : : Texture2D ( image ) ) ;
2019-05-29 13:37:00 +00:00
normalMapTex - > setTextureSize ( image - > s ( ) , image - > t ( ) ) ;
2016-02-20 17:57:19 +01:00
normalMapTex - > setWrap ( osg : : Texture : : WRAP_S , diffuseMap - > getWrap ( osg : : Texture : : WRAP_S ) ) ;
normalMapTex - > setWrap ( osg : : Texture : : WRAP_T , diffuseMap - > getWrap ( osg : : Texture : : WRAP_T ) ) ;
normalMapTex - > setFilter ( osg : : Texture : : MIN_FILTER , diffuseMap - > getFilter ( osg : : Texture : : MIN_FILTER ) ) ;
normalMapTex - > setFilter ( osg : : Texture : : MAG_FILTER , diffuseMap - > getFilter ( osg : : Texture : : MAG_FILTER ) ) ;
normalMapTex - > setMaxAnisotropy ( diffuseMap - > getMaxAnisotropy ( ) ) ;
normalMapTex - > setName ( " normalMap " ) ;
int unit = texAttributes . size ( ) ;
if ( ! writableStateSet )
writableStateSet = getWritableStateSet ( node ) ;
writableStateSet - > setTextureAttributeAndModes ( unit , normalMapTex , osg : : StateAttribute : : ON ) ;
mRequirements . back ( ) . mTextures [ unit ] = " normalMap " ;
mRequirements . back ( ) . mTexStageRequiringTangents = unit ;
2016-02-20 19:02:11 +01:00
mRequirements . back ( ) . mShaderRequired = true ;
2016-03-22 21:22:59 +01:00
mRequirements . back ( ) . mNormalHeight = normalHeight ;
2016-02-20 19:02:11 +01:00
}
}
2018-10-09 10:21:12 +04:00
if ( mAutoUseSpecularMaps & & diffuseMap ! = nullptr & & specularMap = = nullptr & & diffuseMap - > getImage ( 0 ) )
2016-02-20 19:02:11 +01:00
{
2016-08-29 12:20:00 +02:00
std : : string specularMapFileName = diffuseMap - > getImage ( 0 ) - > getFileName ( ) ;
2020-01-09 15:57:05 +04:00
Misc : : StringUtils : : replaceLast ( specularMapFileName , " . " , mSpecularMapPattern + " . " ) ;
2016-08-29 12:20:00 +02:00
if ( mImageManager . getVFS ( ) - > exists ( specularMapFileName ) )
2016-02-20 19:02:11 +01:00
{
2019-05-29 13:37:00 +00:00
osg : : ref_ptr < osg : : Image > image ( mImageManager . getImage ( specularMapFileName ) ) ;
osg : : ref_ptr < osg : : Texture2D > specularMapTex ( new osg : : Texture2D ( image ) ) ;
specularMapTex - > setTextureSize ( image - > s ( ) , image - > t ( ) ) ;
2016-02-20 19:02:11 +01:00
specularMapTex - > setWrap ( osg : : Texture : : WRAP_S , diffuseMap - > getWrap ( osg : : Texture : : WRAP_S ) ) ;
specularMapTex - > setWrap ( osg : : Texture : : WRAP_T , diffuseMap - > getWrap ( osg : : Texture : : WRAP_T ) ) ;
specularMapTex - > setFilter ( osg : : Texture : : MIN_FILTER , diffuseMap - > getFilter ( osg : : Texture : : MIN_FILTER ) ) ;
specularMapTex - > setFilter ( osg : : Texture : : MAG_FILTER , diffuseMap - > getFilter ( osg : : Texture : : MAG_FILTER ) ) ;
specularMapTex - > setMaxAnisotropy ( diffuseMap - > getMaxAnisotropy ( ) ) ;
specularMapTex - > setName ( " specularMap " ) ;
int unit = texAttributes . size ( ) ;
if ( ! writableStateSet )
writableStateSet = getWritableStateSet ( node ) ;
writableStateSet - > setTextureAttributeAndModes ( unit , specularMapTex , osg : : StateAttribute : : ON ) ;
mRequirements . back ( ) . mTextures [ unit ] = " specularMap " ;
mRequirements . back ( ) . mShaderRequired = true ;
2016-02-20 17:57:19 +01:00
}
2018-10-16 21:23:31 +01:00
}
2016-02-16 18:18:48 +01:00
}
const osg : : StateSet : : AttributeList & attributes = stateset - > getAttributeList ( ) ;
2020-12-16 23:44:15 +00:00
osg : : StateSet : : AttributeList removedAttributes ;
2021-04-14 08:07:56 +04:00
if ( osg : : ref_ptr < osg : : StateSet > removedState = getRemovedState ( * stateset ) )
2020-12-16 23:44:15 +00:00
removedAttributes = removedState - > getAttributeList ( ) ;
2021-07-03 21:04:17 +01:00
2021-06-27 01:48:06 +01:00
for ( const auto * attributeMap : std : : initializer_list < const osg : : StateSet : : AttributeList * > { & attributes , & removedAttributes } )
2016-02-16 18:18:48 +01:00
{
2021-06-27 01:48:06 +01:00
for ( osg : : StateSet : : AttributeList : : const_iterator it = attributeMap - > begin ( ) ; it ! = attributeMap - > end ( ) ; + + it )
2016-02-16 18:18:48 +01:00
{
2021-07-03 21:04:17 +01:00
if ( addedState & & attributeMap ! = & removedAttributes & & addedState - > hasAttribute ( it - > first ) )
2021-06-27 00:12:07 +01:00
continue ;
2020-12-16 23:44:15 +00:00
if ( it - > first . first = = osg : : StateAttribute : : MATERIAL )
2016-02-18 22:48:53 +01:00
{
2020-12-16 23:44:15 +00:00
// This should probably be moved out of ShaderRequirements and be applied directly now it's a uniform instead of a define
if ( ! mRequirements . back ( ) . mMaterialOverridden | | it - > second . second & osg : : StateAttribute : : PROTECTED )
{
if ( it - > second . second & osg : : StateAttribute : : OVERRIDE )
mRequirements . back ( ) . mMaterialOverridden = true ;
2016-02-18 22:48:53 +01:00
2020-12-16 23:44:15 +00:00
const osg : : Material * mat = static_cast < const osg : : Material * > ( it - > second . first . get ( ) ) ;
2018-10-15 23:12:15 +01:00
2020-12-16 23:44:15 +00:00
int colorMode ;
switch ( mat - > getColorMode ( ) )
{
case osg : : Material : : OFF :
colorMode = 0 ;
break ;
case osg : : Material : : EMISSION :
colorMode = 1 ;
break ;
default :
case osg : : Material : : AMBIENT_AND_DIFFUSE :
colorMode = 2 ;
break ;
case osg : : Material : : AMBIENT :
colorMode = 3 ;
break ;
case osg : : Material : : DIFFUSE :
colorMode = 4 ;
break ;
case osg : : Material : : SPECULAR :
colorMode = 5 ;
break ;
}
mRequirements . back ( ) . mColorMode = colorMode ;
2018-10-15 23:12:15 +01:00
}
2020-12-16 23:44:15 +00:00
}
else if ( it - > first . first = = osg : : StateAttribute : : ALPHAFUNC )
{
if ( ! mRequirements . back ( ) . mAlphaTestOverridden | | it - > second . second & osg : : StateAttribute : : PROTECTED )
{
if ( it - > second . second & osg : : StateAttribute : : OVERRIDE )
mRequirements . back ( ) . mAlphaTestOverridden = true ;
2018-10-15 23:12:15 +01:00
2020-12-16 23:44:15 +00:00
const osg : : AlphaFunc * alpha = static_cast < const osg : : AlphaFunc * > ( it - > second . first . get ( ) ) ;
mRequirements . back ( ) . mAlphaFunc = alpha - > getFunction ( ) ;
mRequirements . back ( ) . mAlphaRef = alpha - > getReferenceValue ( ) ;
}
2016-02-18 22:48:53 +01:00
}
2022-06-21 22:28:17 +02:00
else if ( it - > first . first = = osg : : StateAttribute : : BLENDFUNC )
{
if ( ! mRequirements . back ( ) . mBlendFuncOverridden | | it - > second . second & osg : : StateAttribute : : PROTECTED )
{
if ( it - > second . second & osg : : StateAttribute : : OVERRIDE )
mRequirements . back ( ) . mBlendFuncOverridden = true ;
const osg : : BlendFunc * blend = static_cast < const osg : : BlendFunc * > ( it - > second . first . get ( ) ) ;
mRequirements . back ( ) . mAdditiveBlending =
blend - > getSource ( ) = = osg : : BlendFunc : : SRC_ALPHA & & blend - > getDestination ( ) = = osg : : BlendFunc : : ONE ;
}
}
2016-02-16 18:18:48 +01:00
}
2020-04-10 15:45:37 +01:00
}
2021-01-02 19:09:06 +00:00
unsigned int alphaBlend = stateset - > getMode ( GL_BLEND ) ;
if ( alphaBlend ! = osg : : StateAttribute : : INHERIT & & ( ! mRequirements . back ( ) . mAlphaBlendOverridden | | alphaBlend & osg : : StateAttribute : : PROTECTED ) )
{
if ( alphaBlend & osg : : StateAttribute : : OVERRIDE )
mRequirements . back ( ) . mAlphaBlendOverridden = true ;
mRequirements . back ( ) . mAlphaBlend = alphaBlend & osg : : StateAttribute : : ON ;
}
2016-02-16 18:18:48 +01:00
}
2017-08-31 21:39:21 +00:00
void ShaderVisitor : : pushRequirements ( osg : : Node & node )
2016-02-16 18:18:48 +01:00
{
2021-10-01 08:11:00 +00:00
if ( mRequirements . empty ( ) )
mRequirements . emplace_back ( ) ;
else
mRequirements . push_back ( mRequirements . back ( ) ) ;
2017-08-31 21:39:21 +00:00
mRequirements . back ( ) . mNode = & node ;
2016-02-16 18:18:48 +01:00
}
void ShaderVisitor : : popRequirements ( )
{
mRequirements . pop_back ( ) ;
}
2017-08-31 21:39:21 +00:00
void ShaderVisitor : : createProgram ( const ShaderRequirements & reqs )
2016-02-16 18:18:48 +01:00
{
2017-09-01 22:01:50 +02:00
if ( ! reqs . mShaderRequired & & ! mForceShaders )
2020-12-03 21:06:02 +00:00
{
ensureFFP ( * reqs . mNode ) ;
2017-09-01 22:01:50 +02:00
return ;
2020-12-03 21:06:02 +00:00
}
2017-09-01 22:01:50 +02:00
2021-12-08 20:44:10 +00:00
/**
* The shader visitor is supposed to be idempotent and undoable .
* That means we need to back up state we ' ve removed ( so it can be restored and / or considered by further
* applications of the visitor ) and track which state we added ( so it can be removed and / or ignored by further
* applications of the visitor ) .
* Before editing writableStateSet in a way that explicitly removes state or might overwrite existing state , it
* should be copied to removedState , another StateSet , unless it ' s there already or was added by a previous
* application of the visitor ( is in previousAddedState ) .
* If it ' s a new class of state that ' s not already handled by ReinstateRemovedStateVisitor : : apply , make sure to
* add handling there .
* Similarly , any time new state is added to writableStateSet , the equivalent method should be called on
* addedState .
* If that method doesn ' t exist yet , implement it - we don ' t use a full StateSet as we only need to check
* existence , not equality , and don ' t need to actually get the value as we can get it from writableStateSet
* instead .
*/
2017-08-31 21:39:21 +00:00
osg : : Node & node = * reqs . mNode ;
2018-10-09 10:21:12 +04:00
osg : : StateSet * writableStateSet = nullptr ;
2016-02-18 22:48:53 +01:00
if ( mAllowedToModifyStateSets )
writableStateSet = node . getOrCreateStateSet ( ) ;
else
writableStateSet = getWritableStateSet ( node ) ;
2021-07-03 21:04:17 +01:00
osg : : ref_ptr < AddedState > addedState = new AddedState ;
osg : : ref_ptr < AddedState > previousAddedState = getAddedState ( * writableStateSet ) ;
if ( ! previousAddedState )
previousAddedState = new AddedState ;
2016-02-18 22:48:53 +01:00
2016-02-16 18:18:48 +01:00
ShaderManager : : DefineMap defineMap ;
for ( unsigned int i = 0 ; i < sizeof ( defaultTextures ) / sizeof ( defaultTextures [ 0 ] ) ; + + i )
{
defineMap [ defaultTextures [ i ] ] = " 0 " ;
defineMap [ std : : string ( defaultTextures [ i ] ) + std : : string ( " UV " ) ] = " 0 " ;
}
for ( std : : map < int , std : : string > : : const_iterator texIt = reqs . mTextures . begin ( ) ; texIt ! = reqs . mTextures . end ( ) ; + + texIt )
{
defineMap [ texIt - > second ] = " 1 " ;
2017-05-06 23:05:13 +02:00
defineMap [ texIt - > second + std : : string ( " UV " ) ] = std : : to_string ( texIt - > first ) ;
2016-02-16 18:18:48 +01:00
}
2021-09-14 11:37:23 +00:00
if ( defineMap [ " diffuseMap " ] = = " 0 " )
2021-12-08 20:44:10 +00:00
{
2021-09-14 11:30:07 +00:00
writableStateSet - > addUniform ( new osg : : Uniform ( " useDiffuseMapForShadowAlpha " , false ) ) ;
2021-12-08 20:44:10 +00:00
addedState - > addUniform ( " useDiffuseMapForShadowAlpha " ) ;
}
2021-09-14 11:30:07 +00:00
2016-03-22 21:43:07 +01:00
defineMap [ " parallax " ] = reqs . mNormalHeight ? " 1 " : " 0 " ;
2018-10-18 16:00:51 +01:00
writableStateSet - > addUniform ( new osg : : Uniform ( " colorMode " , reqs . mColorMode ) ) ;
2021-07-03 21:04:17 +01:00
addedState - > addUniform ( " colorMode " ) ;
2018-10-18 16:00:51 +01:00
2020-12-16 23:44:15 +00:00
defineMap [ " alphaFunc " ] = std : : to_string ( reqs . mAlphaFunc ) ;
2020-12-24 00:32:15 +00:00
2022-06-21 22:28:17 +02:00
defineMap [ " additiveBlending " ] = reqs . mAdditiveBlending ? " 1 " : " 0 " ;
2020-12-24 00:32:15 +00:00
osg : : ref_ptr < osg : : StateSet > removedState ;
if ( ( removedState = getRemovedState ( * writableStateSet ) ) & & ! mAllowedToModifyStateSets )
removedState = new osg : : StateSet ( * removedState , osg : : CopyOp : : SHALLOW_COPY ) ;
if ( ! removedState )
removedState = new osg : : StateSet ( ) ;
2020-12-26 22:45:53 +00:00
defineMap [ " alphaToCoverage " ] = " 0 " ;
2021-07-04 23:29:22 +01:00
defineMap [ " adjustCoverage " ] = " 0 " ;
2020-12-16 23:44:15 +00:00
if ( reqs . mAlphaFunc ! = osg : : AlphaFunc : : ALWAYS )
{
writableStateSet - > addUniform ( new osg : : Uniform ( " alphaRef " , reqs . mAlphaRef ) ) ;
2021-07-03 21:04:17 +01:00
addedState - > addUniform ( " alphaRef " ) ;
2020-12-16 23:44:15 +00:00
2021-06-28 00:11:31 +01:00
if ( ! removedState - > getAttributePair ( osg : : StateAttribute : : ALPHAFUNC ) )
{
const auto * alphaFunc = writableStateSet - > getAttributePair ( osg : : StateAttribute : : ALPHAFUNC ) ;
2021-07-03 21:04:17 +01:00
if ( alphaFunc & & ! previousAddedState - > hasAttribute ( osg : : StateAttribute : : ALPHAFUNC , 0 ) )
2021-06-28 00:11:31 +01:00
removedState - > setAttribute ( alphaFunc - > first , alphaFunc - > second ) ;
}
2020-12-16 23:44:15 +00:00
// This prevents redundant glAlphaFunc calls while letting the shadows bin still see the test
writableStateSet - > setAttribute ( RemovedAlphaFunc : : getInstance ( reqs . mAlphaFunc ) , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2021-07-03 21:04:17 +01:00
addedState - > setAttribute ( RemovedAlphaFunc : : getInstance ( reqs . mAlphaFunc ) ) ;
2020-12-26 22:45:53 +00:00
2021-01-02 19:09:06 +00:00
// Blending won't work with A2C as we use the alpha channel for coverage. gl_SampleCoverage from ARB_sample_shading would save the day, but requires GLSL 130
if ( mConvertAlphaTestToAlphaToCoverage & & ! reqs . mAlphaBlend )
2020-12-26 22:45:53 +00:00
{
writableStateSet - > setMode ( GL_SAMPLE_ALPHA_TO_COVERAGE_ARB , osg : : StateAttribute : : ON ) ;
2021-07-03 21:04:17 +01:00
addedState - > setMode ( GL_SAMPLE_ALPHA_TO_COVERAGE_ARB ) ;
2020-12-26 22:45:53 +00:00
defineMap [ " alphaToCoverage " ] = " 1 " ;
}
2021-01-07 18:13:51 +00:00
2021-07-04 23:29:22 +01:00
// Adjusting coverage isn't safe with blending on as blending requires the alpha to be intact.
// Maybe we could also somehow (e.g. userdata) detect when the diffuse map has coverage-preserving mip maps in the future
if ( ! reqs . mAlphaBlend )
defineMap [ " adjustCoverage " ] = " 1 " ;
2021-01-07 18:13:51 +00:00
// Preventing alpha tested stuff shrinking as lower mip levels are used requires knowing the texture size
osg : : ref_ptr < osg : : GLExtensions > exts = osg : : GLExtensions : : Get ( 0 , false ) ;
if ( exts & & exts - > isGpuShader4Supported )
defineMap [ " useGPUShader4 " ] = " 1 " ;
// We could fall back to a texture size uniform if EXT_gpu_shader4 is missing
2020-12-24 00:32:15 +00:00
}
2020-12-16 23:44:15 +00:00
2021-10-23 17:53:38 -07:00
bool simpleLighting = false ;
node . getUserValue ( " simpleLighting " , simpleLighting ) ;
2021-10-25 10:23:16 -07:00
if ( simpleLighting )
{
defineMap [ " forcePPL " ] = " 1 " ;
defineMap [ " endLight " ] = " 0 " ;
}
2021-10-23 17:53:38 -07:00
2022-05-13 18:58:00 -07:00
if ( reqs . mAlphaBlend & & mSupportsNormalsRT )
{
if ( reqs . mSoftParticles )
defineMap [ " disableNormals " ] = " 1 " ;
else
writableStateSet - > setAttribute ( new osg : : Disablei ( GL_BLEND , 1 ) ) ;
}
2021-07-03 21:04:17 +01:00
if ( writableStateSet - > getMode ( GL_ALPHA_TEST ) ! = osg : : StateAttribute : : INHERIT & & ! previousAddedState - > hasMode ( GL_ALPHA_TEST ) )
2020-12-24 00:32:15 +00:00
removedState - > setMode ( GL_ALPHA_TEST , writableStateSet - > getMode ( GL_ALPHA_TEST ) ) ;
// This disables the deprecated fixed-function alpha test
writableStateSet - > setMode ( GL_ALPHA_TEST , osg : : StateAttribute : : OFF | osg : : StateAttribute : : PROTECTED ) ;
2021-07-03 21:04:17 +01:00
addedState - > setMode ( GL_ALPHA_TEST ) ;
2020-12-16 23:44:15 +00:00
2020-12-24 00:32:15 +00:00
if ( ! removedState - > getModeList ( ) . empty ( ) | | ! removedState - > getAttributeList ( ) . empty ( ) )
{
// user data is normally shallow copied so shared with the original stateset
osg : : ref_ptr < osg : : UserDataContainer > writableUserData ;
if ( mAllowedToModifyStateSets )
writableUserData = writableStateSet - > getOrCreateUserDataContainer ( ) ;
else
writableUserData = getWritableUserDataContainer ( * writableStateSet ) ;
updateRemovedState ( * writableUserData , removedState ) ;
2020-12-16 23:44:15 +00:00
}
2022-05-13 18:58:00 -07:00
if ( reqs . mSoftParticles & & mOpaqueDepthTex . front ( ) )
2021-11-20 18:39:20 -08:00
{
osg : : ref_ptr < osg : : Depth > depth = new SceneUtil : : AutoDepth ;
depth - > setWriteMask ( false ) ;
writableStateSet - > setAttributeAndModes ( depth , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
2021-12-08 20:44:10 +00:00
addedState - > setAttributeAndModes ( depth ) ;
2021-11-20 18:39:20 -08:00
writableStateSet - > addUniform ( new osg : : Uniform ( " particleSize " , reqs . mSoftParticleSize ) ) ;
addedState - > addUniform ( " particleSize " ) ;
2022-05-13 18:58:00 -07:00
constexpr int unit = 2 ;
writableStateSet - > addUniform ( new osg : : Uniform ( " opaqueDepthTex " , unit ) ) ;
2021-11-20 18:39:20 -08:00
addedState - > addUniform ( " opaqueDepthTex " ) ;
2022-05-13 18:58:00 -07:00
osg : : ref_ptr < OpaqueDepthAttribute > opaqueDepthAttr = new OpaqueDepthAttribute ;
opaqueDepthAttr - > setTexturesAndUnit ( mOpaqueDepthTex , unit ) ;
writableStateSet - > setAttributeAndModes ( opaqueDepthAttr , osg : : StateAttribute : : ON | osg : : StateAttribute : : OVERRIDE ) ;
addedState - > setAttributeAndModes ( opaqueDepthAttr ) ;
2021-11-20 18:39:20 -08:00
}
2022-05-13 18:58:00 -07:00
defineMap [ " softParticles " ] = reqs . mSoftParticles & & mOpaqueDepthTex . front ( ) ? " 1 " : " 0 " ;
2021-11-20 18:39:20 -08:00
2022-04-04 22:51:23 +02:00
Stereo : : Manager : : instance ( ) . shaderStereoDefines ( defineMap ) ;
2020-12-17 00:46:09 +03:00
std : : string shaderPrefix ;
if ( ! node . getUserValue ( " shaderPrefix " , shaderPrefix ) )
shaderPrefix = mDefaultShaderPrefix ;
osg : : ref_ptr < osg : : Shader > vertexShader ( mShaderManager . getShader ( shaderPrefix + " _vertex.glsl " , defineMap , osg : : Shader : : VERTEX ) ) ;
osg : : ref_ptr < osg : : Shader > fragmentShader ( mShaderManager . getShader ( shaderPrefix + " _fragment.glsl " , defineMap , osg : : Shader : : FRAGMENT ) ) ;
2016-02-16 18:18:48 +01:00
if ( vertexShader & & fragmentShader )
{
2021-09-29 13:40:37 +00:00
auto program = mShaderManager . getProgram ( vertexShader , fragmentShader , mProgramTemplate ) ;
2021-07-03 21:04:17 +01:00
writableStateSet - > setAttributeAndModes ( program , osg : : StateAttribute : : ON ) ;
addedState - > setAttributeAndModes ( program ) ;
2016-02-16 18:18:48 +01:00
for ( std : : map < int , std : : string > : : const_iterator texIt = reqs . mTextures . begin ( ) ; texIt ! = reqs . mTextures . end ( ) ; + + texIt )
{
2016-02-18 22:48:53 +01:00
writableStateSet - > addUniform ( new osg : : Uniform ( texIt - > second . c_str ( ) , texIt - > first ) , osg : : StateAttribute : : ON ) ;
2021-07-03 21:04:17 +01:00
addedState - > addUniform ( texIt - > second ) ;
2016-02-16 18:18:48 +01:00
}
}
2021-12-08 20:44:10 +00:00
if ( ! addedState - > empty ( ) )
{
// user data is normally shallow copied so shared with the original stateset
osg : : ref_ptr < osg : : UserDataContainer > writableUserData ;
if ( mAllowedToModifyStateSets )
writableUserData = writableStateSet - > getOrCreateUserDataContainer ( ) ;
else
writableUserData = getWritableUserDataContainer ( * writableStateSet ) ;
updateAddedState ( * writableUserData , addedState ) ;
}
2016-02-16 18:18:48 +01:00
}
2020-12-03 21:06:02 +00:00
void ShaderVisitor : : ensureFFP ( osg : : Node & node )
{
if ( ! node . getStateSet ( ) | | ! node . getStateSet ( ) - > getAttribute ( osg : : StateAttribute : : PROGRAM ) )
return ;
osg : : StateSet * writableStateSet = nullptr ;
if ( mAllowedToModifyStateSets )
writableStateSet = node . getStateSet ( ) ;
else
writableStateSet = getWritableStateSet ( node ) ;
2021-12-08 20:44:10 +00:00
/**
* We might have been using shaders temporarily with the node ( e . g . if a GlowUpdater applied a temporary
* environment map for a temporary enchantment ) .
* We therefore need to remove any state doing so added , and restore any that it removed .
* This is kept track of in createProgram in the StateSet ' s userdata .
* If new classes of state get added , handling it here is required - not all StateSet features are implemented
* in AddedState yet as so far they ' ve not been necessary .
* Removed state requires no particular special handling as it ' s dealt with by merging StateSets .
* We don ' t need to worry about state in writableStateSet having the OVERRIDE flag as if it ' s in both , it ' s also
* in addedState , and gets removed first .
*/
2021-07-03 21:04:17 +01:00
// user data is normally shallow copied so shared with the original stateset - we'll need to copy before edits
osg : : ref_ptr < osg : : UserDataContainer > writableUserData ;
if ( osg : : ref_ptr < AddedState > addedState = getAddedState ( * writableStateSet ) )
{
if ( mAllowedToModifyStateSets )
writableUserData = writableStateSet - > getUserDataContainer ( ) ;
else
writableUserData = getWritableUserDataContainer ( * writableStateSet ) ;
unsigned int index = writableUserData - > getUserObjectIndex ( " addedState " ) ;
writableUserData - > removeUserObject ( index ) ;
// O(n log n) to use StateSet::removeX, but this is O(n)
for ( auto itr = writableStateSet - > getUniformList ( ) . begin ( ) ; itr ! = writableStateSet - > getUniformList ( ) . end ( ) ; )
{
if ( addedState - > hasUniform ( itr - > first ) )
2021-07-17 20:47:55 +01:00
writableStateSet - > getUniformList ( ) . erase ( itr + + ) ;
2021-07-03 21:04:17 +01:00
else
+ + itr ;
}
for ( auto itr = writableStateSet - > getModeList ( ) . begin ( ) ; itr ! = writableStateSet - > getModeList ( ) . end ( ) ; )
{
if ( addedState - > hasMode ( itr - > first ) )
2021-07-17 20:47:55 +01:00
writableStateSet - > getModeList ( ) . erase ( itr + + ) ;
2021-07-03 21:04:17 +01:00
else
+ + itr ;
}
// StateAttributes track the StateSets they're attached to
// We don't have access to the function to do that, and can't call removeAttribute with an iterator
for ( const auto & [ type , member ] : addedState - > getAttributes ( ) )
writableStateSet - > removeAttribute ( type , member ) ;
2021-12-08 20:44:10 +00:00
for ( unsigned int unit = 0 ; unit < writableStateSet - > getTextureModeList ( ) . size ( ) ; + + unit )
{
for ( auto itr = writableStateSet - > getTextureModeList ( ) [ unit ] . begin ( ) ; itr ! = writableStateSet - > getTextureModeList ( ) [ unit ] . end ( ) ; )
{
if ( addedState - > hasTextureMode ( unit , itr - > first ) )
writableStateSet - > getTextureModeList ( ) [ unit ] . erase ( itr + + ) ;
else
+ + itr ;
}
}
for ( const auto & [ unit , attributeList ] : addedState - > getTextureAttributes ( ) )
{
for ( const auto & [ type , member ] : attributeList )
writableStateSet - > removeTextureAttribute ( unit , type ) ;
}
2021-07-03 21:04:17 +01:00
}
2020-12-16 23:44:15 +00:00
2021-04-14 08:07:56 +04:00
if ( osg : : ref_ptr < osg : : StateSet > removedState = getRemovedState ( * writableStateSet ) )
2020-12-16 23:44:15 +00:00
{
2021-07-03 21:04:17 +01:00
if ( ! writableUserData )
{
2020-12-16 23:44:15 +00:00
if ( mAllowedToModifyStateSets )
writableUserData = writableStateSet - > getUserDataContainer ( ) ;
else
writableUserData = getWritableUserDataContainer ( * writableStateSet ) ;
2021-07-03 21:04:17 +01:00
}
2020-12-16 23:44:15 +00:00
unsigned int index = writableUserData - > getUserObjectIndex ( " removedState " ) ;
writableUserData - > removeUserObject ( index ) ;
2021-07-03 21:04:17 +01:00
writableStateSet - > merge ( * removedState ) ;
2020-12-16 23:44:15 +00:00
}
2020-12-03 21:06:02 +00:00
}
2017-09-01 22:01:50 +02:00
bool ShaderVisitor : : adjustGeometry ( osg : : Geometry & sourceGeometry , const ShaderRequirements & reqs )
{
bool useShader = reqs . mShaderRequired | | mForceShaders ;
bool generateTangents = reqs . mTexStageRequiringTangents ! = - 1 ;
bool changed = false ;
if ( mAllowedToModifyStateSets & & ( useShader | | generateTangents ) )
{
// make sure that all UV sets are there
for ( std : : map < int , std : : string > : : const_iterator it = reqs . mTextures . begin ( ) ; it ! = reqs . mTextures . end ( ) ; + + it )
{
2018-10-09 10:21:12 +04:00
if ( sourceGeometry . getTexCoordArray ( it - > first ) = = nullptr )
2017-09-01 22:01:50 +02:00
{
sourceGeometry . setTexCoordArray ( it - > first , sourceGeometry . getTexCoordArray ( 0 ) ) ;
changed = true ;
}
}
if ( generateTangents )
{
osg : : ref_ptr < osgUtil : : TangentSpaceGenerator > generator ( new osgUtil : : TangentSpaceGenerator ) ;
generator - > generate ( & sourceGeometry , reqs . mTexStageRequiringTangents ) ;
sourceGeometry . setTexCoordArray ( 7 , generator - > getTangentArray ( ) , osg : : Array : : BIND_PER_VERTEX ) ;
changed = true ;
}
}
return changed ;
}
2016-02-16 18:18:48 +01:00
void ShaderVisitor : : apply ( osg : : Geometry & geometry )
{
2018-10-09 10:21:12 +04:00
bool needPop = ( geometry . getStateSet ( ) ! = nullptr ) ;
2017-08-31 21:39:21 +00:00
if ( geometry . getStateSet ( ) ) // TODO: check if stateset affects shader permutation before pushing it
2016-02-16 18:18:48 +01:00
{
2017-08-31 21:39:21 +00:00
pushRequirements ( geometry ) ;
2016-02-18 22:48:53 +01:00
applyStateSet ( geometry . getStateSet ( ) , geometry ) ;
2016-02-16 18:18:48 +01:00
}
if ( ! mRequirements . empty ( ) )
{
const ShaderRequirements & reqs = mRequirements . back ( ) ;
2016-02-20 19:02:11 +01:00
2017-09-01 22:01:50 +02:00
adjustGeometry ( geometry , reqs ) ;
2017-03-01 03:00:08 +01:00
2017-09-01 22:01:50 +02:00
createProgram ( reqs ) ;
2016-02-16 18:18:48 +01:00
}
2020-12-03 21:06:02 +00:00
else
ensureFFP ( geometry ) ;
2016-02-16 18:18:48 +01:00
if ( needPop )
popRequirements ( ) ;
}
void ShaderVisitor : : apply ( osg : : Drawable & drawable )
{
2021-11-20 18:39:20 -08:00
auto partsys = dynamic_cast < osgParticle : : ParticleSystem * > ( & drawable ) ;
bool needPop = drawable . getStateSet ( ) | | partsys ;
2016-02-16 18:18:48 +01:00
2021-11-20 18:39:20 -08:00
if ( needPop )
2016-02-16 18:18:48 +01:00
{
2017-08-31 21:39:21 +00:00
pushRequirements ( drawable ) ;
2021-11-20 18:39:20 -08:00
2022-05-13 18:58:00 -07:00
if ( partsys )
2021-11-20 18:39:20 -08:00
{
mRequirements . back ( ) . mSoftParticles = true ;
mRequirements . back ( ) . mSoftParticleSize = partsys - > getDefaultParticleTemplate ( ) . getSizeRange ( ) . maximum ;
}
if ( drawable . getStateSet ( ) )
applyStateSet ( drawable . getStateSet ( ) , drawable ) ;
2016-02-16 18:18:48 +01:00
}
if ( ! mRequirements . empty ( ) )
{
2016-02-18 17:08:18 +01:00
const ShaderRequirements & reqs = mRequirements . back ( ) ;
2017-09-01 22:01:50 +02:00
createProgram ( reqs ) ;
2017-09-01 22:56:09 +02:00
if ( auto rig = dynamic_cast < SceneUtil : : RigGeometry * > ( & drawable ) )
2017-09-01 22:01:50 +02:00
{
osg : : ref_ptr < osg : : Geometry > sourceGeometry = rig - > getSourceGeometry ( ) ;
if ( sourceGeometry & & adjustGeometry ( * sourceGeometry , reqs ) )
rig - > setSourceGeometry ( sourceGeometry ) ;
}
2017-09-01 22:56:09 +02:00
else if ( auto morph = dynamic_cast < SceneUtil : : MorphGeometry * > ( & drawable ) )
{
osg : : ref_ptr < osg : : Geometry > sourceGeometry = morph - > getSourceGeometry ( ) ;
if ( sourceGeometry & & adjustGeometry ( * sourceGeometry , reqs ) )
morph - > setSourceGeometry ( sourceGeometry ) ;
}
2022-06-21 15:27:34 +00:00
else if ( auto osgaRig = dynamic_cast < SceneUtil : : RigGeometryHolder * > ( & drawable ) )
{
osg : : ref_ptr < SceneUtil : : OsgaRigGeometry > sourceOsgaRigGeometry = osgaRig - > getSourceRigGeometry ( ) ;
osg : : ref_ptr < osg : : Geometry > sourceGeometry = sourceOsgaRigGeometry - > getSourceGeometry ( ) ;
if ( sourceGeometry & & adjustGeometry ( * sourceGeometry , reqs ) )
{
sourceOsgaRigGeometry - > setSourceGeometry ( sourceGeometry ) ;
osgaRig - > setSourceRigGeometry ( sourceOsgaRigGeometry ) ;
}
}
2016-02-16 18:18:48 +01:00
}
2020-12-03 21:06:02 +00:00
else
ensureFFP ( drawable ) ;
2016-02-16 18:18:48 +01:00
if ( needPop )
popRequirements ( ) ;
}
2016-02-18 22:48:53 +01:00
void ShaderVisitor : : setAllowedToModifyStateSets ( bool allowed )
{
mAllowedToModifyStateSets = allowed ;
}
2016-02-20 17:57:19 +01:00
void ShaderVisitor : : setAutoUseNormalMaps ( bool use )
{
mAutoUseNormalMaps = use ;
}
void ShaderVisitor : : setNormalMapPattern ( const std : : string & pattern )
{
mNormalMapPattern = pattern ;
}
2016-03-22 21:43:07 +01:00
void ShaderVisitor : : setNormalHeightMapPattern ( const std : : string & pattern )
{
mNormalHeightMapPattern = pattern ;
}
2016-02-20 19:02:11 +01:00
void ShaderVisitor : : setAutoUseSpecularMaps ( bool use )
{
mAutoUseSpecularMaps = use ;
}
void ShaderVisitor : : setSpecularMapPattern ( const std : : string & pattern )
{
mSpecularMapPattern = pattern ;
}
2020-11-09 13:53:58 +03:00
void ShaderVisitor : : setApplyLightingToEnvMaps ( bool apply )
{
mApplyLightingToEnvMaps = apply ;
}
2020-12-26 22:45:53 +00:00
void ShaderVisitor : : setConvertAlphaTestToAlphaToCoverage ( bool convert )
{
mConvertAlphaTestToAlphaToCoverage = convert ;
}
2022-06-21 15:55:06 +00:00
void ShaderVisitor : : setOpaqueDepthTex ( osg : : ref_ptr < osg : : Texture > texturePing , osg : : ref_ptr < osg : : Texture > texturePong )
2021-10-20 09:42:18 -07:00
{
2022-05-13 18:58:00 -07:00
mOpaqueDepthTex = { texturePing , texturePong } ;
2021-10-20 09:42:18 -07:00
}
2021-02-19 19:59:48 +00:00
ReinstateRemovedStateVisitor : : ReinstateRemovedStateVisitor ( bool allowedToModifyStateSets )
: osg : : NodeVisitor ( TRAVERSE_ALL_CHILDREN )
, mAllowedToModifyStateSets ( allowedToModifyStateSets )
{
}
void ReinstateRemovedStateVisitor : : apply ( osg : : Node & node )
{
2021-12-08 20:44:10 +00:00
// TODO: this may eventually need to remove added state.
// If so, we can migrate from explicitly copying removed state to just calling osg::StateSet::merge.
// Not everything is transferred from removedState yet - implement more when createProgram starts marking more
// as removed.
2021-02-19 19:59:48 +00:00
if ( node . getStateSet ( ) )
{
osg : : ref_ptr < osg : : StateSet > removedState = getRemovedState ( * node . getStateSet ( ) ) ;
if ( removedState )
{
osg : : ref_ptr < osg : : StateSet > writableStateSet ;
if ( mAllowedToModifyStateSets )
writableStateSet = node . getStateSet ( ) ;
else
writableStateSet = getWritableStateSet ( node ) ;
// user data is normally shallow copied so shared with the original stateset
osg : : ref_ptr < osg : : UserDataContainer > writableUserData ;
if ( mAllowedToModifyStateSets )
writableUserData = writableStateSet - > getUserDataContainer ( ) ;
else
writableUserData = getWritableUserDataContainer ( * writableStateSet ) ;
unsigned int index = writableUserData - > getUserObjectIndex ( " removedState " ) ;
writableUserData - > removeUserObject ( index ) ;
for ( const auto & [ mode , value ] : removedState - > getModeList ( ) )
writableStateSet - > setMode ( mode , value ) ;
for ( const auto & attribute : removedState - > getAttributeList ( ) )
writableStateSet - > setAttribute ( attribute . second . first , attribute . second . second ) ;
2021-11-20 18:39:20 -08:00
for ( unsigned int unit = 0 ; unit < removedState - > getTextureModeList ( ) . size ( ) ; + + unit )
{
for ( const auto & [ mode , value ] : removedState - > getTextureModeList ( ) [ unit ] )
writableStateSet - > setTextureMode ( unit , mode , value ) ;
}
2021-02-19 19:59:48 +00:00
}
}
traverse ( node ) ;
}
2016-02-16 18:18:48 +01:00
}