2017-10-03 02:40:23 +00:00
|
|
|
#include "shadow.hpp"
|
|
|
|
|
2021-10-11 09:27:50 +00:00
|
|
|
#include <osgShadow/ShadowSettings>
|
2017-10-11 19:47:19 +00:00
|
|
|
#include <osgShadow/ShadowedScene>
|
|
|
|
|
2022-08-02 22:00:54 +00:00
|
|
|
#include <components/misc/strings/algorithm.hpp>
|
2017-12-27 02:32:17 +00:00
|
|
|
#include <components/settings/settings.hpp>
|
2022-04-04 20:51:23 +00:00
|
|
|
#include <components/stereo/stereomanager.hpp>
|
2017-12-27 02:32:17 +00:00
|
|
|
|
2021-10-11 09:27:50 +00:00
|
|
|
#include "mwshadowtechnique.hpp"
|
|
|
|
|
2017-11-08 01:44:49 +00:00
|
|
|
namespace SceneUtil
|
2017-10-03 02:40:23 +00:00
|
|
|
{
|
2017-10-11 19:47:19 +00:00
|
|
|
using namespace osgShadow;
|
|
|
|
|
2023-02-09 01:32:48 +00:00
|
|
|
void ShadowManager::setupShadowSettings(Shader::ShaderManager& shaderManager)
|
2017-12-27 02:32:17 +00:00
|
|
|
{
|
2018-02-26 23:52:46 +00:00
|
|
|
mEnableShadows = Settings::Manager::getBool("enable shadows", "Shadows");
|
|
|
|
|
|
|
|
if (!mEnableShadows)
|
2018-02-26 22:27:09 +00:00
|
|
|
{
|
|
|
|
mShadowTechnique->disableShadows();
|
2017-12-27 02:32:17 +00:00
|
|
|
return;
|
2018-02-26 22:27:09 +00:00
|
|
|
}
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2018-02-26 23:52:46 +00:00
|
|
|
mShadowTechnique->enableShadows();
|
2017-12-27 02:32:17 +00:00
|
|
|
|
2018-02-26 23:52:46 +00:00
|
|
|
mShadowSettings->setLightNum(0);
|
2020-04-20 16:47:14 +00:00
|
|
|
mShadowSettings->setReceivesShadowTraversalMask(~0u);
|
2017-12-27 02:32:17 +00:00
|
|
|
|
2021-11-06 04:30:28 +00:00
|
|
|
const int numberOfShadowMapsPerLight
|
|
|
|
= std::clamp(Settings::Manager::getInt("number of shadow maps", "Shadows"), 1, 8);
|
2019-11-17 13:24:20 +00:00
|
|
|
|
2018-02-26 23:52:46 +00:00
|
|
|
mShadowSettings->setNumShadowMapsPerLight(numberOfShadowMapsPerLight);
|
2023-02-10 00:12:55 +00:00
|
|
|
mShadowSettings->setBaseShadowTextureUnit(shaderManager.reserveGlobalTextureUnits(
|
|
|
|
Shader::ShaderManager::Slot::ShadowMaps, numberOfShadowMapsPerLight));
|
2017-12-27 02:32:17 +00:00
|
|
|
|
2019-10-28 18:17:09 +00:00
|
|
|
const float maximumShadowMapDistance = Settings::Manager::getFloat("maximum shadow map distance", "Shadows");
|
|
|
|
if (maximumShadowMapDistance > 0)
|
|
|
|
{
|
2021-11-06 04:30:28 +00:00
|
|
|
const float shadowFadeStart
|
|
|
|
= std::clamp(Settings::Manager::getFloat("shadow fade start", "Shadows"), 0.f, 1.f);
|
2019-10-28 18:17:09 +00:00
|
|
|
mShadowSettings->setMaximumShadowMapDistance(maximumShadowMapDistance);
|
|
|
|
mShadowTechnique->setShadowFadeStart(maximumShadowMapDistance * shadowFadeStart);
|
|
|
|
}
|
|
|
|
|
2018-02-26 23:52:46 +00:00
|
|
|
mShadowSettings->setMinimumShadowMapNearFarRatio(
|
|
|
|
Settings::Manager::getFloat("minimum lispsm near far ratio", "Shadows"));
|
2020-04-29 09:46:52 +00:00
|
|
|
|
2022-08-28 14:38:11 +00:00
|
|
|
const std::string& computeSceneBounds = Settings::Manager::getString("compute scene bounds", "Shadows");
|
|
|
|
if (Misc::StringUtils::ciEqual(computeSceneBounds, "primitives"))
|
2018-02-26 23:52:46 +00:00
|
|
|
mShadowSettings->setComputeNearFarModeOverride(osg::CullSettings::COMPUTE_NEAR_FAR_USING_PRIMITIVES);
|
2022-08-28 14:38:11 +00:00
|
|
|
else if (Misc::StringUtils::ciEqual(computeSceneBounds, "bounds"))
|
2019-06-21 13:37:00 +00:00
|
|
|
mShadowSettings->setComputeNearFarModeOverride(osg::CullSettings::COMPUTE_NEAR_FAR_USING_BOUNDING_VOLUMES);
|
2017-12-27 02:32:17 +00:00
|
|
|
|
|
|
|
int mapres = Settings::Manager::getInt("shadow map resolution", "Shadows");
|
2018-02-26 23:52:46 +00:00
|
|
|
mShadowSettings->setTextureSize(osg::Vec2s(mapres, mapres));
|
2018-02-26 22:27:09 +00:00
|
|
|
|
2018-02-26 23:00:46 +00:00
|
|
|
mShadowTechnique->setSplitPointUniformLogarithmicRatio(
|
|
|
|
Settings::Manager::getFloat("split point uniform logarithmic ratio", "Shadows"));
|
|
|
|
mShadowTechnique->setSplitPointDeltaBias(Settings::Manager::getFloat("split point bias", "Shadows"));
|
|
|
|
|
2018-11-30 00:55:54 +00:00
|
|
|
mShadowTechnique->setPolygonOffset(Settings::Manager::getFloat("polygon offset factor", "Shadows"),
|
|
|
|
Settings::Manager::getFloat("polygon offset units", "Shadows"));
|
|
|
|
|
2018-12-01 00:26:43 +00:00
|
|
|
if (Settings::Manager::getBool("use front face culling", "Shadows"))
|
|
|
|
mShadowTechnique->enableFrontFaceCulling();
|
|
|
|
else
|
|
|
|
mShadowTechnique->disableFrontFaceCulling();
|
|
|
|
|
2018-05-17 15:57:01 +00:00
|
|
|
if (Settings::Manager::getBool("allow shadow map overlap", "Shadows"))
|
|
|
|
mShadowSettings->setMultipleShadowMapHint(osgShadow::ShadowSettings::CASCADED);
|
|
|
|
else
|
|
|
|
mShadowSettings->setMultipleShadowMapHint(osgShadow::ShadowSettings::PARALLEL_SPLIT);
|
|
|
|
|
2018-02-26 22:27:09 +00:00
|
|
|
if (Settings::Manager::getBool("enable debug hud", "Shadows"))
|
|
|
|
mShadowTechnique->enableDebugHUD();
|
|
|
|
else
|
|
|
|
mShadowTechnique->disableDebugHUD();
|
2017-12-27 02:32:17 +00:00
|
|
|
}
|
2017-10-11 19:47:19 +00:00
|
|
|
|
2018-02-26 14:29:31 +00:00
|
|
|
void ShadowManager::disableShadowsForStateSet(osg::ref_ptr<osg::StateSet> stateset)
|
2017-10-11 19:47:19 +00:00
|
|
|
{
|
2020-05-17 07:52:23 +00:00
|
|
|
if (!Settings::Manager::getBool("enable shadows", "Shadows"))
|
|
|
|
return;
|
|
|
|
|
2021-11-06 04:30:28 +00:00
|
|
|
const int numberOfShadowMapsPerLight
|
|
|
|
= std::clamp(Settings::Manager::getInt("number of shadow maps", "Shadows"), 1, 8);
|
2019-11-13 11:07:36 +00:00
|
|
|
|
2017-12-27 02:32:17 +00:00
|
|
|
int baseShadowTextureUnit = 8 - numberOfShadowMapsPerLight;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2017-12-27 02:32:17 +00:00
|
|
|
osg::ref_ptr<osg::Image> fakeShadowMapImage = new osg::Image();
|
|
|
|
fakeShadowMapImage->allocateImage(1, 1, 1, GL_DEPTH_COMPONENT, GL_FLOAT);
|
|
|
|
*(float*)fakeShadowMapImage->data() = std::numeric_limits<float>::infinity();
|
|
|
|
osg::ref_ptr<osg::Texture> fakeShadowMapTexture = new osg::Texture2D(fakeShadowMapImage);
|
2023-06-14 20:14:38 +00:00
|
|
|
fakeShadowMapTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
|
|
|
|
fakeShadowMapTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
|
2017-12-27 02:32:17 +00:00
|
|
|
fakeShadowMapTexture->setShadowComparison(true);
|
|
|
|
fakeShadowMapTexture->setShadowCompareFunc(osg::Texture::ShadowCompareFunc::ALWAYS);
|
|
|
|
for (int i = baseShadowTextureUnit; i < baseShadowTextureUnit + numberOfShadowMapsPerLight; ++i)
|
2018-10-30 17:07:11 +00:00
|
|
|
{
|
2017-12-27 02:32:17 +00:00
|
|
|
stateset->setTextureAttributeAndModes(i, fakeShadowMapTexture,
|
|
|
|
osg::StateAttribute::ON | osg::StateAttribute::OVERRIDE | osg::StateAttribute::PROTECTED);
|
2018-10-30 17:07:11 +00:00
|
|
|
stateset->addUniform(
|
|
|
|
new osg::Uniform(("shadowTexture" + std::to_string(i - baseShadowTextureUnit)).c_str(), i));
|
|
|
|
stateset->addUniform(
|
|
|
|
new osg::Uniform(("shadowTextureUnit" + std::to_string(i - baseShadowTextureUnit)).c_str(), i));
|
|
|
|
}
|
2017-12-27 02:32:17 +00:00
|
|
|
}
|
2017-10-11 19:47:19 +00:00
|
|
|
|
2021-10-11 09:27:50 +00:00
|
|
|
ShadowManager::ShadowManager(osg::ref_ptr<osg::Group> sceneRoot, osg::ref_ptr<osg::Group> rootNode,
|
|
|
|
unsigned int outdoorShadowCastingMask, unsigned int indoorShadowCastingMask, unsigned int worldMask,
|
|
|
|
Shader::ShaderManager& shaderManager)
|
|
|
|
: mShadowedScene(new osgShadow::ShadowedScene)
|
2018-02-27 00:13:51 +00:00
|
|
|
, mShadowTechnique(new MWShadowTechnique)
|
|
|
|
, mOutdoorShadowCastingMask(outdoorShadowCastingMask)
|
|
|
|
, mIndoorShadowCastingMask(indoorShadowCastingMask)
|
2017-10-11 19:47:19 +00:00
|
|
|
{
|
2018-02-26 22:27:09 +00:00
|
|
|
mShadowedScene->setShadowTechnique(mShadowTechnique);
|
2022-09-25 19:19:03 +00:00
|
|
|
|
|
|
|
if (Stereo::getStereo())
|
|
|
|
Stereo::Manager::instance().setShadowTechnique(mShadowTechnique);
|
2018-02-26 23:52:46 +00:00
|
|
|
|
2018-02-26 22:27:09 +00:00
|
|
|
mShadowedScene->addChild(sceneRoot);
|
|
|
|
rootNode->addChild(mShadowedScene);
|
2019-06-21 13:37:00 +00:00
|
|
|
mShadowedScene->setNodeMask(sceneRoot->getNodeMask());
|
2018-02-26 23:52:46 +00:00
|
|
|
|
|
|
|
mShadowSettings = mShadowedScene->getShadowSettings();
|
2023-02-09 01:32:48 +00:00
|
|
|
setupShadowSettings(shaderManager);
|
2018-02-27 00:13:51 +00:00
|
|
|
|
2018-10-16 20:23:31 +00:00
|
|
|
mShadowTechnique->setupCastingShader(shaderManager);
|
2021-10-11 09:27:50 +00:00
|
|
|
mShadowTechnique->setWorldMask(worldMask);
|
2018-10-16 20:23:31 +00:00
|
|
|
|
2018-02-27 00:13:51 +00:00
|
|
|
enableOutdoorMode();
|
2017-10-03 02:40:23 +00:00
|
|
|
}
|
2017-11-22 20:07:07 +00:00
|
|
|
|
2021-10-11 09:27:50 +00:00
|
|
|
ShadowManager::~ShadowManager()
|
|
|
|
{
|
2022-09-25 19:19:03 +00:00
|
|
|
if (Stereo::getStereo())
|
|
|
|
Stereo::Manager::instance().setShadowTechnique(nullptr);
|
2021-10-11 09:27:50 +00:00
|
|
|
}
|
|
|
|
|
2018-02-26 14:29:31 +00:00
|
|
|
Shader::ShaderManager::DefineMap ShadowManager::getShadowDefines()
|
2017-11-22 20:07:07 +00:00
|
|
|
{
|
2018-02-26 23:52:46 +00:00
|
|
|
if (!mEnableShadows)
|
2017-11-22 20:07:07 +00:00
|
|
|
return getShadowsDisabledDefines();
|
|
|
|
|
|
|
|
Shader::ShaderManager::DefineMap definesWithShadows;
|
2019-01-31 20:12:17 +00:00
|
|
|
|
|
|
|
definesWithShadows["shadows_enabled"] = "1";
|
|
|
|
|
2018-02-27 14:15:06 +00:00
|
|
|
for (unsigned int i = 0; i < mShadowSettings->getNumShadowMapsPerLight(); ++i)
|
2017-11-22 20:07:07 +00:00
|
|
|
definesWithShadows["shadow_texture_unit_list"] += std::to_string(i) + ",";
|
|
|
|
// remove extra comma
|
|
|
|
definesWithShadows["shadow_texture_unit_list"] = definesWithShadows["shadow_texture_unit_list"].substr(
|
|
|
|
0, definesWithShadows["shadow_texture_unit_list"].length() - 1);
|
|
|
|
|
2019-01-31 20:12:17 +00:00
|
|
|
definesWithShadows["shadowMapsOverlap"]
|
|
|
|
= Settings::Manager::getBool("allow shadow map overlap", "Shadows") ? "1" : "0";
|
2018-05-17 16:35:55 +00:00
|
|
|
|
2019-01-31 20:12:17 +00:00
|
|
|
definesWithShadows["useShadowDebugOverlay"]
|
|
|
|
= Settings::Manager::getBool("enable debug overlay", "Shadows") ? "1" : "0";
|
2018-06-28 16:24:36 +00:00
|
|
|
|
2018-12-08 20:39:41 +00:00
|
|
|
// switch this to reading settings if it's ever exposed to the user
|
2019-01-31 20:12:17 +00:00
|
|
|
definesWithShadows["perspectiveShadowMaps"]
|
|
|
|
= mShadowSettings->getShadowMapProjectionHint() == ShadowSettings::PERSPECTIVE_SHADOW_MAP ? "1" : "0";
|
2018-12-08 20:39:41 +00:00
|
|
|
|
2019-01-30 22:28:00 +00:00
|
|
|
definesWithShadows["disableNormalOffsetShadows"]
|
|
|
|
= Settings::Manager::getFloat("normal offset distance", "Shadows") == 0.0 ? "1" : "0";
|
|
|
|
|
|
|
|
definesWithShadows["shadowNormalOffset"]
|
|
|
|
= std::to_string(Settings::Manager::getFloat("normal offset distance", "Shadows"));
|
|
|
|
|
2019-10-28 18:17:09 +00:00
|
|
|
definesWithShadows["limitShadowMapDistance"]
|
|
|
|
= Settings::Manager::getFloat("maximum shadow map distance", "Shadows") > 0 ? "1" : "0";
|
|
|
|
|
2017-11-22 20:07:07 +00:00
|
|
|
return definesWithShadows;
|
|
|
|
}
|
|
|
|
|
2018-02-26 14:29:31 +00:00
|
|
|
Shader::ShaderManager::DefineMap ShadowManager::getShadowsDisabledDefines()
|
2017-11-22 20:07:07 +00:00
|
|
|
{
|
2019-01-31 14:58:57 +00:00
|
|
|
Shader::ShaderManager::DefineMap definesWithoutShadows;
|
2019-01-31 20:12:17 +00:00
|
|
|
|
|
|
|
definesWithoutShadows["shadows_enabled"] = "0";
|
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
definesWithoutShadows["shadow_texture_unit_list"] = "";
|
2017-11-22 20:07:07 +00:00
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
definesWithoutShadows["shadowMapsOverlap"] = "0";
|
2018-05-17 16:35:55 +00:00
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
definesWithoutShadows["useShadowDebugOverlay"] = "0";
|
2018-06-28 16:24:36 +00:00
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
definesWithoutShadows["perspectiveShadowMaps"] = "0";
|
2018-12-08 20:39:41 +00:00
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
definesWithoutShadows["disableNormalOffsetShadows"] = "0";
|
2019-01-30 22:28:00 +00:00
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
definesWithoutShadows["shadowNormalOffset"] = "0.0";
|
2019-01-30 22:28:00 +00:00
|
|
|
|
2019-10-28 18:17:09 +00:00
|
|
|
definesWithoutShadows["limitShadowMapDistance"] = "0";
|
|
|
|
|
2019-01-31 14:58:57 +00:00
|
|
|
return definesWithoutShadows;
|
2017-11-22 20:07:07 +00:00
|
|
|
}
|
2019-01-31 20:12:17 +00:00
|
|
|
|
2018-02-27 00:13:51 +00:00
|
|
|
void ShadowManager::enableIndoorMode()
|
|
|
|
{
|
2018-08-08 22:56:11 +00:00
|
|
|
if (Settings::Manager::getBool("enable indoor shadows", "Shadows"))
|
|
|
|
mShadowSettings->setCastsShadowTraversalMask(mIndoorShadowCastingMask);
|
|
|
|
else
|
2020-11-16 21:01:20 +00:00
|
|
|
mShadowTechnique->disableShadows(true);
|
2018-02-27 00:13:51 +00:00
|
|
|
}
|
2019-01-31 20:12:17 +00:00
|
|
|
|
2018-02-27 00:13:51 +00:00
|
|
|
void ShadowManager::enableOutdoorMode()
|
|
|
|
{
|
2018-08-08 22:56:11 +00:00
|
|
|
if (mEnableShadows)
|
|
|
|
mShadowTechnique->enableShadows();
|
2018-02-27 00:13:51 +00:00
|
|
|
mShadowSettings->setCastsShadowTraversalMask(mOutdoorShadowCastingMask);
|
|
|
|
}
|
2017-10-03 02:40:23 +00:00
|
|
|
}
|