2015-04-10 21:16:17 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_SCENEUTIL_LIGHTMANAGER_H
|
|
|
|
#define OPENMW_COMPONENTS_SCENEUTIL_LIGHTMANAGER_H
|
|
|
|
|
2016-10-08 01:49:50 +00:00
|
|
|
#include <set>
|
2021-03-02 07:30:54 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <unordered_set>
|
2016-10-08 01:49:50 +00:00
|
|
|
|
2015-04-10 21:16:17 +00:00
|
|
|
#include <osg/Light>
|
|
|
|
|
|
|
|
#include <osg/Group>
|
|
|
|
#include <osg/NodeVisitor>
|
2016-01-25 20:03:33 +00:00
|
|
|
#include <osg/observer_ptr>
|
2021-02-21 18:38:15 +00:00
|
|
|
|
|
|
|
#include <components/shader/shadermanager.hpp>
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2017-03-03 16:42:38 +00:00
|
|
|
namespace osgUtil
|
|
|
|
{
|
|
|
|
class CullVisitor;
|
|
|
|
}
|
|
|
|
|
2021-03-02 07:30:54 +00:00
|
|
|
namespace osg
|
|
|
|
{
|
|
|
|
class UniformBufferBinding;
|
|
|
|
class UniformBufferObject;
|
|
|
|
}
|
|
|
|
|
2015-04-10 21:16:17 +00:00
|
|
|
namespace SceneUtil
|
|
|
|
{
|
2021-02-21 18:38:15 +00:00
|
|
|
class SunlightBuffer;
|
2021-03-02 07:30:54 +00:00
|
|
|
class PointLightBuffer;
|
2021-02-21 18:38:15 +00:00
|
|
|
|
|
|
|
// Used to override sun. Rarely useful but necassary for local map.
|
|
|
|
class SunlightStateAttribute : public osg::StateAttribute
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
SunlightStateAttribute();
|
|
|
|
SunlightStateAttribute(const SunlightStateAttribute& copy,const osg::CopyOp& copyop=osg::CopyOp::SHALLOW_COPY);
|
|
|
|
|
|
|
|
int compare(const StateAttribute &sa) const override;
|
|
|
|
|
|
|
|
META_StateAttribute(NifOsg, SunlightStateAttribute, osg::StateAttribute::LIGHT)
|
|
|
|
|
|
|
|
void setFromLight(const osg::Light* light);
|
|
|
|
|
|
|
|
void setStateSet(osg::StateSet* stateset, int mode=osg::StateAttribute::ON);
|
|
|
|
|
|
|
|
private:
|
|
|
|
osg::ref_ptr<SunlightBuffer> mBuffer;
|
|
|
|
osg::ref_ptr<osg::UniformBufferBinding> mUbb;
|
|
|
|
};
|
2015-04-10 21:16:17 +00:00
|
|
|
|
|
|
|
/// LightSource managed by a LightManager.
|
2015-12-03 23:55:32 +00:00
|
|
|
/// @par Typically used for point lights. Spot lights are not supported yet. Directional lights affect the whole scene
|
|
|
|
/// so do not need to be managed by a LightManager - so for directional lights use a plain osg::LightSource instead.
|
|
|
|
/// @note LightSources must be decorated by a LightManager node in order to have an effect. Typical use would
|
|
|
|
/// be one LightManager as the root of the scene graph.
|
|
|
|
/// @note One needs to attach LightListCallback's to the scene to have objects receive lighting from LightSources.
|
|
|
|
/// See the documentation of LightListCallback for more information.
|
|
|
|
/// @note The position of the contained osg::Light is automatically updated based on the LightSource's world position.
|
2015-04-10 21:16:17 +00:00
|
|
|
class LightSource : public osg::Node
|
|
|
|
{
|
2015-12-03 23:55:32 +00:00
|
|
|
// double buffered osg::Light's, since one of them may be in use by the draw thread at any given time
|
2015-12-03 23:06:22 +00:00
|
|
|
osg::ref_ptr<osg::Light> mLight[2];
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2015-12-03 23:55:32 +00:00
|
|
|
// LightSource will affect objects within this radius
|
2015-04-10 21:16:17 +00:00
|
|
|
float mRadius;
|
|
|
|
|
2015-06-15 16:09:01 +00:00
|
|
|
int mId;
|
|
|
|
|
2021-03-02 07:30:54 +00:00
|
|
|
float mBrightness[2];
|
|
|
|
|
2015-04-10 21:16:17 +00:00
|
|
|
public:
|
|
|
|
|
2017-02-01 02:00:33 +00:00
|
|
|
META_Node(SceneUtil, LightSource)
|
2015-04-10 21:16:17 +00:00
|
|
|
|
|
|
|
LightSource();
|
|
|
|
|
2015-06-15 16:09:01 +00:00
|
|
|
LightSource(const LightSource& copy, const osg::CopyOp& copyop);
|
2015-04-10 21:16:17 +00:00
|
|
|
|
|
|
|
float getRadius() const
|
|
|
|
{
|
|
|
|
return mRadius;
|
|
|
|
}
|
|
|
|
|
2015-12-03 23:55:32 +00:00
|
|
|
/// The LightSource will affect objects within this radius.
|
2015-04-10 21:16:17 +00:00
|
|
|
void setRadius(float radius)
|
|
|
|
{
|
|
|
|
mRadius = radius;
|
|
|
|
}
|
|
|
|
|
2021-03-02 07:30:54 +00:00
|
|
|
float getBrightness(size_t frame)
|
|
|
|
{
|
|
|
|
return mBrightness[frame % 2];
|
|
|
|
}
|
|
|
|
|
|
|
|
void setBrightness(size_t frame, float brightness)
|
|
|
|
{
|
|
|
|
mBrightness[frame % 2] = brightness;
|
|
|
|
}
|
|
|
|
|
2015-12-03 23:06:22 +00:00
|
|
|
/// Get the osg::Light safe for modification in the given frame.
|
2015-12-03 23:55:32 +00:00
|
|
|
/// @par May be used externally to animate the light's color/attenuation properties,
|
|
|
|
/// and is used internally to synchronize the light's position with the position of the LightSource.
|
2021-02-21 18:38:15 +00:00
|
|
|
osg::Light* getLight(size_t frame)
|
2015-04-10 21:16:17 +00:00
|
|
|
{
|
2015-12-03 23:06:22 +00:00
|
|
|
return mLight[frame % 2];
|
2015-04-10 21:16:17 +00:00
|
|
|
}
|
|
|
|
|
2015-12-03 23:06:22 +00:00
|
|
|
/// @warning It is recommended not to replace an existing osg::Light, because there might still be
|
|
|
|
/// references to it in the light StateSet cache that are associated with this LightSource's ID.
|
|
|
|
/// These references will stay valid due to ref_ptr but will point to the old object.
|
|
|
|
/// @warning Do not modify the \a light after you've called this function.
|
2015-04-10 21:16:17 +00:00
|
|
|
void setLight(osg::Light* light)
|
|
|
|
{
|
2015-12-03 23:06:22 +00:00
|
|
|
mLight[0] = light;
|
2017-02-09 03:50:51 +00:00
|
|
|
mLight[1] = new osg::Light(*light);
|
2015-04-10 21:16:17 +00:00
|
|
|
}
|
2015-06-15 16:09:01 +00:00
|
|
|
|
2015-12-03 23:06:22 +00:00
|
|
|
/// Get the unique ID for this light source.
|
|
|
|
int getId() const
|
2015-06-15 16:09:01 +00:00
|
|
|
{
|
|
|
|
return mId;
|
|
|
|
}
|
2015-04-10 21:16:17 +00:00
|
|
|
};
|
|
|
|
|
2015-12-03 23:55:32 +00:00
|
|
|
/// @brief Decorator node implementing the rendering of any number of LightSources that can be anywhere in the subgraph.
|
2015-04-10 21:16:17 +00:00
|
|
|
class LightManager : public osg::Group
|
|
|
|
{
|
|
|
|
public:
|
2021-02-21 18:38:15 +00:00
|
|
|
struct LightSourceTransform
|
|
|
|
{
|
|
|
|
LightSource* mLightSource;
|
|
|
|
osg::Matrixf mWorldMatrix;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LightSourceViewBound
|
|
|
|
{
|
|
|
|
LightSource* mLightSource;
|
|
|
|
osg::BoundingSphere mViewBound;
|
|
|
|
};
|
|
|
|
|
2021-03-02 07:30:54 +00:00
|
|
|
using LightList = std::vector<const LightSourceViewBound*>;
|
2021-02-21 18:38:15 +00:00
|
|
|
|
|
|
|
static bool queryNonFFPLightingSupport();
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2017-02-01 02:00:33 +00:00
|
|
|
META_Node(SceneUtil, LightManager)
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
LightManager(bool ffp = true);
|
2015-04-10 21:16:17 +00:00
|
|
|
|
|
|
|
LightManager(const LightManager& copy, const osg::CopyOp& copyop);
|
|
|
|
|
2015-11-10 16:19:51 +00:00
|
|
|
/// @param mask This mask is compared with the current Camera's cull mask to determine if lighting is desired.
|
|
|
|
/// By default, it's ~0u i.e. always on.
|
|
|
|
/// If you have some views that do not require lighting, then set the Camera's cull mask to not include
|
|
|
|
/// the lightingMask for a much faster cull and rendering.
|
2021-02-21 18:38:15 +00:00
|
|
|
void setLightingMask (size_t mask);
|
|
|
|
size_t getLightingMask() const;
|
2015-11-10 16:19:51 +00:00
|
|
|
|
2015-12-03 23:55:32 +00:00
|
|
|
/// Set the first light index that should be used by this manager, typically the number of directional lights in the scene.
|
|
|
|
void setStartLight(int start);
|
|
|
|
int getStartLight() const;
|
|
|
|
|
|
|
|
/// Internal use only, called automatically by the LightManager's UpdateCallback
|
2015-04-10 21:16:17 +00:00
|
|
|
void update();
|
|
|
|
|
2015-12-03 23:55:32 +00:00
|
|
|
/// Internal use only, called automatically by the LightSource's UpdateCallback
|
2021-02-21 18:38:15 +00:00
|
|
|
void addLight(LightSource* lightSource, const osg::Matrixf& worldMat, size_t frameNum);
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
const std::vector<LightSourceViewBound>& getLightsInViewSpace(osg::Camera* camera, const osg::RefMatrix* viewMatrix, size_t frameNum);
|
2015-10-22 23:58:22 +00:00
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
osg::ref_ptr<osg::StateSet> getLightListStateSet(const LightList& lightList, size_t frameNum);
|
2015-10-22 23:58:22 +00:00
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
void setSunlight(osg::ref_ptr<osg::Light> sun);
|
|
|
|
osg::ref_ptr<osg::Light> getSunlight();
|
|
|
|
|
|
|
|
osg::ref_ptr<SunlightBuffer> getSunBuffer();
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
bool usingFFP() const;
|
|
|
|
|
|
|
|
int getMaxLights() const;
|
|
|
|
int getMaxLightsInScene() const;
|
|
|
|
|
|
|
|
Shader::ShaderManager::DefineMap getLightDefines() const;
|
2015-04-10 21:16:17 +00:00
|
|
|
|
|
|
|
private:
|
2021-03-02 07:30:54 +00:00
|
|
|
|
|
|
|
friend class LightManagerStateAttribute;
|
|
|
|
|
|
|
|
void updateGPUPointLight(int index, LightSource* lightSource, size_t frameNum);
|
|
|
|
|
2015-04-10 21:16:17 +00:00
|
|
|
// Lights collected from the scene graph. Only valid during the cull traversal.
|
|
|
|
std::vector<LightSourceTransform> mLights;
|
|
|
|
|
2015-10-22 23:58:22 +00:00
|
|
|
typedef std::vector<LightSourceViewBound> LightSourceViewBoundCollection;
|
|
|
|
std::map<osg::observer_ptr<osg::Camera>, LightSourceViewBoundCollection> mLightsInViewSpace;
|
2015-04-10 21:16:17 +00:00
|
|
|
|
|
|
|
// < Light list hash , StateSet >
|
|
|
|
typedef std::map<size_t, osg::ref_ptr<osg::StateSet> > LightStateSetMap;
|
2015-12-03 23:06:22 +00:00
|
|
|
LightStateSetMap mStateSetCache[2];
|
2015-04-10 21:16:17 +00:00
|
|
|
|
2019-02-20 13:37:00 +00:00
|
|
|
std::vector<osg::ref_ptr<osg::StateAttribute>> mDummies;
|
|
|
|
|
2015-04-12 13:34:50 +00:00
|
|
|
int mStartLight;
|
2015-11-10 16:19:51 +00:00
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
size_t mLightingMask;
|
|
|
|
|
|
|
|
osg::ref_ptr<osg::Light> mSun;
|
|
|
|
osg::ref_ptr<SunlightBuffer> mSunBuffer;
|
|
|
|
|
2021-03-02 07:30:54 +00:00
|
|
|
struct PointLightProxyData
|
|
|
|
{
|
|
|
|
osg::Vec4 mPosition;
|
|
|
|
float mBrightness;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<PointLightProxyData> mPointLightProxyData;
|
|
|
|
osg::ref_ptr<PointLightBuffer> mPointBuffer;
|
|
|
|
|
|
|
|
// < Light ID , Buffer Index >
|
|
|
|
using LightDataMap = std::unordered_map<int, int>;
|
|
|
|
LightDataMap mLightData;
|
|
|
|
|
|
|
|
bool mIndexNeedsRecompiling;
|
|
|
|
|
2021-02-21 18:38:15 +00:00
|
|
|
bool mFFP;
|
|
|
|
|
|
|
|
static constexpr int mFFPMaxLights = 8;
|
2015-04-10 21:16:17 +00:00
|
|
|
};
|
|
|
|
|
2015-12-03 23:55:32 +00:00
|
|
|
/// To receive lighting, objects must be decorated by a LightListCallback. Light list callbacks must be added via
|
|
|
|
/// node->addCullCallback(new LightListCallback). Once a light list callback is added to a node, that node and all
|
|
|
|
/// its child nodes can receive lighting.
|
|
|
|
/// @par The placement of these LightListCallbacks affects the granularity of light lists. Having too fine grained
|
|
|
|
/// light lists can result in degraded performance. Too coarse grained light lists can result in lights no longer
|
|
|
|
/// rendering when the size of a light list exceeds the OpenGL limit on the number of concurrent lights (8). A good
|
|
|
|
/// starting point is to attach a LightListCallback to each game object's base node.
|
2015-11-03 23:19:15 +00:00
|
|
|
/// @note Not thread safe for CullThreadPerCamera threading mode.
|
2017-03-03 16:42:38 +00:00
|
|
|
/// @note Due to lack of OSG support, the callback does not work on Drawables.
|
2015-04-12 16:02:29 +00:00
|
|
|
class LightListCallback : public osg::NodeCallback
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
LightListCallback()
|
2018-10-09 06:21:12 +00:00
|
|
|
: mLightManager(nullptr)
|
2015-11-03 23:19:15 +00:00
|
|
|
, mLastFrameNumber(0)
|
2015-04-12 16:02:29 +00:00
|
|
|
{}
|
2015-11-09 17:57:17 +00:00
|
|
|
LightListCallback(const LightListCallback& copy, const osg::CopyOp& copyop)
|
2015-11-04 19:34:50 +00:00
|
|
|
: osg::Object(copy, copyop), osg::NodeCallback(copy, copyop)
|
|
|
|
, mLightManager(copy.mLightManager)
|
|
|
|
, mLastFrameNumber(0)
|
2016-10-08 01:49:50 +00:00
|
|
|
, mIgnoredLightSources(copy.mIgnoredLightSources)
|
2015-04-12 16:02:29 +00:00
|
|
|
{}
|
|
|
|
|
2015-12-06 14:27:43 +00:00
|
|
|
META_Object(SceneUtil, LightListCallback)
|
2015-04-12 16:02:29 +00:00
|
|
|
|
2020-10-16 18:18:54 +00:00
|
|
|
void operator()(osg::Node* node, osg::NodeVisitor* nv) override;
|
2015-04-12 16:02:29 +00:00
|
|
|
|
2017-03-03 16:42:38 +00:00
|
|
|
bool pushLightState(osg::Node* node, osgUtil::CullVisitor* nv);
|
|
|
|
|
2016-10-08 01:49:50 +00:00
|
|
|
std::set<SceneUtil::LightSource*>& getIgnoredLightSources() { return mIgnoredLightSources; }
|
|
|
|
|
2015-04-12 16:02:29 +00:00
|
|
|
private:
|
|
|
|
LightManager* mLightManager;
|
2021-02-21 18:38:15 +00:00
|
|
|
size_t mLastFrameNumber;
|
2015-11-03 23:19:15 +00:00
|
|
|
LightManager::LightList mLightList;
|
2016-10-08 01:49:50 +00:00
|
|
|
std::set<SceneUtil::LightSource*> mIgnoredLightSources;
|
2015-04-12 16:02:29 +00:00
|
|
|
};
|
|
|
|
|
2015-04-10 21:16:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|