2015-03-26 17:02:51 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_RESOURCE_SCENEMANAGER_H
|
|
|
|
#define OPENMW_COMPONENTS_RESOURCE_SCENEMANAGER_H
|
|
|
|
|
2016-02-16 17:18:48 +00:00
|
|
|
#include <memory>
|
2020-06-25 19:46:07 +00:00
|
|
|
#include <mutex>
|
2015-03-26 17:02:51 +00:00
|
|
|
#include <string>
|
|
|
|
|
2016-02-05 17:51:51 +00:00
|
|
|
#include <osg/Texture>
|
2015-03-26 17:02:51 +00:00
|
|
|
#include <osg/ref_ptr>
|
|
|
|
|
2016-02-06 15:57:54 +00:00
|
|
|
#include "resourcemanager.hpp"
|
|
|
|
|
2021-03-15 04:42:34 +00:00
|
|
|
#include <components/sceneutil/lightmanager.hpp>
|
2022-06-19 11:28:33 +00:00
|
|
|
#include <filesystem>
|
2021-03-15 04:42:34 +00:00
|
|
|
|
2022-10-09 10:39:43 +00:00
|
|
|
namespace VFS
|
|
|
|
{
|
|
|
|
class Manager;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace osg
|
|
|
|
{
|
|
|
|
class Group;
|
|
|
|
class Node;
|
|
|
|
class Program;
|
|
|
|
class State;
|
|
|
|
class Stats;
|
|
|
|
}
|
|
|
|
|
2015-03-28 01:20:20 +00:00
|
|
|
namespace Resource
|
|
|
|
{
|
2016-02-05 22:03:53 +00:00
|
|
|
class ImageManager;
|
2015-12-01 22:04:02 +00:00
|
|
|
class NifFileManager;
|
2017-02-23 13:49:12 +00:00
|
|
|
class SharedStateManager;
|
2015-03-28 01:20:20 +00:00
|
|
|
}
|
|
|
|
|
2015-05-26 18:20:18 +00:00
|
|
|
namespace osgUtil
|
|
|
|
{
|
|
|
|
class IncrementalCompileOperation;
|
|
|
|
}
|
|
|
|
|
2016-02-16 17:18:48 +00:00
|
|
|
namespace Shader
|
|
|
|
{
|
|
|
|
class ShaderManager;
|
2017-10-15 15:24:23 +00:00
|
|
|
class ShaderVisitor;
|
2016-02-16 17:18:48 +00:00
|
|
|
}
|
|
|
|
|
2015-03-26 17:02:51 +00:00
|
|
|
namespace Resource
|
|
|
|
{
|
2021-01-13 09:33:46 +00:00
|
|
|
class TemplateRef : public osg::Object
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
TemplateRef(const Object* object)
|
|
|
|
: mObject(object)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
TemplateRef() {}
|
|
|
|
TemplateRef(const TemplateRef& copy, const osg::CopyOp&)
|
|
|
|
: mObject(copy.mObject)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
META_Object(Resource, TemplateRef)
|
|
|
|
|
|
|
|
private:
|
|
|
|
osg::ref_ptr<const Object> mObject;
|
|
|
|
};
|
|
|
|
|
|
|
|
class TemplateMultiRef : public osg::Object
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
TemplateMultiRef() {}
|
|
|
|
TemplateMultiRef(const TemplateMultiRef& copy, const osg::CopyOp&)
|
|
|
|
: mObjects(copy.mObjects)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
void addRef(const osg::Node* node);
|
|
|
|
|
|
|
|
META_Object(Resource, TemplateMultiRef)
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::vector<osg::ref_ptr<const Object>> mObjects;
|
|
|
|
};
|
2015-03-26 17:02:51 +00:00
|
|
|
|
2016-02-05 23:15:12 +00:00
|
|
|
/// @brief Handles loading and caching of scenes, e.g. .nif files or .osg files
|
|
|
|
/// @note Some methods of the scene manager can be used from any thread, see the methods documentation for more
|
|
|
|
/// details.
|
2016-02-06 15:57:54 +00:00
|
|
|
class SceneManager : public ResourceManager
|
2015-03-26 17:02:51 +00:00
|
|
|
{
|
|
|
|
public:
|
2016-02-05 22:10:27 +00:00
|
|
|
SceneManager(
|
|
|
|
const VFS::Manager* vfs, Resource::ImageManager* imageManager, Resource::NifFileManager* nifFileManager);
|
2015-05-08 15:52:35 +00:00
|
|
|
~SceneManager();
|
2015-03-26 17:02:51 +00:00
|
|
|
|
2016-02-19 00:30:15 +00:00
|
|
|
Shader::ShaderManager& getShaderManager();
|
|
|
|
|
2021-02-19 19:59:48 +00:00
|
|
|
/// Re-create shaders for this node, need to call this if alpha testing, texture stages or vertex color mode
|
|
|
|
/// have changed.
|
2022-06-30 01:15:12 +00:00
|
|
|
void recreateShaders(osg::ref_ptr<osg::Node> node, const std::string& shaderPrefix = "objects",
|
|
|
|
bool forceShadersForNode = false, const osg::Program* programTemplate = nullptr);
|
2016-02-18 21:48:53 +00:00
|
|
|
|
2021-02-19 19:59:48 +00:00
|
|
|
/// Applying shaders to a node may replace some fixed-function state.
|
|
|
|
/// This restores it.
|
|
|
|
/// When editing such state, it should be reinstated before the edits, and shaders should be recreated
|
|
|
|
/// afterwards.
|
|
|
|
void reinstateRemovedState(osg::ref_ptr<osg::Node> node);
|
|
|
|
|
2016-02-18 16:08:18 +00:00
|
|
|
/// @see ShaderVisitor::setForceShaders
|
|
|
|
void setForceShaders(bool force);
|
2016-02-19 00:30:15 +00:00
|
|
|
bool getForceShaders() const;
|
2016-02-18 16:08:18 +00:00
|
|
|
|
|
|
|
void setClampLighting(bool clamp);
|
2016-02-19 00:30:15 +00:00
|
|
|
bool getClampLighting() const;
|
2016-02-18 16:08:18 +00:00
|
|
|
|
2016-02-20 16:57:19 +00:00
|
|
|
/// @see ShaderVisitor::setAutoUseNormalMaps
|
|
|
|
void setAutoUseNormalMaps(bool use);
|
|
|
|
|
|
|
|
/// @see ShaderVisitor::setNormalMapPattern
|
|
|
|
void setNormalMapPattern(const std::string& pattern);
|
|
|
|
|
2016-03-22 20:00:31 +00:00
|
|
|
/// @see ShaderVisitor::setNormalHeightMapPattern
|
|
|
|
void setNormalHeightMapPattern(const std::string& pattern);
|
|
|
|
|
2016-02-20 18:02:11 +00:00
|
|
|
void setAutoUseSpecularMaps(bool use);
|
|
|
|
|
|
|
|
void setSpecularMapPattern(const std::string& pattern);
|
|
|
|
|
2020-11-09 10:53:58 +00:00
|
|
|
void setApplyLightingToEnvMaps(bool apply);
|
|
|
|
|
2021-04-16 18:55:40 +00:00
|
|
|
void setSupportedLightingMethods(const SceneUtil::LightManager::SupportedMethods& supported);
|
|
|
|
bool isSupportedLightingMethod(SceneUtil::LightingMethod method) const;
|
|
|
|
|
2022-06-21 15:55:06 +00:00
|
|
|
void setOpaqueDepthTex(osg::ref_ptr<osg::Texture> texturePing, osg::ref_ptr<osg::Texture> texturePong);
|
2021-10-20 16:42:18 +00:00
|
|
|
|
2022-06-30 01:15:12 +00:00
|
|
|
osg::ref_ptr<osg::Texture> getOpaqueDepthTex(size_t frame);
|
|
|
|
|
2021-09-29 13:40:37 +00:00
|
|
|
enum class UBOBinding
|
|
|
|
{
|
|
|
|
// If we add more UBO's, we should probably assign their bindings dynamically according to the current count
|
|
|
|
// of UBO's in the programTemplate
|
2022-05-14 01:58:00 +00:00
|
|
|
LightBuffer,
|
|
|
|
PostProcessor
|
2021-09-29 13:40:37 +00:00
|
|
|
};
|
2021-03-15 04:42:34 +00:00
|
|
|
void setLightingMethod(SceneUtil::LightingMethod method);
|
|
|
|
SceneUtil::LightingMethod getLightingMethod() const;
|
2022-09-22 18:26:05 +00:00
|
|
|
|
2020-12-26 22:45:53 +00:00
|
|
|
void setConvertAlphaTestToAlphaToCoverage(bool convert);
|
2022-12-09 00:22:08 +00:00
|
|
|
void setAdjustCoverageForAlphaTest(bool adjustCoverage);
|
2021-02-21 18:38:15 +00:00
|
|
|
|
2022-06-19 11:28:33 +00:00
|
|
|
void setShaderPath(const std::filesystem::path& path);
|
2016-02-16 17:18:48 +00:00
|
|
|
|
2017-02-14 23:55:35 +00:00
|
|
|
/// Check if a given scene is loaded and if so, update its usage timestamp to prevent it from being unloaded
|
|
|
|
bool checkLoaded(const std::string& name, double referenceTime);
|
|
|
|
|
2015-03-26 17:02:51 +00:00
|
|
|
/// Get a read-only copy of this scene "template"
|
2015-07-27 20:59:20 +00:00
|
|
|
/// @note If the given filename does not exist or fails to load, an error marker mesh will be used instead.
|
|
|
|
/// If even the error marker mesh can not be found, an exception is thrown.
|
2016-02-05 23:15:12 +00:00
|
|
|
/// @note Thread safe.
|
2019-06-13 13:37:00 +00:00
|
|
|
osg::ref_ptr<const osg::Node> getTemplate(const std::string& name, bool compile = true);
|
2015-03-26 17:02:51 +00:00
|
|
|
|
2021-10-23 08:31:46 +00:00
|
|
|
/// Clone osg::Node safely.
|
|
|
|
/// @note Thread safe.
|
|
|
|
static osg::ref_ptr<osg::Node> cloneNode(const osg::Node* base);
|
2017-03-01 02:00:49 +00:00
|
|
|
|
2021-05-18 12:03:16 +00:00
|
|
|
void shareState(osg::ref_ptr<osg::Node> node);
|
2021-10-23 08:31:46 +00:00
|
|
|
|
|
|
|
/// Clone osg::Node and adjust it according to SceneManager's settings.
|
|
|
|
/// @note Thread safe.
|
|
|
|
osg::ref_ptr<osg::Node> getInstance(const osg::Node* base);
|
|
|
|
|
|
|
|
/// Instance the given scene template.
|
2015-07-27 20:59:20 +00:00
|
|
|
/// @see getTemplate
|
2016-02-05 23:15:12 +00:00
|
|
|
/// @note Thread safe.
|
2016-02-09 17:33:02 +00:00
|
|
|
osg::ref_ptr<osg::Node> getInstance(const std::string& name);
|
2015-03-27 23:30:49 +00:00
|
|
|
|
2021-10-23 08:31:46 +00:00
|
|
|
/// Instance the given scene template and immediately attach it to a parent node
|
2015-07-27 20:59:20 +00:00
|
|
|
/// @see getTemplate
|
2016-02-05 23:15:12 +00:00
|
|
|
/// @note Not thread safe, unless parentNode is not part of the main scene graph yet.
|
2016-02-09 17:33:02 +00:00
|
|
|
osg::ref_ptr<osg::Node> getInstance(const std::string& name, osg::Group* parentNode);
|
2015-03-27 23:30:49 +00:00
|
|
|
|
|
|
|
/// Attach the given scene instance to the given parent node
|
|
|
|
/// @note You should have the parentNode in its intended position before calling this method,
|
|
|
|
/// so that world space particles of the \a instance get transformed correctly.
|
|
|
|
/// @note Assumes the given instance was not attached to any parents before.
|
2016-02-05 23:15:12 +00:00
|
|
|
/// @note Not thread safe, unless parentNode is not part of the main scene graph yet.
|
2015-03-27 23:30:49 +00:00
|
|
|
void attachTo(osg::Node* instance, osg::Group* parentNode) const;
|
2015-03-26 17:02:51 +00:00
|
|
|
|
2015-04-02 15:34:44 +00:00
|
|
|
/// Manually release created OpenGL objects for the given graphics context. This may be required
|
|
|
|
/// in cases where multiple contexts are used over the lifetime of the application.
|
2017-08-26 19:28:23 +00:00
|
|
|
void releaseGLObjects(osg::State* state) override;
|
2015-03-28 19:15:17 +00:00
|
|
|
|
2015-05-26 18:20:18 +00:00
|
|
|
/// Set up an IncrementalCompileOperation for background compiling of loaded scenes.
|
|
|
|
void setIncrementalCompileOperation(osgUtil::IncrementalCompileOperation* ico);
|
|
|
|
|
2017-03-06 19:41:02 +00:00
|
|
|
osgUtil::IncrementalCompileOperation* getIncrementalCompileOperation();
|
|
|
|
|
2016-02-05 22:10:27 +00:00
|
|
|
Resource::ImageManager* getImageManager();
|
2015-04-14 13:55:56 +00:00
|
|
|
|
2015-11-10 16:00:33 +00:00
|
|
|
/// @param mask The node mask to apply to loaded particle system nodes.
|
|
|
|
void setParticleSystemMask(unsigned int mask);
|
|
|
|
|
2016-02-14 22:14:52 +00:00
|
|
|
/// @warning It is unsafe to call this method while the draw thread is using textures! call
|
|
|
|
/// Viewer::stopThreading first.
|
2016-02-05 17:51:51 +00:00
|
|
|
void setFilterSettings(
|
|
|
|
const std::string& magfilter, const std::string& minfilter, const std::string& mipmap, int maxAnisotropy);
|
|
|
|
|
2016-02-05 23:15:12 +00:00
|
|
|
/// Apply filter settings to the given texture. Note, when loading an object through this scene manager (i.e.
|
2016-02-05 17:51:51 +00:00
|
|
|
/// calling getTemplate or createInstance) the filter settings are applied automatically. This method is
|
|
|
|
/// provided for textures that were created outside of the SceneManager.
|
|
|
|
void applyFilterSettings(osg::Texture* tex);
|
|
|
|
|
|
|
|
/// Keep a copy of the texture data around in system memory? This is needed when using multiple graphics
|
|
|
|
/// contexts, otherwise should be disabled to reduce memory usage.
|
|
|
|
void setUnRefImageDataAfterApply(bool unref);
|
|
|
|
|
2016-02-09 17:33:02 +00:00
|
|
|
/// @see ResourceManager::updateCache
|
2017-09-18 08:51:11 +00:00
|
|
|
void updateCache(double referenceTime) override;
|
2016-02-09 17:33:02 +00:00
|
|
|
|
2017-09-18 08:51:11 +00:00
|
|
|
void clearCache() override;
|
2017-08-20 23:55:16 +00:00
|
|
|
|
2017-09-18 08:51:11 +00:00
|
|
|
void reportStats(unsigned int frameNumber, osg::Stats* stats) const override;
|
2017-02-22 01:18:18 +00:00
|
|
|
|
2022-05-14 01:58:00 +00:00
|
|
|
void setSupportsNormalsRT(bool supports) { mSupportsNormalsRT = supports; }
|
|
|
|
bool getSupportsNormalsRT() const { return mSupportsNormalsRT; }
|
|
|
|
|
2022-07-02 16:42:15 +00:00
|
|
|
void setSoftParticles(bool enabled) { mSoftParticles = enabled; }
|
|
|
|
bool getSoftParticles() const { return mSoftParticles; }
|
|
|
|
|
2015-03-26 17:02:51 +00:00
|
|
|
private:
|
2021-10-05 12:37:08 +00:00
|
|
|
Shader::ShaderVisitor* createShaderVisitor(const std::string& shaderPrefix = "objects");
|
2017-10-15 15:24:23 +00:00
|
|
|
|
2017-04-28 15:30:26 +00:00
|
|
|
std::unique_ptr<Shader::ShaderManager> mShaderManager;
|
2016-02-18 16:08:18 +00:00
|
|
|
bool mForceShaders;
|
|
|
|
bool mClampLighting;
|
2016-02-20 16:57:19 +00:00
|
|
|
bool mAutoUseNormalMaps;
|
|
|
|
std::string mNormalMapPattern;
|
2016-03-22 20:00:31 +00:00
|
|
|
std::string mNormalHeightMapPattern;
|
2016-02-20 18:02:11 +00:00
|
|
|
bool mAutoUseSpecularMaps;
|
|
|
|
std::string mSpecularMapPattern;
|
2020-11-09 10:53:58 +00:00
|
|
|
bool mApplyLightingToEnvMaps;
|
2021-03-15 04:42:34 +00:00
|
|
|
SceneUtil::LightingMethod mLightingMethod;
|
2021-04-16 18:55:40 +00:00
|
|
|
SceneUtil::LightManager::SupportedMethods mSupportedLightingMethods;
|
2020-12-26 22:45:53 +00:00
|
|
|
bool mConvertAlphaTestToAlphaToCoverage;
|
2022-12-09 00:22:08 +00:00
|
|
|
bool mAdjustCoverageForAlphaTest;
|
2022-05-14 01:58:00 +00:00
|
|
|
bool mSupportsNormalsRT;
|
2022-06-21 15:55:06 +00:00
|
|
|
std::array<osg::ref_ptr<osg::Texture>, 2> mOpaqueDepthTex;
|
2022-07-02 16:42:15 +00:00
|
|
|
bool mSoftParticles = false;
|
2016-02-18 16:08:18 +00:00
|
|
|
|
2017-02-23 13:49:12 +00:00
|
|
|
osg::ref_ptr<Resource::SharedStateManager> mSharedStateManager;
|
2020-06-25 19:46:07 +00:00
|
|
|
mutable std::mutex mSharedStateMutex;
|
2016-02-07 15:37:35 +00:00
|
|
|
|
2016-02-05 22:10:27 +00:00
|
|
|
Resource::ImageManager* mImageManager;
|
2015-12-01 22:04:02 +00:00
|
|
|
Resource::NifFileManager* mNifFileManager;
|
2015-03-26 17:02:51 +00:00
|
|
|
|
2016-02-05 17:51:51 +00:00
|
|
|
osg::Texture::FilterMode mMinFilter;
|
|
|
|
osg::Texture::FilterMode mMagFilter;
|
|
|
|
int mMaxAnisotropy;
|
|
|
|
bool mUnRefImageDataAfterApply;
|
|
|
|
|
2015-05-26 18:20:18 +00:00
|
|
|
osg::ref_ptr<osgUtil::IncrementalCompileOperation> mIncrementalCompileOperation;
|
|
|
|
|
2015-11-10 16:00:33 +00:00
|
|
|
unsigned int mParticleSystemMask;
|
|
|
|
|
2015-05-08 15:52:35 +00:00
|
|
|
SceneManager(const SceneManager&);
|
|
|
|
void operator=(const SceneManager&);
|
2015-03-26 17:02:51 +00:00
|
|
|
};
|
|
|
|
|
2020-11-28 13:03:10 +00:00
|
|
|
std::string getFileExtension(const std::string& file);
|
2015-03-26 17:02:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|