1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-26 09:35:28 +00:00
OpenMW/apps/openmw/mwrender/renderingmanager.cpp

828 lines
26 KiB
C++
Raw Normal View History

2011-10-20 15:02:19 -04:00
#include "renderingmanager.hpp"
#include <stdexcept>
#include <osg/Light>
#include <osg/LightModel>
#include <osg/Fog>
2015-06-01 17:02:44 +02:00
#include <osg/PolygonMode>
#include <osg/Group>
2015-04-23 23:50:46 +02:00
#include <osg/PositionAttitudeTransform>
#include <osg/UserDataContainer>
#include <osg/ComputeBoundsVisitor>
#include <osgUtil/LineSegmentIntersector>
#include <osgUtil/IncrementalCompileOperation>
#include <osgViewer/Viewer>
2013-07-29 02:32:08 +02:00
2015-04-22 19:08:56 +02:00
#include <components/resource/resourcesystem.hpp>
2015-05-14 21:42:04 +02:00
#include <components/resource/texturemanager.hpp>
#include <components/resource/scenemanager.hpp>
2015-04-22 19:08:56 +02:00
#include <components/settings/settings.hpp>
#include <components/sceneutil/util.hpp>
#include <components/sceneutil/lightmanager.hpp>
#include <components/sceneutil/statesetupdater.hpp>
2015-06-03 01:18:36 +02:00
#include <components/terrain/terraingrid.hpp>
#include <components/esm/loadcell.hpp>
#include "sky.hpp"
2015-04-19 17:55:56 +02:00
#include "effectmanager.hpp"
2015-05-01 18:21:50 +02:00
#include "npcanimation.hpp"
#include "vismask.hpp"
#include "pathgrid.hpp"
2015-05-21 23:54:39 +02:00
#include "camera.hpp"
2015-06-02 16:35:35 +02:00
#include "water.hpp"
2015-06-03 01:18:36 +02:00
#include "terrainstorage.hpp"
namespace MWRender
{
class StateUpdater : public SceneUtil::StateSetUpdater
{
public:
StateUpdater()
: mFogEnd(0.f)
2015-06-01 17:02:44 +02: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;
fog->setStart(1);
2015-05-26 18:22:21 +02:00
fog->setMode(osg::Fog::LINEAR);
stateset->setAttributeAndModes(fog, osg::StateAttribute::ON);
2015-06-01 17:02:44 +02: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->setEnd(mFogEnd);
}
void setAmbientColor(const osg::Vec4f& col)
{
mAmbientColor = col;
}
void setFogColor(const osg::Vec4f& col)
{
mFogColor = col;
}
void setFogEnd(float end)
{
mFogEnd = end;
}
2015-06-01 17:02:44 +02:00
void setWireframe(bool wireframe)
{
if (mWireframe != wireframe)
{
mWireframe = wireframe;
reset();
}
}
bool getWireframe() const
{
return mWireframe;
}
private:
osg::Vec4f mAmbientColor;
osg::Vec4f mFogColor;
float mFogEnd;
2015-06-01 17:02:44 +02:00
bool mWireframe;
};
2015-06-16 20:36:48 +02:00
RenderingManager::RenderingManager(osgViewer::Viewer* viewer, osg::ref_ptr<osg::Group> rootNode, Resource::ResourceSystem* resourceSystem, const MWWorld::Fallback* fallback)
: mViewer(viewer)
, mRootNode(rootNode)
, mResourceSystem(resourceSystem)
2015-06-11 23:16:05 +02:00
, mNightEyeFactor(0.f)
{
osg::ref_ptr<SceneUtil::LightManager> lightRoot = new SceneUtil::LightManager;
2015-05-01 18:21:50 +02:00
mLightRoot = lightRoot;
lightRoot->setStartLight(1);
mRootNode->addChild(lightRoot);
2011-11-03 23:47:15 -04:00
mPathgrid.reset(new Pathgrid(mRootNode));
2015-05-02 22:45:27 +02:00
mObjects.reset(new Objects(mResourceSystem, lightRoot));
2012-02-26 13:13:29 +01:00
2015-05-14 21:42:04 +02:00
mViewer->setIncrementalCompileOperation(new osgUtil::IncrementalCompileOperation);
mResourceSystem->getSceneManager()->setIncrementalCompileOperation(mViewer->getIncrementalCompileOperation());
2015-06-02 16:35:35 +02:00
mEffectManager.reset(new EffectManager(lightRoot, mResourceSystem));
2015-06-16 20:36:48 +02:00
mWater.reset(new Water(lightRoot, mResourceSystem, mViewer->getIncrementalCompileOperation(), fallback));
2015-06-03 01:18:36 +02:00
mTerrain.reset(new Terrain::TerrainGrid(lightRoot, mResourceSystem, mViewer->getIncrementalCompileOperation(),
new TerrainStorage(mResourceSystem->getVFS(), false), Mask_Terrain));
2015-04-19 17:55:56 +02:00
2015-05-21 23:54:39 +02:00
mCamera.reset(new Camera(mViewer->getCamera()));
2015-05-14 21:42:04 +02:00
mViewer->setLightingMode(osgViewer::View::NO_LIGHT);
2011-11-03 23:47:15 -04:00
osg::ref_ptr<osg::LightSource> source = new osg::LightSource;
2015-05-26 16:40:44 +02:00
source->setNodeMask(SceneUtil::Mask_Lit);
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 18:22:21 +02:00
mSunLight->setSpecular(osg::Vec4f(0,0,0,0));
mSunLight->setConstantAttenuation(1.f);
lightRoot->addChild(source);
2011-11-03 23:47:15 -04:00
2015-04-24 21:55:30 +02:00
lightRoot->getOrCreateStateSet()->setMode(GL_CULL_FACE, osg::StateAttribute::ON);
lightRoot->getOrCreateStateSet()->setMode(GL_LIGHTING, osg::StateAttribute::ON);
lightRoot->getOrCreateStateSet()->setMode(GL_NORMALIZE, osg::StateAttribute::ON);
lightRoot->setNodeMask(Mask_Scene);
2015-05-26 16:40:44 +02:00
lightRoot->setName("Scene Root");
mSky.reset(new SkyManager(lightRoot, resourceSystem->getSceneManager()));
source->setStateSetModes(*mRootNode->getOrCreateStateSet(), osg::StateAttribute::ON);
mStateUpdater = new StateUpdater;
lightRoot->addUpdateCallback(mStateUpdater);
osg::Camera::CullingMode cullingMode = osg::Camera::DEFAULT_CULLING|osg::Camera::FAR_PLANE_CULLING;
2011-11-03 23:47:15 -04:00
2015-05-24 02:34:20 +02:00
if (!Settings::Manager::getBool("small feature culling", "Camera"))
cullingMode &= ~(osg::CullStack::SMALL_FEATURE_CULLING);
else
cullingMode |= osg::CullStack::SMALL_FEATURE_CULLING;
2015-05-14 21:42:04 +02:00
mViewer->getCamera()->setCullingMode( cullingMode );
2015-05-14 21:42:04 +02:00
mViewer->getCamera()->setComputeNearFarMode(osg::Camera::DO_NOT_COMPUTE_NEAR_FAR);
mViewer->getCamera()->setCullingMode(cullingMode);
2015-05-20 03:35:52 +02:00
mViewer->getCamera()->setCullMask(~(Mask_UpdateVisitor));
2015-05-24 02:34:20 +02:00
mNearClip = Settings::Manager::getFloat("near clip", "Camera");
mViewDistance = Settings::Manager::getFloat("viewing distance", "Camera");
mFieldOfView = Settings::Manager::getFloat("field of view", "General");
updateProjectionMatrix();
2015-06-02 16:35:35 +02:00
mStateUpdater->setFogEnd(mViewDistance);
}
RenderingManager::~RenderingManager()
{
}
MWRender::Objects& RenderingManager::getObjects()
{
return *mObjects.get();
}
Resource::ResourceSystem* RenderingManager::getResourceSystem()
{
return mResourceSystem;
}
2015-06-11 23:16:05 +02:00
void RenderingManager::setNightEyeFactor(float factor)
{
if (factor != mNightEyeFactor)
{
mNightEyeFactor = factor;
updateAmbient();
}
}
void RenderingManager::setAmbientColour(const osg::Vec4f &colour)
{
2015-06-11 23:16:05 +02:00
mAmbientColor = colour;
updateAmbient();
}
2015-06-16 20:56:48 +02: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-23 01:32:36 +02:00
{
setAmbientColour(SceneUtil::colourFromRGB(cell->mAmbi.mAmbient));
mSunLight->setDiffuse(SceneUtil::colourFromRGB(cell->mAmbi.mSunlight));
mSunLight->setDirection(osg::Vec3f(1.f,-1.f,-1.f));
}
void RenderingManager::setSunColour(const osg::Vec4f &colour)
{
2015-05-26 18:22:21 +02:00
// need to wrap this in a StateUpdater?
mSunLight->setDiffuse(colour);
}
void RenderingManager::setSunDirection(const osg::Vec3f &direction)
{
2015-05-20 02:07:18 +02:00
osg::Vec3 position = direction * -1;
2015-05-26 18:22:21 +02:00
// need to wrap this in a StateUpdater?
2015-05-20 02:07:18 +02:00
mSunLight->setPosition(osg::Vec4(position.x(), position.y(), position.z(), 0));
2015-05-20 02:07:18 +02:00
mSky->setSunDirection(position);
}
osg::Vec3f RenderingManager::getEyePos()
{
2015-05-14 21:42:04 +02:00
osg::Vec3d eye = mViewer->getCameraManipulator()->getMatrix().getTrans();
return eye;
2012-05-23 01:32:36 +02:00
}
2012-05-29 06:45:44 +02:00
2015-05-02 22:45:27 +02:00
void RenderingManager::addCell(const MWWorld::CellStore *store)
{
mPathgrid->addCell(store);
2015-06-02 16:35:35 +02:00
mWater->changeCell(store);
2015-06-03 01:18:36 +02:00
if (store->getCell()->isExterior())
mTerrain->loadCell(store->getCell()->getGridX(), store->getCell()->getGridY());
2015-05-02 22:45:27 +02:00
}
void RenderingManager::removeCell(const MWWorld::CellStore *store)
2013-08-20 09:52:27 +02:00
{
mPathgrid->removeCell(store);
mObjects->removeCell(store);
2015-06-03 01:18:36 +02:00
if (store->getCell()->isExterior())
mTerrain->unloadCell(store->getCell()->getGridX(), store->getCell()->getGridY());
2015-06-16 20:36:48 +02:00
mWater->removeCell(store);
2013-08-20 09:52:27 +02:00
}
2012-05-23 01:32:36 +02:00
void RenderingManager::setSkyEnabled(bool enabled)
{
mSky->setEnabled(enabled);
}
2015-05-02 22:45:27 +02:00
bool RenderingManager::toggleRenderMode(RenderMode mode)
{
if (mode == Render_CollisionDebug || mode == Render_Pathgrid)
return mPathgrid->toggleRenderMode(mode);
2015-05-02 22:45:27 +02:00
else if (mode == Render_Wireframe)
{
2015-06-01 17:02:44 +02:00
bool wireframe = !mStateUpdater->getWireframe();
mStateUpdater->setWireframe(wireframe);
return wireframe;
2015-05-02 22:45:27 +02:00
}
2015-06-02 16:35:35 +02:00
else if (mode == Render_Water)
{
return mWater->toggle();
}
else if (mode == Render_Scene)
{
int mask = mViewer->getCamera()->getCullMask();
bool enabled = mask&Mask_Scene;
enabled = !enabled;
if (enabled)
mask |= Mask_Scene;
else
mask &= ~Mask_Scene;
mViewer->getCamera()->setCullMask(mask);
return enabled;
}
2015-05-02 22:45:27 +02:00
/*
else //if (mode == Render_BoundingBoxes)
{
bool show = !mRendering.getScene()->getShowBoundingBoxes();
mRendering.getScene()->showBoundingBoxes(show);
return show;
}
*/
return false;
}
void RenderingManager::configureFog(const ESM::Cell *cell)
{
osg::Vec4f color = SceneUtil::colourFromRGB(cell->mAmbi.mFog);
configureFog (cell->mAmbi.mFogDensity, color);
}
2015-06-02 16:35:35 +02:00
void RenderingManager::configureFog(float /* fogDepth */, const osg::Vec4f &color)
{
2015-06-02 16:35:35 +02:00
mFogColor = color;
}
SkyManager* RenderingManager::getSkyManager()
{
return mSky.get();
}
2015-04-19 01:57:52 +02:00
void RenderingManager::update(float dt, bool paused)
{
2015-04-19 17:55:56 +02:00
mEffectManager->update(dt);
2015-04-19 20:07:18 +02:00
mSky->update(dt);
2015-06-16 20:36:48 +02:00
mWater->update(dt);
2015-05-22 01:43:16 +02:00
mCamera->update(dt, paused);
2015-06-02 16:35:35 +02:00
osg::Vec3f focal, cameraPos;
mCamera->getPosition(focal, cameraPos);
if (mWater->isUnderwater(cameraPos))
{
setFogColor(osg::Vec4f(0.090195f, 0.115685f, 0.12745f, 1.f));
mStateUpdater->setFogEnd(1000);
}
else
{
setFogColor(mFogColor);
mStateUpdater->setFogEnd(mViewDistance);
}
2015-04-19 17:55:56 +02:00
}
2015-05-21 23:54:39 +02:00
void RenderingManager::updatePlayerPtr(const MWWorld::Ptr &ptr)
{
if(mPlayerAnimation.get())
mPlayerAnimation->updatePtr(ptr);
mCamera->attachTo(ptr);
}
2015-06-16 20:36:48 +02:00
void RenderingManager::removePlayer(const MWWorld::Ptr &player)
{
mWater->removeEmitter(player);
}
2015-04-23 23:50:46 +02:00
void RenderingManager::rotateObject(const MWWorld::Ptr &ptr, const osg::Quat& rot)
{
2015-05-21 23:54:39 +02:00
if(ptr == mCamera->getTrackingPtr() &&
!mCamera->isVanityOrPreviewModeEnabled())
{
mCamera->rotateCamera(-ptr.getRefData().getPosition().rot[0], -ptr.getRefData().getPosition().rot[2], false);
}
2015-04-23 23:50:46 +02: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);
}
2015-05-22 00:55:43 +02:00
void RenderingManager::removeObject(const MWWorld::Ptr &ptr)
{
mObjects->removeObject(ptr);
2015-06-16 20:36:48 +02:00
mWater->removeEmitter(ptr);
2015-05-22 00:55:43 +02:00
}
2015-06-02 16:35:35 +02:00
void RenderingManager::setWaterEnabled(bool enabled)
{
mWater->setEnabled(enabled);
}
void RenderingManager::setWaterHeight(float height)
{
mWater->setHeight(height);
}
2015-06-03 16:40:16 +02:00
class NotifyDrawCompletedCallback : public osg::Camera::DrawCallback
{
public:
virtual void operator () (osg::RenderInfo& renderInfo) const
{
mCondition.signal();
}
mutable OpenThreads::Condition mCondition;
};
void RenderingManager::screenshot(osg::Image *image, int w, int h)
{
int oldCullMask = mViewer->getCamera()->getCullMask();
mViewer->getCamera()->setCullMask(oldCullMask & (~Mask_GUI));
osg::ref_ptr<osg::Camera> rttCamera (new osg::Camera);
rttCamera->setNodeMask(Mask_RenderToTexture);
rttCamera->attach(osg::Camera::COLOR_BUFFER, image);
rttCamera->setRenderOrder(osg::Camera::PRE_RENDER);
rttCamera->setReferenceFrame(osg::Camera::ABSOLUTE_RF);
rttCamera->setRenderTargetImplementation(osg::Camera::FRAME_BUFFER_OBJECT, osg::Camera::PIXEL_BUFFER_RTT);
rttCamera->setClearColor(mViewer->getCamera()->getClearColor());
rttCamera->setClearMask(mViewer->getCamera()->getClearMask());
rttCamera->setProjectionMatrixAsPerspective(mFieldOfView, w/float(h), mNearClip, mViewDistance);
rttCamera->setViewMatrix(mViewer->getCamera()->getViewMatrix());
rttCamera->setViewport(0, 0, w, h);
rttCamera->setGraphicsContext(mViewer->getCamera()->getGraphicsContext());
osg::ref_ptr<osg::Texture2D> texture (new osg::Texture2D);
texture->setInternalFormat(GL_RGB);
texture->setTextureSize(w, h);
texture->setResizeNonPowerOfTwoHint(false);
rttCamera->attach(osg::Camera::COLOR_BUFFER, texture);
image->setDataType(GL_UNSIGNED_BYTE);
image->setPixelFormat(texture->getInternalFormat());
rttCamera->addChild(mLightRoot);
mRootNode->addChild(rttCamera);
mViewer->frame(mViewer->getFrameStamp()->getSimulationTime());
2015-06-03 16:40:16 +02:00
// The draw needs to complete before we can copy back our image.
osg::ref_ptr<NotifyDrawCompletedCallback> callback (new NotifyDrawCompletedCallback);
rttCamera->setFinalDrawCallback(callback);
OpenThreads::Mutex m;
m.lock();
callback->mCondition.wait(&m);
m.unlock();
rttCamera->removeChildren(0, rttCamera->getNumChildren());
rttCamera->setGraphicsContext(NULL);
mRootNode->removeChild(rttCamera);
mViewer->getCamera()->setCullMask(oldCullMask);
}
osg::Vec4f RenderingManager::getScreenBounds(const MWWorld::Ptr& ptr)
{
if (!ptr.getRefData().getBaseNode())
return osg::Vec4f();
osg::ComputeBoundsVisitor computeBoundsVisitor;
computeBoundsVisitor.setTraversalMask(~MWRender::Mask_ParticleSystem);
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);
}
RenderingManager::RayResult getIntersectionResult (osgUtil::LineSegmentIntersector* intersector)
{
RenderingManager::RayResult result;
result.mHit = false;
if (intersector->containsIntersections())
{
result.mHit = true;
osgUtil::LineSegmentIntersector::Intersection intersection = intersector->getFirstIntersection();
result.mHitPointWorld = intersection.getWorldIntersectPoint();
result.mHitNormalWorld = intersection.getWorldIntersectNormal();
PtrHolder* ptrHolder = NULL;
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;
}
RenderingManager::RayResult RenderingManager::castRay(const osg::Vec3f& origin, const osg::Vec3f& dest, bool ignorePlayer, bool ignoreActors)
{
osg::ref_ptr<osgUtil::LineSegmentIntersector> intersector (new osgUtil::LineSegmentIntersector(osgUtil::LineSegmentIntersector::MODEL,
origin, dest));
intersector->setIntersectionLimit(osgUtil::LineSegmentIntersector::LIMIT_NEAREST);
osgUtil::IntersectionVisitor intersectionVisitor(intersector);
int mask = intersectionVisitor.getTraversalMask();
mask &= ~(Mask_RenderToTexture|Mask_Sky|Mask_Debug|Mask_Effect|Mask_Water);
if (ignorePlayer)
mask &= ~(Mask_Player);
if (ignoreActors)
mask &= ~(Mask_Actor|Mask_Player);
intersectionVisitor.setTraversalMask(mask);
mRootNode->accept(intersectionVisitor);
return getIntersectionResult(intersector);
}
RenderingManager::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);
osgUtil::IntersectionVisitor intersectionVisitor(intersector);
2015-05-26 16:40:44 +02:00
int mask = intersectionVisitor.getTraversalMask();
2015-06-02 16:35:35 +02:00
mask &= ~(Mask_RenderToTexture|Mask_Sky|Mask_Debug|Mask_Effect|Mask_Water);
if (ignorePlayer)
2015-05-26 16:40:44 +02:00
mask &= ~(Mask_Player);
if (ignoreActors)
mask &= ~(Mask_Actor|Mask_Player);
2015-05-26 16:40:44 +02:00
intersectionVisitor.setTraversalMask(mask);
mViewer->getCamera()->accept(intersectionVisitor);
return getIntersectionResult(intersector);
}
2015-05-14 17:34:55 +02:00
void RenderingManager::updatePtr(const MWWorld::Ptr &old, const MWWorld::Ptr &updated)
{
mObjects->updatePtr(old, updated);
}
2015-04-19 17:55:56 +02:00
void RenderingManager::spawnEffect(const std::string &model, const std::string &texture, const osg::Vec3f &worldPosition, float scale)
{
mEffectManager->addEffect(model, texture, worldPosition, scale);
}
void RenderingManager::notifyWorldSpaceChanged()
{
mEffectManager->clear();
2015-06-16 20:36:48 +02:00
mWater->clearRipples();
2015-04-19 17:55:56 +02:00
}
void RenderingManager::clear()
{
2015-06-16 20:56:48 +02:00
mSky->setMoonColour(false);
2015-04-19 17:55:56 +02:00
notifyWorldSpaceChanged();
2015-04-19 01:57:52 +02:00
}
2015-04-25 15:19:17 +02:00
MWRender::Animation* RenderingManager::getAnimation(const MWWorld::Ptr &ptr)
{
2015-05-22 04:36:17 +02:00
if (mPlayerAnimation.get() && ptr == mPlayerAnimation->getPtr())
return mPlayerAnimation.get();
2015-04-25 15:19:17 +02:00
return mObjects->getAnimation(ptr);
}
2015-05-01 18:21:50 +02:00
MWRender::Animation* RenderingManager::getPlayerAnimation()
{
return mPlayerAnimation.get();
}
void RenderingManager::setupPlayer(const MWWorld::Ptr &player)
{
if (!mPlayerNode)
{
mPlayerNode = new osg::PositionAttitudeTransform;
mPlayerNode->setNodeMask(Mask_Player);
2015-05-01 18:21:50 +02:00
mLightRoot->addChild(mPlayerNode);
}
2015-05-24 04:00:35 +02:00
mPlayerNode->setUserDataContainer(new osg::DefaultUserDataContainer);
mPlayerNode->getUserDataContainer()->addUserObject(new PtrHolder(player));
2015-05-01 18:21:50 +02:00
2015-05-24 04:00:35 +02:00
player.getRefData().setBaseNode(mPlayerNode);
2015-06-16 20:36:48 +02:00
mWater->addEmitter(player);
2015-05-01 18:21:50 +02:00
}
void RenderingManager::renderPlayer(const MWWorld::Ptr &player)
{
mPlayerAnimation.reset(new NpcAnimation(player, player.getRefData().getBaseNode(), mResourceSystem, 0));
2015-05-21 23:54:39 +02:00
mCamera->setAnimation(mPlayerAnimation.get());
mCamera->attachTo(player);
2015-05-01 18:21:50 +02:00
}
2015-05-21 23:54:39 +02:00
void RenderingManager::rebuildPtr(const MWWorld::Ptr &ptr)
{
NpcAnimation *anim = NULL;
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 20:36:48 +02:00
void RenderingManager::addWaterRippleEmitter(const MWWorld::Ptr &ptr)
{
mWater->addEmitter(ptr);
}
void RenderingManager::removeWaterRippleEmitter(const MWWorld::Ptr &ptr)
{
mWater->removeEmitter(ptr);
}
void RenderingManager::updateProjectionMatrix()
{
2015-05-26 16:40:44 +02:00
double aspect = mViewer->getCamera()->getViewport()->aspectRatio();
mViewer->getCamera()->setProjectionMatrixAsPerspective(mFieldOfView, aspect, mNearClip, mViewDistance);
2015-05-14 21:42:04 +02:00
}
void RenderingManager::updateTextureFiltering()
{
osg::Texture::FilterMode min = osg::Texture::LINEAR_MIPMAP_NEAREST;
osg::Texture::FilterMode mag = osg::Texture::LINEAR;
if (Settings::Manager::getString("texture filtering", "General") == "trilinear")
min = osg::Texture::LINEAR_MIPMAP_LINEAR;
int maxAnisotropy = Settings::Manager::getInt("anisotropy", "General");
mViewer->stopThreading();
mResourceSystem->getTextureManager()->setFilterSettings(min, mag, maxAnisotropy);
mViewer->startThreading();
}
2015-06-11 23:16:05 +02: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 16:35:35 +02:00
void RenderingManager::setFogColor(const osg::Vec4f &color)
{
mViewer->getCamera()->setClearColor(color);
mStateUpdater->setFogColor(color);
}
void RenderingManager::processChangedSettings(const Settings::CategorySettingVector &changed)
{
for (Settings::CategorySettingVector::const_iterator it = changed.begin(); it != changed.end(); ++it)
{
if (it->first == "General" && it->second == "field of view")
{
mFieldOfView = Settings::Manager::getFloat("field of view", "General");
updateProjectionMatrix();
}
2015-05-24 02:34:20 +02:00
else if (it->first == "Camera" && it->second == "viewing distance")
{
2015-05-24 02:34:20 +02:00
mViewDistance = Settings::Manager::getFloat("viewing distance", "Camera");
2015-05-22 01:43:04 +02:00
mStateUpdater->setFogEnd(mViewDistance);
updateProjectionMatrix();
}
2015-05-14 21:42:04 +02:00
else if (it->first == "General" && (it->second == "texture filtering" || it->second == "anisotropy"))
updateTextureFiltering();
}
}
2015-06-01 15:34:46 +02:00
float RenderingManager::getNearClipDistance() const
{
return mNearClip;
}
2015-06-03 01:18:36 +02:00
float RenderingManager::getTerrainHeightAt(const osg::Vec3f &pos)
{
return mTerrain->getHeightAt(pos);
}
2015-05-21 23:54:39 +02:00
bool RenderingManager::vanityRotateCamera(const float *rot)
{
if(!mCamera->isVanityOrPreviewModeEnabled())
return false;
mCamera->rotateCamera(rot[0], rot[2], true);
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();
}
void RenderingManager::togglePOV()
{
mCamera->toggleViewMode();
}
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::togglePlayerLooking(bool enable)
{
mCamera->togglePlayerLooking(enable);
}
void RenderingManager::changeVanityModeScale(float factor)
{
if(mCamera->isVanityOrPreviewModeEnabled())
mCamera->setCameraDistance(-factor/120.f*10, true, true);
}
}