1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-16 18:59:57 +00:00
openmw/components/sceneutil/lightutil.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

148 lines
5.2 KiB
C++
Raw Normal View History

2016-01-06 12:19:15 +00:00
#include "lightutil.hpp"
2016-01-06 12:37:55 +00:00
#include <osg/Group>
2016-01-06 12:19:15 +00:00
#include <osg/Light>
2016-01-06 12:37:55 +00:00
2022-06-05 04:01:01 +00:00
#include <osgParticle/ParticleSystem>
#include <components/esm3/loadligh.hpp>
#include <components/fallback/fallback.hpp>
#include <components/sceneutil/lightcommon.hpp>
2016-01-06 12:37:55 +00:00
#include "lightcontroller.hpp"
#include "lightmanager.hpp"
#include "util.hpp"
#include "visitor.hpp"
2016-01-06 12:19:15 +00:00
2022-06-05 04:01:01 +00:00
namespace
{
class CheckEmptyLightVisitor : public osg::NodeVisitor
{
public:
CheckEmptyLightVisitor()
: osg::NodeVisitor(TRAVERSE_ALL_CHILDREN)
{
}
void apply(osg::Drawable& drawable) override
{
if (!mEmpty)
return;
if (dynamic_cast<const osgParticle::ParticleSystem*>(&drawable))
mEmpty = false;
else
traverse(drawable);
}
void apply(osg::Geometry& geometry) override { mEmpty = false; }
bool mEmpty = true;
};
}
2016-01-06 12:19:15 +00:00
namespace SceneUtil
{
void configureLight(osg::Light* light, float radius, bool isExterior)
2016-01-06 12:19:15 +00:00
{
float quadraticAttenuation = 0.f;
float linearAttenuation = 0.f;
float constantAttenuation = 0.f;
2016-01-06 12:19:15 +00:00
static const bool useConstant = Fallback::Map::getBool("LightAttenuation_UseConstant");
static const bool useLinear = Fallback::Map::getBool("LightAttenuation_UseLinear");
static const bool useQuadratic = Fallback::Map::getBool("LightAttenuation_UseQuadratic");
static const float constantValue = Fallback::Map::getFloat("LightAttenuation_ConstantValue");
static const float linearValue = Fallback::Map::getFloat("LightAttenuation_LinearValue");
static const float quadraticValue = Fallback::Map::getFloat("LightAttenuation_QuadraticValue");
static const float linearRadiusMult = Fallback::Map::getFloat("LightAttenuation_LinearRadiusMult");
static const float quadraticRadiusMult = Fallback::Map::getFloat("LightAttenuation_QuadraticRadiusMult");
static const int linearMethod = Fallback::Map::getInt("LightAttenuation_LinearMethod");
static const int quadraticMethod = Fallback::Map::getInt("LightAttenuation_QuadraticMethod");
static const bool outQuadInLin = Fallback::Map::getBool("LightAttenuation_OutQuadInLin");
if (useConstant)
constantAttenuation = constantValue;
2016-01-06 12:19:15 +00:00
if (useLinear)
{
linearAttenuation = linearMethod == 0 ? linearValue : 0.01f;
2016-01-06 12:19:15 +00:00
float r = radius * linearRadiusMult;
if (r && (linearMethod == 1 || linearMethod == 2))
linearAttenuation = linearValue / std::pow(r, linearMethod);
}
if (useQuadratic && (!outQuadInLin || isExterior))
{
quadraticAttenuation = quadraticMethod == 0 ? quadraticValue : 0.01f;
float r = radius * quadraticRadiusMult;
if (r && (quadraticMethod == 1 || quadraticMethod == 2))
quadraticAttenuation = quadraticValue / std::pow(r, quadraticMethod);
2016-01-06 12:19:15 +00:00
}
light->setConstantAttenuation(constantAttenuation);
2016-01-06 12:19:15 +00:00
light->setLinearAttenuation(linearAttenuation);
light->setQuadraticAttenuation(quadraticAttenuation);
}
osg::ref_ptr<LightSource> addLight(
osg::Group* node, const SceneUtil::LightCommon& esmLight, unsigned int lightMask, bool isExterior)
2016-01-06 12:37:55 +00:00
{
SceneUtil::FindByNameVisitor visitor("AttachLight");
node->accept(visitor);
osg::Group* attachTo = visitor.mFoundNode ? visitor.mFoundNode : node;
osg::ref_ptr<LightSource> lightSource
= createLightSource(esmLight, lightMask, isExterior, osg::Vec4f(0, 0, 0, 1));
attachTo->addChild(lightSource);
2022-06-05 04:01:01 +00:00
CheckEmptyLightVisitor emptyVisitor;
node->accept(emptyVisitor);
lightSource->setEmpty(emptyVisitor.mEmpty);
2021-04-05 17:43:17 +00:00
return lightSource;
}
osg::ref_ptr<LightSource> createLightSource(
const SceneUtil::LightCommon& esmLight, unsigned int lightMask, bool isExterior, const osg::Vec4f& ambient)
{
2016-01-06 12:37:55 +00:00
osg::ref_ptr<SceneUtil::LightSource> lightSource(new SceneUtil::LightSource);
osg::ref_ptr<osg::Light> light(new osg::Light);
lightSource->setNodeMask(lightMask);
2016-01-06 12:37:55 +00:00
float radius = esmLight.mRadius;
2016-01-06 12:37:55 +00:00
lightSource->setRadius(radius);
configureLight(light, radius, isExterior);
2016-01-06 12:37:55 +00:00
osg::Vec4f diffuse = esmLight.mColor;
if (esmLight.mNegative)
2016-01-06 12:37:55 +00:00
{
diffuse *= -1;
diffuse.a() = 1;
}
light->setDiffuse(diffuse);
light->setAmbient(ambient);
2016-01-06 12:37:55 +00:00
light->setSpecular(osg::Vec4f(0, 0, 0, 0));
lightSource->setLight(light);
osg::ref_ptr<SceneUtil::LightController> ctrl(new SceneUtil::LightController);
ctrl->setDiffuse(light->getDiffuse());
if (esmLight.mFlicker)
2016-01-06 12:37:55 +00:00
ctrl->setType(SceneUtil::LightController::LT_Flicker);
if (esmLight.mFlickerSlow)
2016-01-06 12:37:55 +00:00
ctrl->setType(SceneUtil::LightController::LT_FlickerSlow);
if (esmLight.mPulse)
2016-01-06 12:37:55 +00:00
ctrl->setType(SceneUtil::LightController::LT_Pulse);
if (esmLight.mPulseSlow)
2016-01-06 12:37:55 +00:00
ctrl->setType(SceneUtil::LightController::LT_PulseSlow);
lightSource->addUpdateCallback(ctrl);
return lightSource;
2016-01-06 12:37:55 +00:00
}
2016-01-06 12:19:15 +00:00
}