1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-23 22:23:52 +00:00
openmw-tes3mp/apps/openmw/mwrender/renderingmanager.cpp

1584 lines
57 KiB
C++
Raw Normal View History

2011-10-20 19:02:19 +00:00
#include "renderingmanager.hpp"
2015-08-21 10:00:08 +00:00
#include <limits>
#include <cstdlib>
#include <osg/Light>
#include <osg/LightModel>
#include <osg/Fog>
#include <osg/Material>
2015-06-01 15:02:44 +00:00
#include <osg/PolygonMode>
#include <osg/Group>
#include <osg/UserDataContainer>
#include <osg/ComputeBoundsVisitor>
2017-11-15 14:20:59 +00:00
#include <osg/ShapeDrawable>
#include <osg/TextureCubeMap>
#include <osgUtil/LineSegmentIntersector>
2017-11-07 14:02:01 +00:00
#include <osg/ImageUtils>
#include <osgViewer/Viewer>
2013-07-29 00:32:08 +00:00
2020-04-22 13:57:24 +00:00
#include <components/nifosg/nifloader.hpp>
2018-08-14 19:05:43 +00:00
#include <components/debug/debuglog.hpp>
#include <components/misc/stringops.hpp>
2015-04-22 17:08:56 +00:00
#include <components/resource/resourcesystem.hpp>
2016-02-05 22:03:53 +00:00
#include <components/resource/imagemanager.hpp>
#include <components/resource/scenemanager.hpp>
#include <components/resource/keyframemanager.hpp>
2017-11-15 14:20:59 +00:00
#include <components/shader/shadermanager.hpp>
2015-04-22 17:08:56 +00:00
#include <components/settings/settings.hpp>
#include <components/sceneutil/util.hpp>
2020-03-15 14:31:38 +00:00
#include <components/sceneutil/visitor.hpp>
#include <components/sceneutil/lightmanager.hpp>
#include <components/sceneutil/statesetupdater.hpp>
#include <components/sceneutil/positionattitudetransform.hpp>
#include <components/sceneutil/workqueue.hpp>
#include <components/sceneutil/unrefqueue.hpp>
#include <components/sceneutil/writescene.hpp>
2017-11-08 01:44:49 +00:00
#include <components/sceneutil/shadow.hpp>
2015-06-02 23:18:36 +00:00
#include <components/terrain/terraingrid.hpp>
#include <components/terrain/quadtreeworld.hpp>
2015-06-02 23:18:36 +00:00
#include <components/esm/loadcell.hpp>
2016-01-06 11:46:06 +00:00
#include <components/fallback/fallback.hpp>
#include <components/detournavigator/navigator.hpp>
2015-11-03 01:17:42 +00:00
#include "../mwworld/cellstore.hpp"
#include "../mwworld/class.hpp"
#include "../mwgui/loadingscreen.hpp"
#include "../mwbase/environment.hpp"
#include "../mwbase/windowmanager.hpp"
#include "sky.hpp"
2015-04-19 15:55:56 +00:00
#include "effectmanager.hpp"
2015-05-01 16:21:50 +00:00
#include "npcanimation.hpp"
#include "vismask.hpp"
#include "pathgrid.hpp"
2015-05-21 21:54:39 +00:00
#include "camera.hpp"
2015-06-02 14:35:35 +00:00
#include "water.hpp"
2015-06-02 23:18:36 +00:00
#include "terrainstorage.hpp"
#include "util.hpp"
#include "navmesh.hpp"
#include "actorspaths.hpp"
2019-11-27 22:45:01 +00:00
#include "recastmesh.hpp"
2020-02-14 21:11:19 +00:00
#ifdef USE_OPENXR
2020-03-15 14:31:38 +00:00
#include "../mwvr/vranimation.hpp"
#include "../mwvr/openxrviewer.hpp"
2020-03-15 14:31:38 +00:00
#include "../mwvr/vrenvironment.hpp"
2020-02-14 21:11:19 +00:00
#endif
namespace
{
float DLLandFogStart;
float DLLandFogEnd;
float DLUnderwaterFogStart;
float DLUnderwaterFogEnd;
float DLInteriorFogStart;
float DLInteriorFogEnd;
}
namespace MWRender
{
class StateUpdater : public SceneUtil::StateSetUpdater
{
public:
StateUpdater()
: mFogStart(0.f)
, mFogEnd(0.f)
2015-06-01 15:02:44 +00:00
, mWireframe(false)
{
}
virtual void setDefaults(osg::StateSet *stateset)
{
osg::LightModel* lightModel = new osg::LightModel;
stateset->setAttribute(lightModel, osg::StateAttribute::ON);
osg::Fog* fog = new osg::Fog;
2015-05-26 16:22:21 +00:00
fog->setMode(osg::Fog::LINEAR);
stateset->setAttributeAndModes(fog, osg::StateAttribute::ON);
2015-06-01 15:02:44 +00:00
if (mWireframe)
{
osg::PolygonMode* polygonmode = new osg::PolygonMode;
polygonmode->setMode(osg::PolygonMode::FRONT_AND_BACK, osg::PolygonMode::LINE);
stateset->setAttributeAndModes(polygonmode, osg::StateAttribute::ON);
}
else
stateset->removeAttribute(osg::StateAttribute::POLYGONMODE);
}
virtual void apply(osg::StateSet* stateset, osg::NodeVisitor*)
{
osg::LightModel* lightModel = static_cast<osg::LightModel*>(stateset->getAttribute(osg::StateAttribute::LIGHTMODEL));
lightModel->setAmbientIntensity(mAmbientColor);
osg::Fog* fog = static_cast<osg::Fog*>(stateset->getAttribute(osg::StateAttribute::FOG));
fog->setColor(mFogColor);
fog->setStart(mFogStart);
fog->setEnd(mFogEnd);
}
void setAmbientColor(const osg::Vec4f& col)
{
mAmbientColor = col;
}
void setFogColor(const osg::Vec4f& col)
{
mFogColor = col;
}
void setFogStart(float start)
{
mFogStart = start;
}
void setFogEnd(float end)
{
mFogEnd = end;
}
2015-06-01 15:02:44 +00:00
void setWireframe(bool wireframe)
{
if (mWireframe != wireframe)
{
mWireframe = wireframe;
reset();
}
}
bool getWireframe() const
{
return mWireframe;
}
private:
osg::Vec4f mAmbientColor;
osg::Vec4f mFogColor;
float mFogStart;
float mFogEnd;
2015-06-01 15:02:44 +00:00
bool mWireframe;
};
2016-02-09 00:17:02 +00:00
class PreloadCommonAssetsWorkItem : public SceneUtil::WorkItem
{
public:
PreloadCommonAssetsWorkItem(Resource::ResourceSystem* resourceSystem)
: mResourceSystem(resourceSystem)
{
}
virtual void doWork()
{
try
{
for (std::vector<std::string>::const_iterator it = mModels.begin(); it != mModels.end(); ++it)
mResourceSystem->getSceneManager()->cacheInstance(*it);
for (std::vector<std::string>::const_iterator it = mTextures.begin(); it != mTextures.end(); ++it)
mResourceSystem->getImageManager()->getImage(*it);
for (std::vector<std::string>::const_iterator it = mKeyframes.begin(); it != mKeyframes.end(); ++it)
mResourceSystem->getKeyframeManager()->get(*it);
}
catch (std::exception&)
{
// ignore error (will be shown when these are needed proper)
}
2016-02-09 00:17:02 +00:00
}
std::vector<std::string> mModels;
std::vector<std::string> mTextures;
std::vector<std::string> mKeyframes;
2016-02-09 00:17:02 +00:00
private:
Resource::ResourceSystem* mResourceSystem;
};
RenderingManager::RenderingManager(osgViewer::Viewer* viewer, osg::ref_ptr<osg::Group> rootNode,
Resource::ResourceSystem* resourceSystem, SceneUtil::WorkQueue* workQueue,
const std::string& resourcePath, DetourNavigator::Navigator& navigator)
: mViewer(viewer)
, mRootNode(rootNode)
, mResourceSystem(resourceSystem)
2017-02-14 02:37:45 +00:00
, mWorkQueue(workQueue)
, mUnrefQueue(new SceneUtil::UnrefQueue)
, mNavigator(navigator)
, mLandFogStart(0.f)
, mLandFogEnd(std::numeric_limits<float>::max())
, mUnderwaterFogStart(0.f)
, mUnderwaterFogEnd(std::numeric_limits<float>::max())
, mUnderwaterColor(Fallback::Map::getColour("Water_UnderwaterColor"))
, mUnderwaterWeight(Fallback::Map::getFloat("Water_UnderwaterColorWeight"))
, mUnderwaterIndoorFog(Fallback::Map::getFloat("Water_UnderwaterIndoorFog"))
2015-06-11 21:16:05 +00:00
, mNightEyeFactor(0.f)
, mDistantFog(false)
, mDistantTerrain(false)
2018-08-01 16:18:37 +00:00
, mFieldOfViewOverridden(false)
, mFieldOfViewOverride(0.f)
2018-06-12 23:48:31 +00:00
, mBorders(false)
{
resourceSystem->getSceneManager()->setParticleSystemMask(MWRender::Mask_ParticleSystem);
resourceSystem->getSceneManager()->setShaderPath(resourcePath + "/shaders");
2020-03-14 12:39:32 +00:00
// Shadows and radial fog have problems with fixed-function mode
bool forceShaders = Settings::Manager::getBool("radial fog", "Shaders") || Settings::Manager::getBool("force shaders", "Shaders") || Settings::Manager::getBool("enable shadows", "Shadows");
resourceSystem->getSceneManager()->setForceShaders(forceShaders);
// FIXME: calling dummy method because terrain needs to know whether lighting is clamped
2016-02-18 16:08:18 +00:00
resourceSystem->getSceneManager()->setClampLighting(Settings::Manager::getBool("clamp lighting", "Shaders"));
resourceSystem->getSceneManager()->setAutoUseNormalMaps(Settings::Manager::getBool("auto use object normal maps", "Shaders"));
resourceSystem->getSceneManager()->setNormalMapPattern(Settings::Manager::getString("normal map pattern", "Shaders"));
2016-03-22 20:00:31 +00:00
resourceSystem->getSceneManager()->setNormalHeightMapPattern(Settings::Manager::getString("normal height map pattern", "Shaders"));
2016-02-20 18:02:11 +00:00
resourceSystem->getSceneManager()->setAutoUseSpecularMaps(Settings::Manager::getBool("auto use object specular maps", "Shaders"));
resourceSystem->getSceneManager()->setSpecularMapPattern(Settings::Manager::getString("specular map pattern", "Shaders"));
2015-11-10 16:00:33 +00:00
2016-02-08 15:45:56 +00:00
osg::ref_ptr<SceneUtil::LightManager> sceneRoot = new SceneUtil::LightManager;
sceneRoot->setLightingMask(Mask_Lighting);
2016-02-08 15:45:56 +00:00
mSceneRoot = sceneRoot;
sceneRoot->setStartLight(1);
sceneRoot->setNodeMask(Mask_Scene);
sceneRoot->setName("Scene Root");
int shadowCastingTraversalMask = Mask_Scene;
2017-12-27 02:32:17 +00:00
if (Settings::Manager::getBool("actor shadows", "Shadows"))
shadowCastingTraversalMask |= Mask_Actor;
2017-12-27 02:32:17 +00:00
if (Settings::Manager::getBool("player shadows", "Shadows"))
shadowCastingTraversalMask |= Mask_Player;
2017-12-27 02:32:17 +00:00
if (Settings::Manager::getBool("terrain shadows", "Shadows"))
shadowCastingTraversalMask |= Mask_Terrain;
2018-03-03 15:13:36 +00:00
int indoorShadowCastingTraversalMask = shadowCastingTraversalMask;
if (Settings::Manager::getBool("object shadows", "Shadows"))
shadowCastingTraversalMask |= (Mask_Object|Mask_Static);
mShadowManager.reset(new SceneUtil::ShadowManager(sceneRoot, mRootNode, shadowCastingTraversalMask, indoorShadowCastingTraversalMask, mResourceSystem->getSceneManager()->getShaderManager()));
2017-04-14 13:32:18 +00:00
2018-02-26 22:27:09 +00:00
Shader::ShaderManager::DefineMap shadowDefines = mShadowManager->getShadowDefines();
Shader::ShaderManager::DefineMap globalDefines = mResourceSystem->getSceneManager()->getShaderManager().getGlobalDefines();
for (auto itr = shadowDefines.begin(); itr != shadowDefines.end(); itr++)
globalDefines[itr->first] = itr->second;
globalDefines["forcePPL"] = Settings::Manager::getBool("force per pixel lighting", "Shaders") ? "1" : "0";
globalDefines["clamp"] = Settings::Manager::getBool("clamp lighting", "Shaders") ? "1" : "0";
globalDefines["preLightEnv"] = Settings::Manager::getBool("apply lighting to environment maps", "Shaders") ? "1" : "0";
2020-03-14 12:39:32 +00:00
globalDefines["radialFog"] = Settings::Manager::getBool("radial fog", "Shaders") ? "1" : "0";
// It is unnecessary to stop/start the viewer as no frames are being rendered yet.
mResourceSystem->getSceneManager()->getShaderManager().setGlobalDefines(globalDefines);
2011-11-04 03:47:15 +00:00
mNavMesh.reset(new NavMesh(mRootNode, Settings::Manager::getBool("enable nav mesh render", "Navigator")));
mActorsPaths.reset(new ActorsPaths(mRootNode, Settings::Manager::getBool("enable agents paths render", "Navigator")));
2019-11-27 22:45:01 +00:00
mRecastMesh.reset(new RecastMesh(mRootNode, Settings::Manager::getBool("enable recast mesh render", "Navigator")));
mPathgrid.reset(new Pathgrid(mRootNode));
2015-05-02 20:45:27 +00:00
2016-02-08 15:45:56 +00:00
mObjects.reset(new Objects(mResourceSystem, sceneRoot, mUnrefQueue.get()));
2012-02-26 12:13:29 +00:00
2018-10-09 06:21:12 +00:00
if (getenv("OPENMW_DONT_PRECOMPILE") == nullptr)
{
mViewer->setIncrementalCompileOperation(new osgUtil::IncrementalCompileOperation);
mViewer->getIncrementalCompileOperation()->setTargetFrameRate(Settings::Manager::getFloat("target framerate", "Cells"));
mViewer->getIncrementalCompileOperation()->setMaximumNumOfObjectsToCompilePerFrame(100);
}
mResourceSystem->getSceneManager()->setIncrementalCompileOperation(mViewer->getIncrementalCompileOperation());
2016-02-08 15:45:56 +00:00
mEffectManager.reset(new EffectManager(sceneRoot, mResourceSystem));
2015-06-02 14:35:35 +00:00
DLLandFogStart = Settings::Manager::getFloat("distant land fog start", "Fog");
DLLandFogEnd = Settings::Manager::getFloat("distant land fog end", "Fog");
DLUnderwaterFogStart = Settings::Manager::getFloat("distant underwater fog start", "Fog");
DLUnderwaterFogEnd = Settings::Manager::getFloat("distant underwater fog end", "Fog");
DLInteriorFogStart = Settings::Manager::getFloat("distant interior fog start", "Fog");
DLInteriorFogEnd = Settings::Manager::getFloat("distant interior fog end", "Fog");
mDistantFog = Settings::Manager::getBool("use distant fog", "Fog");
mDistantTerrain = Settings::Manager::getBool("distant terrain", "Terrain");
const std::string normalMapPattern = Settings::Manager::getString("normal map pattern", "Shaders");
const std::string heightMapPattern = Settings::Manager::getString("normal height map pattern", "Shaders");
const std::string specularMapPattern = Settings::Manager::getString("terrain specular map pattern", "Shaders");
const bool useTerrainNormalMaps = Settings::Manager::getBool("auto use terrain normal maps", "Shaders");
const bool useTerrainSpecularMaps = Settings::Manager::getBool("auto use terrain specular maps", "Shaders");
mTerrainStorage = new TerrainStorage(mResourceSystem, normalMapPattern, heightMapPattern, useTerrainNormalMaps, specularMapPattern, useTerrainSpecularMaps);
if (mDistantTerrain)
{
const int compMapResolution = Settings::Manager::getInt("composite map resolution", "Terrain");
int compMapPower = Settings::Manager::getInt("composite map level", "Terrain");
compMapPower = std::max(-3, compMapPower);
float compMapLevel = pow(2, compMapPower);
const float lodFactor = Settings::Manager::getFloat("lod factor", "Terrain");
2019-02-20 13:37:00 +00:00
const int vertexLodMod = Settings::Manager::getInt("vertex lod mod", "Terrain");
float maxCompGeometrySize = Settings::Manager::getFloat("max composite geometry size", "Terrain");
maxCompGeometrySize = std::max(maxCompGeometrySize, 1.f);
mTerrain.reset(new Terrain::QuadTreeWorld(
sceneRoot, mRootNode, mResourceSystem, mTerrainStorage, Mask_Terrain, Mask_PreCompile, Mask_Debug,
compMapResolution, compMapLevel, lodFactor, vertexLodMod, maxCompGeometrySize));
}
else
mTerrain.reset(new Terrain::TerrainGrid(sceneRoot, mRootNode, mResourceSystem, mTerrainStorage, Mask_Terrain, Mask_PreCompile, Mask_Debug));
2018-06-12 23:48:31 +00:00
mTerrain->setTargetFrameRate(Settings::Manager::getFloat("target framerate", "Cells"));
mTerrain->setWorkQueue(mWorkQueue.get());
2015-04-19 15:55:56 +00:00
2019-06-13 13:37:00 +00:00
// water goes after terrain for correct waterculling order
mWater.reset(new Water(mRootNode, sceneRoot, mResourceSystem, mViewer->getIncrementalCompileOperation(), resourcePath));
2015-05-21 21:54:39 +00:00
mCamera.reset(new Camera(mViewer->getCamera()));
2015-05-14 19:42:04 +00:00
mViewer->setLightingMode(osgViewer::View::NO_LIGHT);
2011-11-04 03:47:15 +00:00
osg::ref_ptr<osg::LightSource> source = new osg::LightSource;
source->setNodeMask(Mask_Lighting);
mSunLight = new osg::Light;
source->setLight(mSunLight);
mSunLight->setDiffuse(osg::Vec4f(0,0,0,1));
mSunLight->setAmbient(osg::Vec4f(0,0,0,1));
2015-05-26 16:22:21 +00:00
mSunLight->setSpecular(osg::Vec4f(0,0,0,0));
mSunLight->setConstantAttenuation(1.f);
2016-02-08 15:45:56 +00:00
sceneRoot->addChild(source);
2011-11-04 03:47:15 +00:00
2016-02-08 15:45:56 +00:00
sceneRoot->getOrCreateStateSet()->setMode(GL_CULL_FACE, osg::StateAttribute::ON);
sceneRoot->getOrCreateStateSet()->setMode(GL_LIGHTING, osg::StateAttribute::ON);
sceneRoot->getOrCreateStateSet()->setMode(GL_NORMALIZE, osg::StateAttribute::ON);
osg::ref_ptr<osg::Material> defaultMat (new osg::Material);
defaultMat->setColorMode(osg::Material::OFF);
defaultMat->setAmbient(osg::Material::FRONT_AND_BACK, osg::Vec4f(1,1,1,1));
defaultMat->setDiffuse(osg::Material::FRONT_AND_BACK, osg::Vec4f(1,1,1,1));
defaultMat->setSpecular(osg::Material::FRONT_AND_BACK, osg::Vec4f(0.f, 0.f, 0.f, 0.f));
sceneRoot->getOrCreateStateSet()->setAttribute(defaultMat);
2016-02-08 15:45:56 +00:00
mSky.reset(new SkyManager(sceneRoot, resourceSystem->getSceneManager()));
2017-10-24 12:12:41 +00:00
2017-10-14 16:45:29 +00:00
mSky->setCamera(mViewer->getCamera());
mSky->setRainIntensityUniform(mWater->getRainIntensityUniform());
source->setStateSetModes(*mRootNode->getOrCreateStateSet(), osg::StateAttribute::ON);
mStateUpdater = new StateUpdater;
2016-02-08 15:45:56 +00:00
sceneRoot->addUpdateCallback(mStateUpdater);
osg::Camera::CullingMode cullingMode = osg::Camera::DEFAULT_CULLING|osg::Camera::FAR_PLANE_CULLING;
2011-11-04 03:47:15 +00:00
2015-05-24 00:34:20 +00:00
if (!Settings::Manager::getBool("small feature culling", "Camera"))
cullingMode &= ~(osg::CullStack::SMALL_FEATURE_CULLING);
else
{
mViewer->getCamera()->setSmallFeatureCullingPixelSize(Settings::Manager::getFloat("small feature culling pixel size", "Camera"));
cullingMode |= osg::CullStack::SMALL_FEATURE_CULLING;
}
2015-05-14 19:42:04 +00:00
mViewer->getCamera()->setCullingMode( cullingMode );
2015-05-14 19:42:04 +00:00
mViewer->getCamera()->setComputeNearFarMode(osg::Camera::DO_NOT_COMPUTE_NEAR_FAR);
mViewer->getCamera()->setCullingMode(cullingMode);
mViewer->getCamera()->setCullMask(~(Mask_UpdateVisitor|Mask_SimpleWater));
NifOsg::Loader::setHiddenNodeMask(Mask_UpdateVisitor);
2020-04-29 13:20:03 +00:00
NifOsg::Loader::setIntersectionDisabledNodeMask(Mask_Effect);
2015-05-20 01:35:52 +00:00
2015-05-24 00:34:20 +00:00
mNearClip = Settings::Manager::getFloat("near clip", "Camera");
mViewDistance = Settings::Manager::getFloat("viewing distance", "Camera");
2019-03-03 17:29:51 +00:00
float fov = Settings::Manager::getFloat("field of view", "Camera");
mFieldOfView = std::min(std::max(1.f, fov), 179.f);
float firstPersonFov = Settings::Manager::getFloat("first person field of view", "Camera");
mFirstPersonFieldOfView = std::min(std::max(1.f, firstPersonFov), 179.f);
2015-06-02 14:35:35 +00:00
mStateUpdater->setFogEnd(mViewDistance);
2015-10-28 17:59:35 +00:00
mRootNode->getOrCreateStateSet()->addUniform(new osg::Uniform("near", mNearClip));
mRootNode->getOrCreateStateSet()->addUniform(new osg::Uniform("far", mViewDistance));
mUniformNear = mRootNode->getOrCreateStateSet()->getUniform("near");
mUniformFar = mRootNode->getOrCreateStateSet()->getUniform("far");
updateProjectionMatrix();
}
RenderingManager::~RenderingManager()
{
// let background loading thread finish before we delete anything else
2018-10-09 06:21:12 +00:00
mWorkQueue = nullptr;
}
2019-11-24 13:40:19 +00:00
osgUtil::IncrementalCompileOperation* RenderingManager::getIncrementalCompileOperation()
{
return mViewer->getIncrementalCompileOperation();
}
MWRender::Objects& RenderingManager::getObjects()
{
return *mObjects.get();
}
Resource::ResourceSystem* RenderingManager::getResourceSystem()
{
return mResourceSystem;
}
2016-02-09 19:57:30 +00:00
SceneUtil::WorkQueue* RenderingManager::getWorkQueue()
{
return mWorkQueue.get();
}
2016-02-09 19:57:30 +00:00
SceneUtil::UnrefQueue* RenderingManager::getUnrefQueue()
{
return mUnrefQueue.get();
}
2016-02-09 19:57:30 +00:00
Terrain::World* RenderingManager::getTerrain()
{
return mTerrain.get();
}
2016-02-09 00:17:02 +00:00
void RenderingManager::preloadCommonAssets()
{
osg::ref_ptr<PreloadCommonAssetsWorkItem> workItem (new PreloadCommonAssetsWorkItem(mResourceSystem));
mSky->listAssetsToPreload(workItem->mModels, workItem->mTextures);
mWater->listAssetsToPreload(workItem->mTextures);
const char* basemodels[] = {"xbase_anim", "xbase_anim.1st", "xbase_anim_female", "xbase_animkna"};
for (size_t i=0; i<sizeof(basemodels)/sizeof(basemodels[0]); ++i)
{
workItem->mModels.push_back(std::string("meshes/") + basemodels[i] + ".nif");
workItem->mKeyframes.push_back(std::string("meshes/") + basemodels[i] + ".kf");
}
2016-02-09 00:17:02 +00:00
workItem->mTextures.push_back("textures/_land_default.dds");
mWorkQueue->addWorkItem(workItem);
}
double RenderingManager::getReferenceTime() const
{
return mViewer->getFrameStamp()->getReferenceTime();
}
2015-12-04 23:44:04 +00:00
osg::Group* RenderingManager::getLightRoot()
{
2016-02-08 15:45:56 +00:00
return mSceneRoot.get();
2015-12-04 23:44:04 +00:00
}
2015-06-11 21:16:05 +00:00
void RenderingManager::setNightEyeFactor(float factor)
{
if (factor != mNightEyeFactor)
{
mNightEyeFactor = factor;
updateAmbient();
}
}
void RenderingManager::setAmbientColour(const osg::Vec4f &colour)
{
2015-06-11 21:16:05 +00:00
mAmbientColor = colour;
updateAmbient();
}
2015-06-16 18:56:48 +00:00
void RenderingManager::skySetDate(int day, int month)
{
mSky->setDate(day, month);
}
int RenderingManager::skyGetMasserPhase() const
{
return mSky->getMasserPhase();
}
int RenderingManager::skyGetSecundaPhase() const
{
return mSky->getSecundaPhase();
}
void RenderingManager::skySetMoonColour(bool red)
{
mSky->setMoonColour(red);
}
void RenderingManager::configureAmbient(const ESM::Cell *cell)
2012-05-22 23:32:36 +00:00
{
setAmbientColour(SceneUtil::colourFromRGB(cell->mAmbi.mAmbient));
2016-02-20 18:02:11 +00:00
osg::Vec4f diffuse = SceneUtil::colourFromRGB(cell->mAmbi.mSunlight);
mSunLight->setDiffuse(diffuse);
mSunLight->setSpecular(diffuse);
2018-08-08 22:30:33 +00:00
mSunLight->setPosition(osg::Vec4f(-0.15f, 0.15f, 1.f, 0.f));
}
void RenderingManager::setSunColour(const osg::Vec4f& diffuse, const osg::Vec4f& specular)
{
2015-05-26 16:22:21 +00:00
// need to wrap this in a StateUpdater?
mSunLight->setDiffuse(diffuse);
mSunLight->setSpecular(specular);
}
void RenderingManager::setSunDirection(const osg::Vec3f &direction)
{
2015-05-20 00:07:18 +00:00
osg::Vec3 position = direction * -1;
2015-05-26 16:22:21 +00:00
// need to wrap this in a StateUpdater?
2015-05-20 00:07:18 +00:00
mSunLight->setPosition(osg::Vec4(position.x(), position.y(), position.z(), 0));
2015-05-20 00:07:18 +00:00
mSky->setSunDirection(position);
}
2015-05-02 20:45:27 +00:00
void RenderingManager::addCell(const MWWorld::CellStore *store)
{
mPathgrid->addCell(store);
2015-06-02 14:35:35 +00:00
mWater->changeCell(store);
2015-06-02 23:18:36 +00:00
if (store->getCell()->isExterior())
mTerrain->loadCell(store->getCell()->getGridX(), store->getCell()->getGridY());
2015-05-02 20:45:27 +00:00
}
void RenderingManager::removeCell(const MWWorld::CellStore *store)
2013-08-20 07:52:27 +00:00
{
mPathgrid->removeCell(store);
mActorsPaths->removeCell(store);
mObjects->removeCell(store);
2015-06-02 23:18:36 +00:00
if (store->getCell()->isExterior())
mTerrain->unloadCell(store->getCell()->getGridX(), store->getCell()->getGridY());
2015-06-16 18:36:48 +00:00
mWater->removeCell(store);
2013-08-20 07:52:27 +00:00
}
2012-05-22 23:32:36 +00:00
2017-03-09 01:17:25 +00:00
void RenderingManager::enableTerrain(bool enable)
{
2019-06-13 13:37:00 +00:00
if (!enable)
mWater->setCullCallback(nullptr);
2017-03-09 01:17:25 +00:00
mTerrain->enable(enable);
}
void RenderingManager::setSkyEnabled(bool enabled)
{
mSky->setEnabled(enabled);
if (enabled)
mShadowManager->enableOutdoorMode();
else
mShadowManager->enableIndoorMode();
}
2018-06-12 23:48:31 +00:00
bool RenderingManager::toggleBorders()
{
mBorders = !mBorders;
mTerrain->setBordersVisible(mBorders);
return mBorders;
}
2015-05-02 20:45:27 +00:00
bool RenderingManager::toggleRenderMode(RenderMode mode)
{
if (mode == Render_CollisionDebug || mode == Render_Pathgrid)
return mPathgrid->toggleRenderMode(mode);
2015-05-02 20:45:27 +00:00
else if (mode == Render_Wireframe)
{
2015-06-01 15:02:44 +00:00
bool wireframe = !mStateUpdater->getWireframe();
mStateUpdater->setWireframe(wireframe);
return wireframe;
2015-05-02 20:45:27 +00:00
}
2015-06-02 14:35:35 +00:00
else if (mode == Render_Water)
{
return mWater->toggle();
}
else if (mode == Render_Scene)
{
int mask = mViewer->getCamera()->getCullMask();
bool enabled = mask&Mask_Scene;
2015-06-02 14:35:35 +00:00
enabled = !enabled;
if (enabled)
mask |= Mask_Scene;
2015-06-02 14:35:35 +00:00
else
mask &= ~Mask_Scene;
2015-06-02 14:35:35 +00:00
mViewer->getCamera()->setCullMask(mask);
return enabled;
}
else if (mode == Render_NavMesh)
{
return mNavMesh->toggle();
}
else if (mode == Render_ActorsPaths)
{
return mActorsPaths->toggle();
}
2019-11-27 22:45:01 +00:00
else if (mode == Render_RecastMesh)
{
return mRecastMesh->toggle();
}
2015-05-02 20:45:27 +00:00
return false;
}
void RenderingManager::configureFog(const ESM::Cell *cell)
{
osg::Vec4f color = SceneUtil::colourFromRGB(cell->mAmbi.mFog);
if(mDistantFog)
{
float density = std::max(0.2f, cell->mAmbi.mFogDensity);
mLandFogStart = (DLInteriorFogEnd*(1.0f-density) + DLInteriorFogStart*density);
mLandFogEnd = DLInteriorFogEnd;
mUnderwaterFogStart = DLUnderwaterFogStart;
mUnderwaterFogEnd = DLUnderwaterFogEnd;
mFogColor = color;
}
else
configureFog(cell->mAmbi.mFogDensity, mUnderwaterIndoorFog, 1.0f, 0.0f, color);
}
void RenderingManager::configureFog(float fogDepth, float underwaterFog, float dlFactor, float dlOffset, const osg::Vec4f &color)
{
if(mDistantFog)
{
mLandFogStart = dlFactor * (DLLandFogStart - dlOffset*DLLandFogEnd);
mLandFogEnd = dlFactor * (1.0f-dlOffset) * DLLandFogEnd;
mUnderwaterFogStart = DLUnderwaterFogStart;
mUnderwaterFogEnd = DLUnderwaterFogEnd;
}
else
{
if(fogDepth == 0.0)
{
mLandFogStart = 0.0f;
mLandFogEnd = std::numeric_limits<float>::max();
}
else
{
mLandFogStart = mViewDistance * (1 - fogDepth);
mLandFogEnd = mViewDistance;
}
mUnderwaterFogStart = std::min(mViewDistance, 6666.f) * (1 - underwaterFog);
mUnderwaterFogEnd = std::min(mViewDistance, 6666.f);
}
2015-06-02 14:35:35 +00:00
mFogColor = color;
}
SkyManager* RenderingManager::getSkyManager()
{
return mSky.get();
}
2015-04-18 23:57:52 +00:00
void RenderingManager::update(float dt, bool paused)
{
reportStats();
mUnrefQueue->flush(mWorkQueue.get());
if (!paused)
2015-06-20 15:08:46 +00:00
{
mEffectManager->update(dt);
2015-06-20 15:08:46 +00:00
mSky->update(dt);
mWater->update(dt);
2015-06-20 15:08:46 +00:00
}
2019-01-20 16:27:52 +00:00
updateNavMesh();
2019-11-27 22:45:01 +00:00
updateRecastMesh();
2015-05-21 23:43:16 +00:00
mCamera->update(dt, paused);
2015-06-02 14:35:35 +00:00
osg::Vec3f focal, cameraPos;
mCamera->getPosition(focal, cameraPos);
mCurrentCameraPos = cameraPos;
2015-06-02 14:35:35 +00:00
if (mWater->isUnderwater(cameraPos))
{
setFogColor(mUnderwaterColor * mUnderwaterWeight + mFogColor * (1.f-mUnderwaterWeight));
mStateUpdater->setFogStart(mUnderwaterFogStart);
mStateUpdater->setFogEnd(mUnderwaterFogEnd);
2015-06-02 14:35:35 +00:00
}
else
{
setFogColor(mFogColor);
mStateUpdater->setFogStart(mLandFogStart);
mStateUpdater->setFogEnd(mLandFogEnd);
2015-06-02 14:35:35 +00:00
}
2015-04-19 15:55:56 +00:00
}
2015-05-21 21:54:39 +00:00
void RenderingManager::updatePlayerPtr(const MWWorld::Ptr &ptr)
{
if(mPlayerAnimation.get())
{
setupPlayer(ptr);
2015-05-21 21:54:39 +00:00
mPlayerAnimation->updatePtr(ptr);
}
2015-05-21 21:54:39 +00:00
mCamera->attachTo(ptr);
}
2015-06-16 18:36:48 +00:00
void RenderingManager::removePlayer(const MWWorld::Ptr &player)
{
mWater->removeEmitter(player);
}
2015-04-23 21:50:46 +00:00
void RenderingManager::rotateObject(const MWWorld::Ptr &ptr, const osg::Quat& rot)
{
2015-05-21 21:54:39 +00:00
if(ptr == mCamera->getTrackingPtr() &&
!mCamera->isVanityOrPreviewModeEnabled())
{
2020-02-23 10:02:38 +00:00
mCamera->rotateCamera(-ptr.getRefData().getPosition().rot[0], -ptr.getRefData().getPosition().rot[1], -ptr.getRefData().getPosition().rot[2], false);
2015-05-21 21:54:39 +00:00
}
2015-04-23 21:50:46 +00:00
ptr.getRefData().getBaseNode()->setAttitude(rot);
}
void RenderingManager::moveObject(const MWWorld::Ptr &ptr, const osg::Vec3f &pos)
{
ptr.getRefData().getBaseNode()->setPosition(pos);
}
void RenderingManager::scaleObject(const MWWorld::Ptr &ptr, const osg::Vec3f &scale)
{
ptr.getRefData().getBaseNode()->setScale(scale);
if (ptr == mCamera->getTrackingPtr()) // update height of camera
mCamera->processViewChange();
2015-04-23 21:50:46 +00:00
}
2015-05-21 22:55:43 +00:00
void RenderingManager::removeObject(const MWWorld::Ptr &ptr)
{
mActorsPaths->remove(ptr);
2015-05-21 22:55:43 +00:00
mObjects->removeObject(ptr);
2015-06-16 18:36:48 +00:00
mWater->removeEmitter(ptr);
2015-05-21 22:55:43 +00:00
}
2015-06-02 14:35:35 +00:00
void RenderingManager::setWaterEnabled(bool enabled)
{
mWater->setEnabled(enabled);
mSky->setWaterEnabled(enabled);
2015-06-02 14:35:35 +00:00
}
void RenderingManager::setWaterHeight(float height)
{
2019-06-13 13:37:00 +00:00
mWater->setCullCallback(mTerrain->getHeightCullCallback(height, Mask_Water));
2015-06-02 14:35:35 +00:00
mWater->setHeight(height);
mSky->setWaterHeight(height);
2015-06-02 14:35:35 +00:00
}
2015-06-03 14:40:16 +00:00
class NotifyDrawCompletedCallback : public osg::Camera::DrawCallback
{
public:
2019-02-20 13:37:00 +00:00
NotifyDrawCompletedCallback(unsigned int frame)
: mDone(false), mFrame(frame)
{
Log(Debug::Verbose) << "NotifyDrawCompletedCallback: " << mFrame;
}
2015-06-03 14:40:16 +00:00
virtual void operator () (osg::RenderInfo& renderInfo) const
{
Log(Debug::Verbose) << "NotifyDrawCompletedCallback: " << renderInfo.getState()->getFrameStamp()->getFrameNumber() << " >= " << mFrame;
2019-02-20 13:37:00 +00:00
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
if (renderInfo.getState()->getFrameStamp()->getFrameNumber() >= mFrame)
{
mDone = true;
mCondition.signal();
}
2015-06-03 14:40:16 +00:00
}
void waitTillDone()
{
OpenThreads::ScopedLock<OpenThreads::Mutex> lock(mMutex);
if (mDone)
return;
mCondition.wait(&mMutex);
}
2015-06-03 14:40:16 +00:00
mutable OpenThreads::Condition mCondition;
mutable OpenThreads::Mutex mMutex;
mutable bool mDone;
2019-02-20 13:37:00 +00:00
unsigned int mFrame;
2015-06-03 14:40:16 +00:00
};
bool RenderingManager::screenshot360(osg::Image* image, std::string settingStr)
2017-11-07 12:07:11 +00:00
{
int screenshotW = mViewer->getCamera()->getViewport()->width();
int screenshotH = mViewer->getCamera()->getViewport()->height();
2017-11-15 16:01:16 +00:00
int screenshotMapping = 0;
std::vector<std::string> settingArgs;
Misc::StringUtils::split(settingStr, settingArgs);
if (settingArgs.size() > 0)
{
std::string typeStrings[4] = {"spherical","cylindrical","planet","cubemap"};
bool found = false;
for (int i = 0; i < 4; ++i)
if (settingArgs[0].compare(typeStrings[i]) == 0)
{
2017-11-15 16:01:16 +00:00
screenshotMapping = i;
found = true;
break;
}
if (!found)
{
2018-08-14 19:05:43 +00:00
Log(Debug::Warning) << "Wrong screenshot type: " << settingArgs[0] << ".";
return false;
}
}
// planet mapping needs higher resolution
int cubeSize = screenshotMapping == 2 ? screenshotW : screenshotW / 2;
if (settingArgs.size() > 1)
screenshotW = std::min(10000,std::atoi(settingArgs[1].c_str()));
if (settingArgs.size() > 2)
screenshotH = std::min(10000,std::atoi(settingArgs[2].c_str()));
if (settingArgs.size() > 3)
cubeSize = std::min(5000,std::atoi(settingArgs[3].c_str()));
if (mCamera->isVanityOrPreviewModeEnabled())
{
2018-08-14 19:05:43 +00:00
Log(Debug::Warning) << "Spherical screenshots are not allowed in preview mode.";
return false;
}
2017-11-09 19:25:29 +00:00
2017-11-15 16:01:16 +00:00
bool rawCubemap = screenshotMapping == 3;
if (rawCubemap)
screenshotW = cubeSize * 6; // the image will consist of 6 cube sides in a row
2017-11-15 16:01:16 +00:00
else if (screenshotMapping == 2)
screenshotH = screenshotW; // use square resolution for planet mapping
2017-11-10 09:50:28 +00:00
2017-11-15 16:01:16 +00:00
std::vector<osg::ref_ptr<osg::Image>> images;
for (int i = 0; i < 6; ++i)
images.push_back(new osg::Image);
2017-11-07 18:47:36 +00:00
2017-11-07 14:02:01 +00:00
osg::Vec3 directions[6] = {
2017-11-15 16:01:16 +00:00
rawCubemap ? osg::Vec3(1,0,0) : osg::Vec3(0,0,1),
osg::Vec3(0,0,-1),
osg::Vec3(-1,0,0),
2017-11-15 16:01:16 +00:00
rawCubemap ? osg::Vec3(0,0,1) : osg::Vec3(1,0,0),
2017-11-15 15:07:01 +00:00
osg::Vec3(0,1,0),
osg::Vec3(0,-1,0)};
double rotations[] = {
-osg::PI / 2.0,
osg::PI / 2.0,
osg::PI,
0,
osg::PI / 2.0,
osg::PI / 2.0};
2017-11-07 14:02:01 +00:00
double fovBackup = mFieldOfView;
mFieldOfView = 90.0; // each cubemap side sees 90 degrees
2017-11-07 14:02:01 +00:00
2017-11-09 22:09:13 +00:00
int maskBackup = mPlayerAnimation->getObjectRoot()->getNodeMask();
if (mCamera->isFirstPerson())
mPlayerAnimation->getObjectRoot()->setNodeMask(0);
2017-11-16 22:20:24 +00:00
for (int i = 0; i < 6; ++i) // for each cubemap side
2017-11-07 14:02:01 +00:00
{
2017-11-15 16:01:16 +00:00
osg::Matrixd transform = osg::Matrixd::rotate(osg::Vec3(0,0,-1),directions[i]);
if (!rawCubemap)
transform *= osg::Matrixd::rotate(rotations[i],osg::Vec3(0,0,-1));
osg::Image *sideImage = images[i].get();
2017-11-15 15:07:01 +00:00
screenshot(sideImage,cubeSize,cubeSize,transform);
2017-11-07 14:02:01 +00:00
2017-11-15 16:01:16 +00:00
if (!rawCubemap)
sideImage->flipHorizontal();
}
2017-11-07 21:13:05 +00:00
2017-11-09 22:09:13 +00:00
mPlayerAnimation->getObjectRoot()->setNodeMask(maskBackup);
2017-11-07 14:02:01 +00:00
mFieldOfView = fovBackup;
2017-11-15 16:01:16 +00:00
if (rawCubemap) // for raw cubemap don't run on GPU, just merge the images
{
2017-11-15 16:01:16 +00:00
image->allocateImage(cubeSize * 6,cubeSize,images[0]->r(),images[0]->getPixelFormat(),images[0]->getDataType());
2017-11-15 14:20:59 +00:00
2017-11-15 16:01:16 +00:00
for (int i = 0; i < 6; ++i)
osg::copyImage(images[i].get(),0,0,0,images[i]->s(),images[i]->t(),images[i]->r(),image,i * cubeSize,0,0);
2017-11-15 14:20:59 +00:00
2017-11-15 16:01:16 +00:00
return true;
}
2017-11-15 16:01:16 +00:00
// run on GPU now:
2017-11-15 14:20:59 +00:00
2017-11-15 15:07:01 +00:00
osg::ref_ptr<osg::TextureCubeMap> cubeTexture (new osg::TextureCubeMap);
2017-11-16 19:06:34 +00:00
cubeTexture->setResizeNonPowerOfTwoHint(false);
2017-11-16 21:43:41 +00:00
cubeTexture->setFilter(osg::Texture::MIN_FILTER,osg::Texture::NEAREST);
cubeTexture->setFilter(osg::Texture::MAG_FILTER,osg::Texture::NEAREST);
2017-11-15 16:01:16 +00:00
cubeTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
cubeTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
2017-11-15 14:20:59 +00:00
2017-11-15 15:07:01 +00:00
for (int i = 0; i < 6; ++i)
2017-11-15 16:01:16 +00:00
cubeTexture->setImage(i,images[i].get());
2017-11-15 14:20:59 +00:00
2017-11-15 15:07:01 +00:00
osg::ref_ptr<osg::Camera> screenshotCamera (new osg::Camera);
osg::ref_ptr<osg::ShapeDrawable> quad (new osg::ShapeDrawable(new osg::Box(osg::Vec3(0,0,0),2.0)));
2017-11-15 14:20:59 +00:00
2017-11-15 15:07:01 +00:00
std::map<std::string, std::string> defineMap;
2017-11-15 14:20:59 +00:00
2017-11-15 15:07:01 +00:00
Shader::ShaderManager& shaderMgr = mResourceSystem->getSceneManager()->getShaderManager();
osg::ref_ptr<osg::Shader> fragmentShader (shaderMgr.getShader("s360_fragment.glsl",defineMap,osg::Shader::FRAGMENT));
osg::ref_ptr<osg::Shader> vertexShader (shaderMgr.getShader("s360_vertex.glsl", defineMap, osg::Shader::VERTEX));
osg::ref_ptr<osg::StateSet> stateset = new osg::StateSet;
osg::ref_ptr<osg::Program> program (new osg::Program);
program->addShader(fragmentShader);
program->addShader(vertexShader);
stateset->setAttributeAndModes(program, osg::StateAttribute::ON);
stateset->addUniform(new osg::Uniform("cubeMap",0));
2017-11-15 16:01:16 +00:00
stateset->addUniform(new osg::Uniform("mapping",screenshotMapping));
2017-11-15 15:07:01 +00:00
stateset->setTextureAttributeAndModes(0,cubeTexture,osg::StateAttribute::ON);
2017-11-15 15:07:01 +00:00
quad->setStateSet(stateset);
2018-10-09 06:21:12 +00:00
quad->setUpdateCallback(nullptr);
2017-11-15 14:20:59 +00:00
2017-11-15 15:07:01 +00:00
screenshotCamera->addChild(quad);
2017-11-15 14:20:59 +00:00
2017-11-15 16:01:16 +00:00
renderCameraToImage(screenshotCamera,image,screenshotW,screenshotH);
2017-11-15 14:20:59 +00:00
return true;
2017-11-07 12:07:11 +00:00
}
2017-11-15 14:20:59 +00:00
void RenderingManager::renderCameraToImage(osg::Camera *camera, osg::Image *image, int w, int h)
{
camera->setNodeMask(Mask_RenderToTexture);
2017-11-15 14:20:59 +00:00
camera->attach(osg::Camera::COLOR_BUFFER, image);
camera->setRenderOrder(osg::Camera::PRE_RENDER);
camera->setReferenceFrame(osg::Camera::ABSOLUTE_RF);
camera->setRenderTargetImplementation(osg::Camera::FRAME_BUFFER_OBJECT,osg::Camera::PIXEL_BUFFER_RTT);
camera->setViewport(0, 0, w, h);
osg::ref_ptr<osg::Texture2D> texture (new osg::Texture2D);
texture->setInternalFormat(GL_RGB);
texture->setTextureSize(w,h);
texture->setResizeNonPowerOfTwoHint(false);
texture->setFilter(osg::Texture::MIN_FILTER, osg::Texture::LINEAR);
texture->setFilter(osg::Texture::MAG_FILTER, osg::Texture::LINEAR);
camera->attach(osg::Camera::COLOR_BUFFER,texture);
image->setDataType(GL_UNSIGNED_BYTE);
image->setPixelFormat(texture->getInternalFormat());
mRootNode->addChild(camera);
2017-11-15 14:20:59 +00:00
// The draw needs to complete before we can copy back our image.
2019-02-20 13:37:00 +00:00
osg::ref_ptr<NotifyDrawCompletedCallback> callback (new NotifyDrawCompletedCallback(0));
2017-11-15 14:20:59 +00:00
camera->setFinalDrawCallback(callback);
MWBase::Environment::get().getWindowManager()->getLoadingScreen()->loadingOn(false);
2017-11-16 11:46:46 +00:00
2017-11-15 14:20:59 +00:00
mViewer->eventTraversal();
mViewer->updateTraversal();
mViewer->renderingTraversals();
callback->waitTillDone();
MWBase::Environment::get().getWindowManager()->getLoadingScreen()->loadingOff();
2017-11-16 11:17:52 +00:00
2017-11-15 14:20:59 +00:00
// now that we've "used up" the current frame, get a fresh framenumber for the next frame() following after the screenshot is completed
mViewer->advance(mViewer->getFrameStamp()->getSimulationTime());
camera->removeChildren(0, camera->getNumChildren());
mRootNode->removeChild(camera);
2017-11-15 14:20:59 +00:00
}
2019-02-20 13:37:00 +00:00
class ReadImageFromFramebufferCallback : public osg::Drawable::DrawCallback
{
public:
ReadImageFromFramebufferCallback(osg::Image* image, int width, int height)
: mWidth(width), mHeight(height), mImage(image)
{
}
virtual void drawImplementation(osg::RenderInfo& renderInfo,const osg::Drawable* /*drawable*/) const
{
int screenW = renderInfo.getCurrentCamera()->getViewport()->width();
int screenH = renderInfo.getCurrentCamera()->getViewport()->height();
double imageaspect = (double)mWidth/(double)mHeight;
int leftPadding = std::max(0, static_cast<int>(screenW - screenH * imageaspect) / 2);
int topPadding = std::max(0, static_cast<int>(screenH - screenW / imageaspect) / 2);
int width = screenW - leftPadding*2;
int height = screenH - topPadding*2;
mImage->readPixels(leftPadding, topPadding, width, height, GL_RGB, GL_UNSIGNED_BYTE);
mImage->scaleImage(mWidth, mHeight, 1);
}
private:
int mWidth;
int mHeight;
osg::ref_ptr<osg::Image> mImage;
};
void RenderingManager::screenshotFramebuffer(osg::Image* image, int w, int h)
2019-02-20 13:37:00 +00:00
{
osg::Camera* camera = mViewer->getCamera();
#ifdef USE_OPENXR
// In VR mode, the main camera is disabled.
auto& slave = mViewer->getSlave(1);
camera = slave._camera;
#endif
2019-02-20 13:37:00 +00:00
osg::ref_ptr<osg::Drawable> tempDrw = new osg::Drawable;
tempDrw->setDrawCallback(new ReadImageFromFramebufferCallback(image, w, h));
tempDrw->setCullingActive(false);
tempDrw->getOrCreateStateSet()->setRenderBinDetails(100, "RenderBin", osg::StateSet::USE_RENDERBIN_DETAILS); // so its after all scene bins but before POST_RENDER gui camera
camera->addChild(tempDrw);
osg::ref_ptr<NotifyDrawCompletedCallback> callback (new NotifyDrawCompletedCallback(mViewer->getFrameStamp()->getFrameNumber()));
auto* oldCb = camera->getFinalDrawCallback();
2019-02-20 13:37:00 +00:00
camera->setFinalDrawCallback(callback);
mViewer->eventTraversal();
mViewer->updateTraversal();
mViewer->renderingTraversals();
callback->waitTillDone();
// now that we've "used up" the current frame, get a fresh frame number for the next frame() following after the screenshot is completed
2019-02-20 13:37:00 +00:00
mViewer->advance(mViewer->getFrameStamp()->getSimulationTime());
camera->removeChild(tempDrw);
camera->setFinalDrawCallback(oldCb);
2019-02-20 13:37:00 +00:00
}
2017-11-15 15:07:01 +00:00
void RenderingManager::screenshot(osg::Image *image, int w, int h, osg::Matrixd cameraTransform)
2015-06-03 14:40:16 +00:00
{
osg::ref_ptr<osg::Camera> rttCamera (new osg::Camera);
rttCamera->setProjectionMatrixAsPerspective(mFieldOfView, w/float(h), mNearClip, mViewDistance);
2017-11-15 15:07:01 +00:00
rttCamera->setViewMatrix(mViewer->getCamera()->getViewMatrix() * cameraTransform);
2017-11-07 14:02:01 +00:00
rttCamera->setUpdateCallback(new NoTraverseCallback);
2016-02-08 15:45:56 +00:00
rttCamera->addChild(mSceneRoot);
2015-06-03 14:40:16 +00:00
rttCamera->addChild(mWater->getReflectionCamera());
rttCamera->addChild(mWater->getRefractionCamera());
2015-06-03 14:40:16 +00:00
rttCamera->setCullMask(mViewer->getCamera()->getCullMask() & (~Mask_GUI));
2017-11-14 17:23:12 +00:00
rttCamera->setClearMask(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
2015-06-03 14:40:16 +00:00
2017-11-15 14:20:59 +00:00
renderCameraToImage(rttCamera.get(),image,w,h);
2015-06-03 14:40:16 +00:00
}
osg::Vec4f RenderingManager::getScreenBounds(const MWWorld::Ptr& ptr)
{
if (!ptr.getRefData().getBaseNode())
return osg::Vec4f();
osg::ComputeBoundsVisitor computeBoundsVisitor;
computeBoundsVisitor.setTraversalMask(~(Mask_ParticleSystem|Mask_Effect));
ptr.getRefData().getBaseNode()->accept(computeBoundsVisitor);
osg::Matrix viewProj = mViewer->getCamera()->getViewMatrix() * mViewer->getCamera()->getProjectionMatrix();
float min_x = 1.0f, max_x = 0.0f, min_y = 1.0f, max_y = 0.0f;
for (int i=0; i<8; ++i)
{
osg::Vec3f corner = computeBoundsVisitor.getBoundingBox().corner(i);
corner = corner * viewProj;
float x = (corner.x() + 1.f) * 0.5f;
float y = (corner.y() - 1.f) * (-0.5f);
if (x < min_x)
min_x = x;
if (x > max_x)
max_x = x;
if (y < min_y)
min_y = y;
if (y > max_y)
max_y = y;
}
return osg::Vec4f(min_x, min_y, max_x, max_y);
}
2020-03-10 21:39:11 +00:00
RayResult getIntersectionResult (osgUtil::LineSegmentIntersector* intersector)
{
2020-03-10 21:39:11 +00:00
RayResult result;
result.mHit = false;
result.mRatio = 0;
result.mHitNode = nullptr;
if (intersector->containsIntersections())
{
result.mHit = true;
osgUtil::LineSegmentIntersector::Intersection intersection = intersector->getFirstIntersection();
2020-03-08 13:14:24 +00:00
result.mHitPointLocal = intersection.getLocalIntersectPoint();
result.mHitPointWorld = intersection.getWorldIntersectPoint();
result.mHitNormalWorld = intersection.getWorldIntersectNormal();
result.mRatio = intersection.ratio;
if(!intersection.nodePath.empty())
result.mHitNode = intersection.nodePath.back();
2018-10-09 06:21:12 +00:00
PtrHolder* ptrHolder = nullptr;
for (osg::NodePath::const_iterator it = intersection.nodePath.begin(); it != intersection.nodePath.end(); ++it)
{
osg::UserDataContainer* userDataContainer = (*it)->getUserDataContainer();
if (!userDataContainer)
continue;
for (unsigned int i=0; i<userDataContainer->getNumUserObjects(); ++i)
{
if (PtrHolder* p = dynamic_cast<PtrHolder*>(userDataContainer->getUserObject(i)))
ptrHolder = p;
}
}
if (ptrHolder)
result.mHitObject = ptrHolder->mPtr;
}
return result;
}
osg::ref_ptr<osgUtil::IntersectionVisitor> RenderingManager::getIntersectionVisitor(osgUtil::Intersector *intersector, bool ignorePlayer, bool ignoreActors)
{
if (!mIntersectionVisitor)
mIntersectionVisitor = new osgUtil::IntersectionVisitor;
mIntersectionVisitor->setTraversalNumber(mViewer->getFrameStamp()->getFrameNumber());
mIntersectionVisitor->setIntersector(intersector);
int mask = ~0;
mask &= ~(Mask_RenderToTexture|Mask_Sky|Mask_Debug|Mask_Effect|Mask_Water|Mask_SimpleWater);
if (ignorePlayer)
mask &= ~(Mask_Player);
if (ignoreActors)
mask &= ~(Mask_Actor|Mask_Player);
mIntersectionVisitor->setTraversalMask(mask);
return mIntersectionVisitor;
2015-10-31 00:30:02 +00:00
}
2020-03-10 21:39:11 +00:00
RayResult RenderingManager::castRay(const osg::Vec3f& origin, const osg::Vec3f& dest, bool ignorePlayer, bool ignoreActors)
2015-10-31 00:30:02 +00:00
{
osg::ref_ptr<osgUtil::LineSegmentIntersector> intersector (new osgUtil::LineSegmentIntersector(osgUtil::LineSegmentIntersector::MODEL,
origin, dest));
intersector->setIntersectionLimit(osgUtil::LineSegmentIntersector::LIMIT_NEAREST);
mRootNode->accept(*getIntersectionVisitor(intersector, ignorePlayer, ignoreActors));
return getIntersectionResult(intersector);
}
2020-03-15 14:31:38 +00:00
RayResult RenderingManager::castRay(const osg::Transform* source, float maxDistance, bool ignorePlayer, bool ignoreActors)
{
if (source)
{
osg::Matrix worldMatrix = osg::computeLocalToWorld(source->getParentalNodePaths()[0]);
2020-03-23 22:32:47 +00:00
osg::Vec3f direction = worldMatrix.getRotate() * osg::Vec3f(0, 1, 0);
2020-03-15 14:31:38 +00:00
direction.normalize();
osg::Vec3f raySource = worldMatrix.getTrans();
osg::Vec3f rayTarget = worldMatrix.getTrans() + direction * maxDistance;
return castRay(raySource, rayTarget, ignorePlayer, ignoreActors);
}
return RayResult();
}
2020-03-10 21:39:11 +00:00
RayResult RenderingManager::castCameraToViewportRay(const float nX, const float nY, float maxDistance, bool ignorePlayer, bool ignoreActors)
{
osg::ref_ptr<osgUtil::LineSegmentIntersector> intersector (new osgUtil::LineSegmentIntersector(osgUtil::LineSegmentIntersector::PROJECTION,
nX * 2.f - 1.f, nY * (-2.f) + 1.f));
osg::Vec3d dist (0.f, 0.f, -maxDistance);
dist = dist * mViewer->getCamera()->getProjectionMatrix();
osg::Vec3d end = intersector->getEnd();
end.z() = dist.z();
intersector->setEnd(end);
intersector->setIntersectionLimit(osgUtil::LineSegmentIntersector::LIMIT_NEAREST);
mViewer->getCamera()->accept(*getIntersectionVisitor(intersector, ignorePlayer, ignoreActors));
return getIntersectionResult(intersector);
}
2015-05-14 15:34:55 +00:00
void RenderingManager::updatePtr(const MWWorld::Ptr &old, const MWWorld::Ptr &updated)
{
mObjects->updatePtr(old, updated);
mActorsPaths->updatePtr(old, updated);
2015-05-14 15:34:55 +00:00
}
2016-09-14 14:18:29 +00:00
void RenderingManager::spawnEffect(const std::string &model, const std::string &texture, const osg::Vec3f &worldPosition, float scale, bool isMagicVFX)
2015-04-19 15:55:56 +00:00
{
2016-09-14 14:18:29 +00:00
mEffectManager->addEffect(model, texture, worldPosition, scale, isMagicVFX);
2015-04-19 15:55:56 +00:00
}
void RenderingManager::notifyWorldSpaceChanged()
{
mEffectManager->clear();
2020-02-19 20:26:42 +00:00
mWater->clearRipples();
2015-04-19 15:55:56 +00:00
}
void RenderingManager::clear()
{
2015-06-16 18:56:48 +00:00
mSky->setMoonColour(false);
2015-04-19 15:55:56 +00:00
notifyWorldSpaceChanged();
2015-04-18 23:57:52 +00:00
}
2015-04-25 13:19:17 +00:00
MWRender::Animation* RenderingManager::getAnimation(const MWWorld::Ptr &ptr)
{
2015-05-22 02:36:17 +00:00
if (mPlayerAnimation.get() && ptr == mPlayerAnimation->getPtr())
return mPlayerAnimation.get();
2015-04-25 13:19:17 +00:00
return mObjects->getAnimation(ptr);
}
2015-12-18 16:21:51 +00:00
const MWRender::Animation* RenderingManager::getAnimation(const MWWorld::ConstPtr &ptr) const
{
if (mPlayerAnimation.get() && ptr == mPlayerAnimation->getPtr())
return mPlayerAnimation.get();
return mObjects->getAnimation(ptr);
}
2015-05-01 16:21:50 +00:00
void RenderingManager::setupPlayer(const MWWorld::Ptr &player)
{
if (!mPlayerNode)
{
mPlayerNode = new SceneUtil::PositionAttitudeTransform;
mPlayerNode->setNodeMask(Mask_Player);
mPlayerNode->setName("Player Root");
2016-02-08 15:45:56 +00:00
mSceneRoot->addChild(mPlayerNode);
2015-05-01 16:21:50 +00:00
}
2015-05-24 02:00:35 +00:00
mPlayerNode->setUserDataContainer(new osg::DefaultUserDataContainer);
mPlayerNode->getUserDataContainer()->addUserObject(new PtrHolder(player));
2015-05-01 16:21:50 +00:00
2015-05-24 02:00:35 +00:00
player.getRefData().setBaseNode(mPlayerNode);
2015-06-16 18:36:48 +00:00
mWater->removeEmitter(player);
2015-06-16 18:36:48 +00:00
mWater->addEmitter(player);
2015-05-01 16:21:50 +00:00
}
void RenderingManager::renderPlayer(const MWWorld::Ptr &player)
{
2020-02-14 21:11:19 +00:00
#ifdef USE_OPENXR
2020-03-15 14:31:38 +00:00
MWVR::Environment::get().setPlayerAnimation(new MWVR::VRAnimation(player, player.getRefData().getBaseNode(), mResourceSystem, false, nullptr));
mPlayerAnimation = MWVR::Environment::get().getPlayerAnimation();
2020-02-14 21:11:19 +00:00
#else
mPlayerAnimation = new NpcAnimation(player, player.getRefData().getBaseNode(), mResourceSystem, 0, NpcAnimation::VM_Normal,
mFirstPersonFieldOfView);
2020-02-14 21:11:19 +00:00
#endif
2015-05-01 16:21:50 +00:00
2015-05-21 21:54:39 +00:00
mCamera->setAnimation(mPlayerAnimation.get());
mCamera->attachTo(player);
2015-05-01 16:21:50 +00:00
}
2015-05-21 21:54:39 +00:00
void RenderingManager::rebuildPtr(const MWWorld::Ptr &ptr)
{
2018-10-09 06:21:12 +00:00
NpcAnimation *anim = nullptr;
2015-05-21 21:54:39 +00:00
if(ptr == mPlayerAnimation->getPtr())
anim = mPlayerAnimation.get();
else
anim = dynamic_cast<NpcAnimation*>(mObjects->getAnimation(ptr));
if(anim)
{
anim->rebuild();
if(mCamera->getTrackingPtr() == ptr)
{
mCamera->attachTo(ptr);
mCamera->setAnimation(anim);
}
}
}
2015-06-16 18:36:48 +00:00
void RenderingManager::addWaterRippleEmitter(const MWWorld::Ptr &ptr)
{
mWater->addEmitter(ptr);
}
void RenderingManager::removeWaterRippleEmitter(const MWWorld::Ptr &ptr)
{
mWater->removeEmitter(ptr);
}
void RenderingManager::emitWaterRipple(const osg::Vec3f &pos)
{
mWater->emitRipple(pos);
}
void RenderingManager::updateProjectionMatrix()
{
2015-05-26 14:40:44 +00:00
double aspect = mViewer->getCamera()->getViewport()->aspectRatio();
float fov = mFieldOfView;
if (mFieldOfViewOverridden)
fov = mFieldOfViewOverride;
mViewer->getCamera()->setProjectionMatrixAsPerspective(fov, aspect, mNearClip, mViewDistance);
mUniformNear->set(mNearClip);
mUniformFar->set(mViewDistance);
// Since our fog is not radial yet, we should take FOV in account, otherwise terrain near viewing distance may disappear.
2019-03-03 17:29:51 +00:00
// Limit FOV here just for sure, otherwise viewing distance can be too high.
fov = std::min(mFieldOfView, 140.f);
float distanceMult = std::cos(osg::DegreesToRadians(fov)/2.f);
mTerrain->setViewDistance(mViewDistance * (distanceMult ? 1.f/distanceMult : 1.f));
2015-05-14 19:42:04 +00:00
}
void RenderingManager::updateTextureFiltering()
{
mViewer->stopThreading();
mResourceSystem->getSceneManager()->setFilterSettings(
Settings::Manager::getString("texture mag filter", "General"),
Settings::Manager::getString("texture min filter", "General"),
2015-12-14 00:51:27 +00:00
Settings::Manager::getString("texture mipmap", "General"),
Settings::Manager::getInt("anisotropy", "General")
2015-12-14 00:51:27 +00:00
);
mTerrain->updateTextureFiltering();
mViewer->startThreading();
}
2015-06-11 21:16:05 +00:00
void RenderingManager::updateAmbient()
{
osg::Vec4f color = mAmbientColor;
if (mNightEyeFactor > 0.f)
color += osg::Vec4f(0.7, 0.7, 0.7, 0.0) * mNightEyeFactor;
mStateUpdater->setAmbientColor(color);
}
2015-06-02 14:35:35 +00:00
void RenderingManager::setFogColor(const osg::Vec4f &color)
{
mViewer->getCamera()->setClearColor(color);
mStateUpdater->setFogColor(color);
}
2017-03-07 03:02:06 +00:00
void RenderingManager::reportStats() const
{
osg::Stats* stats = mViewer->getViewerStats();
unsigned int frameNumber = mViewer->getFrameStamp()->getFrameNumber();
if (stats->collectStats("resource"))
{
stats->setAttribute(frameNumber, "UnrefQueue", mUnrefQueue->getNumItems());
mTerrain->reportStats(frameNumber, stats);
}
}
void RenderingManager::processChangedSettings(const Settings::CategorySettingVector &changed)
{
for (Settings::CategorySettingVector::const_iterator it = changed.begin(); it != changed.end(); ++it)
{
if (it->first == "Camera" && it->second == "field of view")
{
mFieldOfView = Settings::Manager::getFloat("field of view", "Camera");
updateProjectionMatrix();
}
2015-05-24 00:34:20 +00:00
else if (it->first == "Camera" && it->second == "viewing distance")
{
2015-05-24 00:34:20 +00:00
mViewDistance = Settings::Manager::getFloat("viewing distance", "Camera");
if(!mDistantFog)
mStateUpdater->setFogEnd(mViewDistance);
updateProjectionMatrix();
}
else if (it->first == "General" && (it->second == "texture filter" ||
it->second == "texture mipmap" ||
it->second == "anisotropy"))
2015-05-14 19:42:04 +00:00
updateTextureFiltering();
2015-10-28 20:22:14 +00:00
else if (it->first == "Water")
mWater->processChangedSettings(changed);
}
}
2015-06-01 13:34:46 +00:00
float RenderingManager::getNearClipDistance() const
{
return mNearClip;
}
2015-06-02 23:18:36 +00:00
float RenderingManager::getTerrainHeightAt(const osg::Vec3f &pos)
{
return mTerrain->getHeightAt(pos);
}
2015-05-21 21:54:39 +00:00
bool RenderingManager::vanityRotateCamera(const float *rot)
{
if(!mCamera->isVanityOrPreviewModeEnabled())
return false;
2020-02-23 10:02:38 +00:00
mCamera->rotateCamera(rot[0], 0.f, rot[2], true);
2015-05-21 21:54:39 +00:00
return true;
}
void RenderingManager::setCameraDistance(float dist, bool adjust, bool override)
{
if(!mCamera->isVanityOrPreviewModeEnabled() && !mCamera->isFirstPerson())
{
if(mCamera->isNearest() && dist > 0.f)
mCamera->toggleViewMode();
else
mCamera->setCameraDistance(-dist / 120.f * 10, adjust, override);
}
else if(mCamera->isFirstPerson() && dist < 0.f)
{
mCamera->toggleViewMode();
mCamera->setCameraDistance(0.f, false, override);
}
}
void RenderingManager::resetCamera()
{
mCamera->reset();
}
float RenderingManager::getCameraDistance() const
{
return mCamera->getCameraDistance();
}
Camera* RenderingManager::getCamera()
{
return mCamera.get();
}
const osg::Vec3f &RenderingManager::getCameraPosition() const
{
return mCurrentCameraPos;
}
void RenderingManager::togglePOV(bool force)
2015-05-21 21:54:39 +00:00
{
mCamera->toggleViewMode(force);
2015-05-21 21:54:39 +00:00
}
void RenderingManager::togglePreviewMode(bool enable)
{
mCamera->togglePreviewMode(enable);
}
bool RenderingManager::toggleVanityMode(bool enable)
{
return mCamera->toggleVanityMode(enable);
}
void RenderingManager::allowVanityMode(bool allow)
{
mCamera->allowVanityMode(allow);
}
void RenderingManager::changeVanityModeScale(float factor)
{
if(mCamera->isVanityOrPreviewModeEnabled())
mCamera->setCameraDistance(-factor/120.f*10, true, true);
}
void RenderingManager::overrideFieldOfView(float val)
{
if (mFieldOfViewOverridden != true || mFieldOfViewOverride != val)
{
mFieldOfViewOverridden = true;
mFieldOfViewOverride = val;
updateProjectionMatrix();
}
}
osg::Vec3f RenderingManager::getHalfExtents(const MWWorld::ConstPtr& object) const
{
osg::Vec3f halfExtents(0, 0, 0);
std::string modelName = object.getClass().getModel(object);
if (modelName.empty())
return halfExtents;
osg::ref_ptr<const osg::Node> node = mResourceSystem->getSceneManager()->getTemplate(modelName);
osg::ComputeBoundsVisitor computeBoundsVisitor;
computeBoundsVisitor.setTraversalMask(~(MWRender::Mask_ParticleSystem|MWRender::Mask_Effect));
const_cast<osg::Node*>(node.get())->accept(computeBoundsVisitor);
osg::BoundingBox bounds = computeBoundsVisitor.getBoundingBox();
if (bounds.valid())
{
halfExtents[0] = std::abs(bounds.xMax() - bounds.xMin()) / 2.f;
halfExtents[1] = std::abs(bounds.yMax() - bounds.yMin()) / 2.f;
halfExtents[2] = std::abs(bounds.zMax() - bounds.zMin()) / 2.f;
}
return halfExtents;
}
void RenderingManager::resetFieldOfView()
{
if (mFieldOfViewOverridden == true)
{
mFieldOfViewOverridden = false;
updateProjectionMatrix();
}
}
void RenderingManager::exportSceneGraph(const MWWorld::Ptr &ptr, const std::string &filename, const std::string &format)
{
osg::Node* node = mViewer->getSceneData();
if (!ptr.isEmpty())
node = ptr.getRefData().getBaseNode();
SceneUtil::writeScene(node, filename, format);
}
LandManager *RenderingManager::getLandManager() const
{
return mTerrainStorage->getLandManager();
}
void RenderingManager::updateActorPath(const MWWorld::ConstPtr& actor, const std::deque<osg::Vec3f>& path,
const osg::Vec3f& halfExtents, const osg::Vec3f& start, const osg::Vec3f& end) const
{
mActorsPaths->update(actor, path, halfExtents, start, end, mNavigator.getSettings());
}
void RenderingManager::removeActorPath(const MWWorld::ConstPtr& actor) const
{
mActorsPaths->remove(actor);
}
void RenderingManager::setNavMeshNumber(const std::size_t value)
{
mNavMeshNumber = value;
}
2019-01-20 16:27:52 +00:00
void RenderingManager::updateNavMesh()
{
if (!mNavMesh->isEnabled())
return;
2019-01-20 16:27:52 +00:00
const auto navMeshes = mNavigator.getNavMeshes();
auto it = navMeshes.begin();
for (std::size_t i = 0; it != navMeshes.end() && i < mNavMeshNumber; ++i)
++it;
if (it == navMeshes.end())
{
mNavMesh->reset();
}
else
{
try
{
const auto locked = it->second->lockConst();
mNavMesh->update(locked->getImpl(), mNavMeshNumber, locked->getGeneration(),
2019-01-20 16:27:52 +00:00
locked->getNavMeshRevision(), mNavigator.getSettings());
}
catch (const std::exception& e)
{
Log(Debug::Error) << "NavMesh render update exception: " << e.what();
}
}
}
2019-11-27 22:45:01 +00:00
void RenderingManager::updateRecastMesh()
{
if (!mRecastMesh->isEnabled())
return;
mRecastMesh->update(mNavigator.getRecastMeshTiles(), mNavigator.getSettings());
}
}