mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-26 09:35:28 +00:00
ed0c31b485
The logic is now handled by RenderingManager, and Camera sets its animation's first person offset height. Due to how NpcAnimation seems to be updated, it has to be the one to actually set its own nodes, in the case of the hands. Otherwise, the hands would not move without a messier hack.
1048 lines
33 KiB
C++
1048 lines
33 KiB
C++
#include "renderingmanager.hpp"
|
|
|
|
#include <cassert>
|
|
|
|
#include <OgreRoot.h>
|
|
#include <OgreRenderWindow.h>
|
|
#include <OgreSceneManager.h>
|
|
#include <OgreViewport.h>
|
|
#include <OgreCamera.h>
|
|
#include <OgreTextureManager.h>
|
|
#include <OgreCompositorManager.h>
|
|
#include <OgreCompositorChain.h>
|
|
#include <OgreCompositionTargetPass.h>
|
|
#include <OgreCompositionPass.h>
|
|
#include <OgreHardwarePixelBuffer.h>
|
|
#include <OgreControllerManager.h>
|
|
#include <OgreMeshManager.h>
|
|
|
|
#include <SDL_video.h>
|
|
|
|
#include <extern/shiny/Main/Factory.hpp>
|
|
#include <extern/shiny/Platforms/Ogre/OgrePlatform.hpp>
|
|
|
|
#include <openengine/bullet/physic.hpp>
|
|
|
|
#include <components/esm/loadstat.hpp>
|
|
#include <components/settings/settings.hpp>
|
|
#include <components/terrain/world.hpp>
|
|
|
|
#include "../mwworld/esmstore.hpp"
|
|
#include "../mwworld/class.hpp"
|
|
|
|
#include "../mwbase/world.hpp" // these includes can be removed once the static-hack is gone
|
|
#include "../mwbase/environment.hpp"
|
|
#include "../mwbase/inputmanager.hpp" // FIXME
|
|
#include "../mwbase/windowmanager.hpp" // FIXME
|
|
|
|
#include "../mwmechanics/creaturestats.hpp"
|
|
|
|
#include "../mwworld/ptr.hpp"
|
|
#include "../mwworld/player.hpp"
|
|
|
|
#include "shadows.hpp"
|
|
#include "localmap.hpp"
|
|
#include "water.hpp"
|
|
#include "compositors.hpp"
|
|
#include "npcanimation.hpp"
|
|
#include "externalrendering.hpp"
|
|
#include "globalmap.hpp"
|
|
#include "videoplayer.hpp"
|
|
#include "terrainstorage.hpp"
|
|
|
|
using namespace MWRender;
|
|
using namespace Ogre;
|
|
|
|
namespace MWRender {
|
|
|
|
RenderingManager::RenderingManager(OEngine::Render::OgreRenderer& _rend, const boost::filesystem::path& resDir,
|
|
const boost::filesystem::path& cacheDir, OEngine::Physic::PhysicEngine* engine,
|
|
MWWorld::Fallback* fallback)
|
|
: mRendering(_rend)
|
|
, mFallback(fallback)
|
|
, mObjects(mRendering)
|
|
, mActors(mRendering, this)
|
|
, mPlayerAnimation(NULL)
|
|
, mAmbientMode(0)
|
|
, mSunEnabled(0)
|
|
, mPhysicsEngine(engine)
|
|
, mTerrain(NULL)
|
|
{
|
|
// select best shader mode
|
|
bool openGL = (Ogre::Root::getSingleton ().getRenderSystem ()->getName().find("OpenGL") != std::string::npos);
|
|
bool glES = (Ogre::Root::getSingleton ().getRenderSystem ()->getName().find("OpenGL ES") != std::string::npos);
|
|
|
|
// glsl is only supported in opengl mode and hlsl only in direct3d mode.
|
|
std::string currentMode = Settings::Manager::getString("shader mode", "General");
|
|
if (currentMode == ""
|
|
|| (openGL && currentMode == "hlsl")
|
|
|| (!openGL && currentMode == "glsl")
|
|
|| (glES && currentMode != "glsles"))
|
|
{
|
|
Settings::Manager::setString("shader mode", "General", openGL ? (glES ? "glsles" : "glsl") : "hlsl");
|
|
}
|
|
|
|
mRendering.adjustCamera(Settings::Manager::getFloat("field of view", "General"), 5);
|
|
|
|
mRendering.getWindow()->addListener(this);
|
|
mRendering.setWindowListener(this);
|
|
|
|
mCompositors = new Compositors(mRendering.getViewport());
|
|
|
|
mWater = 0;
|
|
|
|
// material system
|
|
sh::OgrePlatform* platform = new sh::OgrePlatform("General", (resDir / "materials").string());
|
|
if (!boost::filesystem::exists (cacheDir))
|
|
boost::filesystem::create_directories (cacheDir);
|
|
platform->setCacheFolder (cacheDir.string());
|
|
mFactory = new sh::Factory(platform);
|
|
|
|
sh::Language lang;
|
|
std::string l = Settings::Manager::getString("shader mode", "General");
|
|
if (l == "glsl")
|
|
lang = sh::Language_GLSL;
|
|
else if (l == "glsles")
|
|
lang = sh::Language_GLSLES;
|
|
else if (l == "hlsl")
|
|
lang = sh::Language_HLSL;
|
|
else
|
|
lang = sh::Language_CG;
|
|
mFactory->setCurrentLanguage (lang);
|
|
mFactory->setWriteSourceCache (true);
|
|
mFactory->setReadSourceCache (true);
|
|
mFactory->setReadMicrocodeCache (true);
|
|
mFactory->setWriteMicrocodeCache (true);
|
|
|
|
mFactory->loadAllFiles();
|
|
|
|
// Set default mipmap level (NB some APIs ignore this)
|
|
// Mipmap generation is currently disabled because it causes issues on Intel/AMD
|
|
//TextureManager::getSingleton().setDefaultNumMipmaps(Settings::Manager::getInt("num mipmaps", "General"));
|
|
TextureManager::getSingleton().setDefaultNumMipmaps(0);
|
|
|
|
// Set default texture filtering options
|
|
TextureFilterOptions tfo;
|
|
std::string filter = Settings::Manager::getString("texture filtering", "General");
|
|
if (filter == "anisotropic") tfo = TFO_ANISOTROPIC;
|
|
else if (filter == "trilinear") tfo = TFO_TRILINEAR;
|
|
else if (filter == "bilinear") tfo = TFO_BILINEAR;
|
|
else /*if (filter == "none")*/ tfo = TFO_NONE;
|
|
|
|
MaterialManager::getSingleton().setDefaultTextureFiltering(tfo);
|
|
MaterialManager::getSingleton().setDefaultAnisotropy( (filter == "anisotropic") ? Settings::Manager::getInt("anisotropy", "General") : 1 );
|
|
|
|
Ogre::TextureManager::getSingleton().setMemoryBudget(126*1024*1024);
|
|
Ogre::MeshManager::getSingleton().setMemoryBudget(64*1024*1024);
|
|
|
|
Ogre::ResourceGroupManager::getSingleton().initialiseAllResourceGroups();
|
|
|
|
// disable unsupported effects
|
|
if (!Settings::Manager::getBool("shaders", "Objects"))
|
|
Settings::Manager::setBool("enabled", "Shadows", false);
|
|
|
|
sh::Factory::getInstance ().setShadersEnabled (Settings::Manager::getBool("shaders", "Objects"));
|
|
|
|
sh::Factory::getInstance ().setGlobalSetting ("fog", "true");
|
|
sh::Factory::getInstance ().setGlobalSetting ("num_lights", Settings::Manager::getString ("num lights", "Objects"));
|
|
sh::Factory::getInstance ().setGlobalSetting ("simple_water", Settings::Manager::getBool("shader", "Water") ? "false" : "true");
|
|
sh::Factory::getInstance ().setGlobalSetting ("render_refraction", "false");
|
|
|
|
sh::Factory::getInstance ().setSharedParameter ("waterEnabled", sh::makeProperty<sh::FloatValue> (new sh::FloatValue(0.0)));
|
|
sh::Factory::getInstance ().setSharedParameter ("waterLevel", sh::makeProperty<sh::FloatValue>(new sh::FloatValue(0)));
|
|
sh::Factory::getInstance ().setSharedParameter ("waterTimer", sh::makeProperty<sh::FloatValue>(new sh::FloatValue(0)));
|
|
sh::Factory::getInstance ().setSharedParameter ("windDir_windSpeed", sh::makeProperty<sh::Vector3>(new sh::Vector3(0.5, -0.8, 0.2)));
|
|
sh::Factory::getInstance ().setSharedParameter ("waterSunFade_sunHeight", sh::makeProperty<sh::Vector2>(new sh::Vector2(1, 0.6)));
|
|
sh::Factory::getInstance ().setGlobalSetting ("refraction", Settings::Manager::getBool("refraction", "Water") ? "true" : "false");
|
|
sh::Factory::getInstance ().setGlobalSetting ("viewproj_fix", "false");
|
|
sh::Factory::getInstance ().setSharedParameter ("vpRow2Fix", sh::makeProperty<sh::Vector4> (new sh::Vector4(0,0,0,0)));
|
|
|
|
applyCompositors();
|
|
|
|
mRootNode = mRendering.getScene()->getRootSceneNode();
|
|
mRootNode->createChildSceneNode("player");
|
|
|
|
mObjects.setRootNode(mRootNode);
|
|
mActors.setRootNode(mRootNode);
|
|
|
|
mCamera = new MWRender::Camera(mRendering.getCamera());
|
|
|
|
mShadows = new Shadows(&mRendering);
|
|
|
|
mSkyManager = new SkyManager(mRootNode, mRendering.getCamera());
|
|
|
|
mOcclusionQuery = new OcclusionQuery(&mRendering, mSkyManager->getSunNode());
|
|
|
|
mVideoPlayer = new VideoPlayer(mRendering.getScene (), mRendering.getWindow());
|
|
mVideoPlayer->setResolution (Settings::Manager::getInt ("resolution x", "Video"), Settings::Manager::getInt ("resolution y", "Video"));
|
|
|
|
mSun = 0;
|
|
|
|
mDebugging = new Debugging(mRootNode, engine);
|
|
mLocalMap = new MWRender::LocalMap(&mRendering, this);
|
|
|
|
mWater = new MWRender::Water(mRendering.getCamera(), this);
|
|
|
|
setMenuTransparency(Settings::Manager::getFloat("menu transparency", "GUI"));
|
|
}
|
|
|
|
RenderingManager::~RenderingManager ()
|
|
{
|
|
mRendering.getWindow()->removeListener(this);
|
|
|
|
delete mPlayerAnimation;
|
|
delete mCamera;
|
|
delete mSkyManager;
|
|
delete mDebugging;
|
|
delete mShadows;
|
|
delete mTerrain;
|
|
delete mLocalMap;
|
|
delete mOcclusionQuery;
|
|
delete mCompositors;
|
|
delete mWater;
|
|
delete mVideoPlayer;
|
|
delete mFactory;
|
|
}
|
|
|
|
MWRender::SkyManager* RenderingManager::getSkyManager()
|
|
{
|
|
return mSkyManager;
|
|
}
|
|
|
|
MWRender::Objects& RenderingManager::getObjects(){
|
|
return mObjects;
|
|
}
|
|
MWRender::Actors& RenderingManager::getActors(){
|
|
return mActors;
|
|
}
|
|
|
|
OEngine::Render::Fader* RenderingManager::getFader()
|
|
{
|
|
return mRendering.getFader();
|
|
}
|
|
|
|
void RenderingManager::removeCell (MWWorld::Ptr::CellStore *store)
|
|
{
|
|
mObjects.removeCell(store);
|
|
mActors.removeCell(store);
|
|
mDebugging->cellRemoved(store);
|
|
}
|
|
|
|
void RenderingManager::removeWater ()
|
|
{
|
|
mWater->setActive(false);
|
|
}
|
|
|
|
void RenderingManager::toggleWater()
|
|
{
|
|
mWater->toggle();
|
|
}
|
|
|
|
void RenderingManager::cellAdded (MWWorld::Ptr::CellStore *store)
|
|
{
|
|
mObjects.buildStaticGeometry (*store);
|
|
sh::Factory::getInstance().unloadUnreferencedMaterials();
|
|
mDebugging->cellAdded(store);
|
|
waterAdded(store);
|
|
}
|
|
|
|
void RenderingManager::addObject (const MWWorld::Ptr& ptr){
|
|
const MWWorld::Class& class_ =
|
|
MWWorld::Class::get (ptr);
|
|
class_.insertObjectRendering(ptr, *this);
|
|
}
|
|
|
|
void RenderingManager::removeObject (const MWWorld::Ptr& ptr)
|
|
{
|
|
if (!mObjects.deleteObject (ptr))
|
|
mActors.deleteObject (ptr);
|
|
}
|
|
|
|
void RenderingManager::moveObject (const MWWorld::Ptr& ptr, const Ogre::Vector3& position)
|
|
{
|
|
/// \todo move this to the rendering-subsystems
|
|
ptr.getRefData().getBaseNode()->setPosition(position);
|
|
}
|
|
|
|
void RenderingManager::scaleObject (const MWWorld::Ptr& ptr, const Ogre::Vector3& scale)
|
|
{
|
|
ptr.getRefData().getBaseNode()->setScale(scale);
|
|
}
|
|
|
|
void RenderingManager::rotateObject(const MWWorld::Ptr &ptr)
|
|
{
|
|
Ogre::Vector3 rot(ptr.getRefData().getPosition().rot);
|
|
|
|
if(ptr.getRefData().getHandle() == mCamera->getHandle() &&
|
|
!mCamera->isVanityOrPreviewModeEnabled())
|
|
mCamera->rotateCamera(rot, false);
|
|
|
|
Ogre::Quaternion newo = Ogre::Quaternion(Ogre::Radian(-rot.z), Ogre::Vector3::UNIT_Z);
|
|
if(!MWWorld::Class::get(ptr).isActor())
|
|
newo = Ogre::Quaternion(Ogre::Radian(-rot.x), Ogre::Vector3::UNIT_X) *
|
|
Ogre::Quaternion(Ogre::Radian(-rot.y), Ogre::Vector3::UNIT_Y) * newo;
|
|
|
|
ptr.getRefData().getBaseNode()->setOrientation(newo);
|
|
}
|
|
|
|
void
|
|
RenderingManager::updateObjectCell(const MWWorld::Ptr &old, const MWWorld::Ptr &cur)
|
|
{
|
|
Ogre::SceneNode *child =
|
|
mRendering.getScene()->getSceneNode(old.getRefData().getHandle());
|
|
|
|
Ogre::SceneNode *parent = child->getParentSceneNode();
|
|
parent->removeChild(child);
|
|
|
|
if (MWWorld::Class::get(old).isActor()) {
|
|
mActors.updateObjectCell(old, cur);
|
|
} else {
|
|
mObjects.updateObjectCell(old, cur);
|
|
}
|
|
}
|
|
|
|
void RenderingManager::updatePlayerPtr(const MWWorld::Ptr &ptr)
|
|
{
|
|
if(mPlayerAnimation)
|
|
mPlayerAnimation->updatePtr(ptr);
|
|
if(mCamera->getHandle() == ptr.getRefData().getHandle())
|
|
mCamera->attachTo(ptr);
|
|
}
|
|
|
|
void RenderingManager::rebuildPtr(const MWWorld::Ptr &ptr)
|
|
{
|
|
NpcAnimation *anim = NULL;
|
|
if(ptr.getRefData().getHandle() == "player")
|
|
anim = mPlayerAnimation;
|
|
else if(MWWorld::Class::get(ptr).isActor())
|
|
anim = dynamic_cast<NpcAnimation*>(mActors.getAnimation(ptr));
|
|
if(anim)
|
|
{
|
|
anim->rebuild();
|
|
if(mCamera->getHandle() == ptr.getRefData().getHandle())
|
|
{
|
|
mCamera->attachTo(ptr);
|
|
mCamera->setAnimation(anim);
|
|
}
|
|
}
|
|
}
|
|
|
|
void RenderingManager::update (float duration, bool paused)
|
|
{
|
|
MWBase::World *world = MWBase::Environment::get().getWorld();
|
|
|
|
MWWorld::Ptr player = world->getPlayer().getPlayer();
|
|
|
|
int blind = MWWorld::Class::get(player).getCreatureStats(player).getMagicEffects().get(MWMechanics::EffectKey(ESM::MagicEffect::Blind)).mMagnitude;
|
|
mRendering.getFader()->setFactor(std::max(0.f, 1.f-(blind / 100.f)));
|
|
setAmbientMode();
|
|
|
|
// player position
|
|
MWWorld::RefData &data = player.getRefData();
|
|
Ogre::Vector3 playerPos(data.getPosition().pos);
|
|
|
|
mCamera->setCameraDistance();
|
|
if(!mCamera->isFirstPerson())
|
|
{
|
|
Ogre::Vector3 orig, dest;
|
|
mCamera->getPosition(orig, dest);
|
|
|
|
btVector3 btOrig(orig.x, orig.y, orig.z);
|
|
btVector3 btDest(dest.x, dest.y, dest.z);
|
|
std::pair<bool,float> test = mPhysicsEngine->sphereCast(mRendering.getCamera()->getNearClipDistance()*2.5, btOrig, btDest);
|
|
if(test.first)
|
|
mCamera->setCameraDistance(test.second * orig.distance(dest), false, false);
|
|
}
|
|
|
|
// Sink the camera while sneaking
|
|
bool isSneaking = MWWorld::Class::get(player).getStance(player, MWWorld::Class::Sneak);
|
|
bool isInAir = !world->isOnGround(player);
|
|
bool isSwimming = world->isSwimming(player);
|
|
|
|
if(isSneaking && !(isSwimming || isInAir))
|
|
mCamera->setSneakOffset();
|
|
|
|
|
|
mOcclusionQuery->update(duration);
|
|
|
|
mVideoPlayer->update ();
|
|
|
|
mRendering.update(duration);
|
|
|
|
Ogre::ControllerManager::getSingleton().setTimeFactor(paused ? 0.f : 1.f);
|
|
|
|
Ogre::Vector3 cam = mRendering.getCamera()->getRealPosition();
|
|
|
|
applyFog(world->isUnderwater(player.getCell(), cam));
|
|
|
|
mCamera->update(duration, paused);
|
|
|
|
if(paused)
|
|
return;
|
|
|
|
mActors.update (duration);
|
|
mObjects.update (duration);
|
|
|
|
|
|
mSkyManager->update(duration);
|
|
|
|
mSkyManager->setGlare(mOcclusionQuery->getSunVisibility());
|
|
|
|
Ogre::SceneNode *node = data.getBaseNode();
|
|
Ogre::Quaternion orient = node->_getDerivedOrientation();
|
|
|
|
mLocalMap->updatePlayer(playerPos, orient);
|
|
|
|
mWater->updateUnderwater(world->isUnderwater(player.getCell(), cam));
|
|
|
|
mWater->update(duration, playerPos);
|
|
}
|
|
|
|
void RenderingManager::preRenderTargetUpdate(const RenderTargetEvent &evt)
|
|
{
|
|
mOcclusionQuery->setActive(true);
|
|
}
|
|
|
|
void RenderingManager::postRenderTargetUpdate(const RenderTargetEvent &evt)
|
|
{
|
|
// deactivate queries to make sure we aren't getting false results from several misc render targets
|
|
// (will be reactivated at the bottom of this method)
|
|
mOcclusionQuery->setActive(false);
|
|
}
|
|
|
|
void RenderingManager::waterAdded (MWWorld::Ptr::CellStore *store)
|
|
{
|
|
const MWWorld::Store<ESM::Land> &lands =
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Land>();
|
|
|
|
if(store->mCell->mData.mFlags & ESM::Cell::HasWater
|
|
|| ((store->mCell->isExterior())
|
|
&& !lands.search(store->mCell->getGridX(),store->mCell->getGridY()) )) // always use water, if the cell does not have land.
|
|
{
|
|
mWater->changeCell(store->mCell);
|
|
mWater->setActive(true);
|
|
}
|
|
else
|
|
removeWater();
|
|
}
|
|
|
|
void RenderingManager::setWaterHeight(const float height)
|
|
{
|
|
mWater->setHeight(height);
|
|
}
|
|
|
|
void RenderingManager::skyEnable ()
|
|
{
|
|
mSkyManager->enable();
|
|
mOcclusionQuery->setSunNode(mSkyManager->getSunNode());
|
|
}
|
|
|
|
void RenderingManager::skyDisable ()
|
|
{
|
|
mSkyManager->disable();
|
|
}
|
|
|
|
void RenderingManager::skySetHour (double hour)
|
|
{
|
|
mSkyManager->setHour(hour);
|
|
}
|
|
|
|
|
|
void RenderingManager::skySetDate (int day, int month)
|
|
{
|
|
mSkyManager->setDate(day, month);
|
|
}
|
|
|
|
int RenderingManager::skyGetMasserPhase() const
|
|
{
|
|
|
|
return mSkyManager->getMasserPhase();
|
|
}
|
|
|
|
int RenderingManager::skyGetSecundaPhase() const
|
|
{
|
|
return mSkyManager->getSecundaPhase();
|
|
}
|
|
|
|
void RenderingManager::skySetMoonColour (bool red){
|
|
mSkyManager->setMoonColour(red);
|
|
}
|
|
|
|
bool RenderingManager::toggleRenderMode(int mode)
|
|
{
|
|
if (mode == MWBase::World::Render_CollisionDebug || mode == MWBase::World::Render_Pathgrid)
|
|
return mDebugging->toggleRenderMode(mode);
|
|
else if (mode == MWBase::World::Render_Wireframe)
|
|
{
|
|
if (mRendering.getCamera()->getPolygonMode() == PM_SOLID)
|
|
{
|
|
mCompositors->setEnabled(false);
|
|
|
|
mRendering.getCamera()->setPolygonMode(PM_WIREFRAME);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
mCompositors->setEnabled(true);
|
|
|
|
mRendering.getCamera()->setPolygonMode(PM_SOLID);
|
|
return false;
|
|
}
|
|
}
|
|
else if (mode == MWBase::World::Render_BoundingBoxes)
|
|
{
|
|
bool show = !mRendering.getScene()->getShowBoundingBoxes();
|
|
mRendering.getScene()->showBoundingBoxes(show);
|
|
return show;
|
|
}
|
|
else //if (mode == MWBase::World::Render_Compositors)
|
|
{
|
|
return mCompositors->toggle();
|
|
}
|
|
}
|
|
|
|
void RenderingManager::configureFog(MWWorld::Ptr::CellStore &mCell)
|
|
{
|
|
Ogre::ColourValue color;
|
|
color.setAsABGR (mCell.mCell->mAmbi.mFog);
|
|
|
|
configureFog(mCell.mCell->mAmbi.mFogDensity, color);
|
|
}
|
|
|
|
void RenderingManager::configureFog(const float density, const Ogre::ColourValue& colour)
|
|
{
|
|
mFogColour = colour;
|
|
float max = Settings::Manager::getFloat("max viewing distance", "Viewing distance");
|
|
|
|
mFogStart = max / (density) * Settings::Manager::getFloat("fog start factor", "Viewing distance");
|
|
mFogEnd = max / (density) * Settings::Manager::getFloat("fog end factor", "Viewing distance");
|
|
|
|
mRendering.getCamera()->setFarClipDistance ( Settings::Manager::getFloat("max viewing distance", "Viewing distance") / density );
|
|
}
|
|
|
|
void RenderingManager::applyFog (bool underwater)
|
|
{
|
|
if (!underwater)
|
|
{
|
|
mRendering.getScene()->setFog (FOG_LINEAR, mFogColour, 0, mFogStart, mFogEnd);
|
|
mRendering.getViewport()->setBackgroundColour (mFogColour);
|
|
mWater->setViewportBackground (mFogColour);
|
|
}
|
|
else
|
|
{
|
|
mRendering.getScene()->setFog (FOG_LINEAR, Ogre::ColourValue(0.18039, 0.23137, 0.25490), 0, 0, 1000);
|
|
mRendering.getViewport()->setBackgroundColour (Ogre::ColourValue(0.18039, 0.23137, 0.25490));
|
|
mWater->setViewportBackground (Ogre::ColourValue(0.18039, 0.23137, 0.25490));
|
|
}
|
|
}
|
|
|
|
void RenderingManager::setAmbientMode()
|
|
{
|
|
switch (mAmbientMode)
|
|
{
|
|
case 0:
|
|
setAmbientColour(mAmbientColor);
|
|
break;
|
|
|
|
case 1:
|
|
setAmbientColour(0.7f*mAmbientColor + 0.3f*ColourValue(1,1,1));
|
|
break;
|
|
|
|
case 2:
|
|
setAmbientColour(ColourValue(1,1,1));
|
|
break;
|
|
}
|
|
}
|
|
|
|
void RenderingManager::configureAmbient(MWWorld::Ptr::CellStore &mCell)
|
|
{
|
|
if (mCell.mCell->mData.mFlags & ESM::Cell::Interior)
|
|
mAmbientColor.setAsABGR (mCell.mCell->mAmbi.mAmbient);
|
|
setAmbientMode();
|
|
|
|
// Create a "sun" that shines light downwards. It doesn't look
|
|
// completely right, but leave it for now.
|
|
if(!mSun)
|
|
{
|
|
mSun = mRendering.getScene()->createLight();
|
|
mSun->setType(Ogre::Light::LT_DIRECTIONAL);
|
|
}
|
|
if (mCell.mCell->mData.mFlags & ESM::Cell::Interior)
|
|
{
|
|
Ogre::ColourValue colour;
|
|
colour.setAsABGR (mCell.mCell->mAmbi.mSunlight);
|
|
mSun->setDiffuseColour (colour);
|
|
mSun->setDirection(0,-1,0);
|
|
}
|
|
}
|
|
// Switch through lighting modes.
|
|
|
|
void RenderingManager::toggleLight()
|
|
{
|
|
if (mAmbientMode==2)
|
|
mAmbientMode = 0;
|
|
else
|
|
++mAmbientMode;
|
|
|
|
switch (mAmbientMode)
|
|
{
|
|
case 0: std::cout << "Setting lights to normal\n"; break;
|
|
case 1: std::cout << "Turning the lights up\n"; break;
|
|
case 2: std::cout << "Turning the lights to full\n"; break;
|
|
}
|
|
|
|
setAmbientMode();
|
|
}
|
|
|
|
void RenderingManager::setSunColour(const Ogre::ColourValue& colour)
|
|
{
|
|
if (!mSunEnabled) return;
|
|
mSun->setDiffuseColour(colour);
|
|
mSun->setSpecularColour(colour);
|
|
}
|
|
|
|
void RenderingManager::setAmbientColour(const Ogre::ColourValue& colour)
|
|
{
|
|
mAmbientColor = colour;
|
|
|
|
MWWorld::Ptr player = MWBase::Environment::get().getWorld()->getPlayer().getPlayer();
|
|
int nightEye = MWWorld::Class::get(player).getCreatureStats(player).getMagicEffects().get(MWMechanics::EffectKey(ESM::MagicEffect::NightEye)).mMagnitude;
|
|
Ogre::ColourValue final = colour;
|
|
final += Ogre::ColourValue(0.7,0.7,0.7,0) * std::min(1.f, (nightEye/100.f));
|
|
|
|
mRendering.getScene()->setAmbientLight(final);
|
|
}
|
|
|
|
void RenderingManager::sunEnable(bool real)
|
|
{
|
|
if (real && mSun) mSun->setVisible(true);
|
|
else
|
|
{
|
|
// Don't disable the light, as the shaders assume the first light to be directional.
|
|
mSunEnabled = true;
|
|
}
|
|
}
|
|
|
|
void RenderingManager::sunDisable(bool real)
|
|
{
|
|
if (real && mSun) mSun->setVisible(false);
|
|
else
|
|
{
|
|
// Don't disable the light, as the shaders assume the first light to be directional.
|
|
mSunEnabled = false;
|
|
if (mSun)
|
|
{
|
|
mSun->setDiffuseColour(ColourValue(0,0,0));
|
|
mSun->setSpecularColour(ColourValue(0,0,0));
|
|
}
|
|
}
|
|
}
|
|
|
|
void RenderingManager::setSunDirection(const Ogre::Vector3& direction)
|
|
{
|
|
// direction * -1 (because 'direction' is camera to sun vector and not sun to camera),
|
|
if (mSun) mSun->setDirection(Vector3(-direction.x, -direction.y, -direction.z));
|
|
|
|
mSkyManager->setSunDirection(direction);
|
|
}
|
|
|
|
void RenderingManager::setGlare(bool glare)
|
|
{
|
|
mSkyManager->setGlare(glare);
|
|
}
|
|
|
|
void RenderingManager::requestMap(MWWorld::Ptr::CellStore* cell)
|
|
{
|
|
if (cell->mCell->isExterior())
|
|
{
|
|
assert(mTerrain);
|
|
|
|
Ogre::AxisAlignedBox dims = mObjects.getDimensions(cell);
|
|
Ogre::Vector2 center(cell->mCell->getGridX() + 0.5, cell->mCell->getGridY() + 0.5);
|
|
dims.merge(mTerrain->getWorldBoundingBox(center));
|
|
|
|
if (dims.isFinite())
|
|
mTerrain->update(dims.getCenter());
|
|
|
|
mLocalMap->requestMap(cell, dims.getMinimum().z, dims.getMaximum().z);
|
|
}
|
|
else
|
|
mLocalMap->requestMap(cell, mObjects.getDimensions(cell));
|
|
}
|
|
|
|
void RenderingManager::preCellChange(MWWorld::Ptr::CellStore* cell)
|
|
{
|
|
mLocalMap->saveFogOfWar(cell);
|
|
}
|
|
|
|
void RenderingManager::disableLights(bool sun)
|
|
{
|
|
mObjects.disableLights();
|
|
sunDisable(sun);
|
|
}
|
|
|
|
void RenderingManager::enableLights(bool sun)
|
|
{
|
|
mObjects.enableLights();
|
|
sunEnable(sun);
|
|
}
|
|
|
|
Shadows* RenderingManager::getShadows()
|
|
{
|
|
return mShadows;
|
|
}
|
|
|
|
void RenderingManager::switchToInterior()
|
|
{
|
|
// causes light flicker in opengl when moving..
|
|
//mRendering.getScene()->setCameraRelativeRendering(false);
|
|
}
|
|
|
|
void RenderingManager::switchToExterior()
|
|
{
|
|
// causes light flicker in opengl when moving..
|
|
//mRendering.getScene()->setCameraRelativeRendering(true);
|
|
}
|
|
|
|
Ogre::Vector4 RenderingManager::boundingBoxToScreen(Ogre::AxisAlignedBox bounds)
|
|
{
|
|
Ogre::Matrix4 mat = mRendering.getCamera()->getViewMatrix();
|
|
|
|
const Ogre::Vector3* corners = bounds.getAllCorners();
|
|
|
|
float min_x = 1.0f, max_x = 0.0f, min_y = 1.0f, max_y = 0.0f;
|
|
|
|
// expand the screen-space bounding-box so that it completely encloses
|
|
// the object's AABB
|
|
for (int i=0; i<8; i++)
|
|
{
|
|
Ogre::Vector3 corner = corners[i];
|
|
|
|
// multiply the AABB corner vertex by the view matrix to
|
|
// get a camera-space vertex
|
|
corner = mat * corner;
|
|
|
|
// make 2D relative/normalized coords from the view-space vertex
|
|
// by dividing out the Z (depth) factor -- this is an approximation
|
|
float x = corner.x / corner.z + 0.5;
|
|
float y = corner.y / corner.z + 0.5;
|
|
|
|
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 Vector4(min_x, min_y, max_x, max_y);
|
|
}
|
|
|
|
Compositors* RenderingManager::getCompositors()
|
|
{
|
|
return mCompositors;
|
|
}
|
|
|
|
void RenderingManager::processChangedSettings(const Settings::CategorySettingVector& settings)
|
|
{
|
|
bool changeRes = false;
|
|
bool rebuild = false; // rebuild static geometry (necessary after any material changes)
|
|
for (Settings::CategorySettingVector::const_iterator it=settings.begin();
|
|
it != settings.end(); ++it)
|
|
{
|
|
if (it->second == "menu transparency" && it->first == "GUI")
|
|
{
|
|
setMenuTransparency(Settings::Manager::getFloat("menu transparency", "GUI"));
|
|
}
|
|
else if (it->second == "max viewing distance" && it->first == "Viewing distance")
|
|
{
|
|
if (!MWBase::Environment::get().getWorld()->isCellExterior() && !MWBase::Environment::get().getWorld()->isCellQuasiExterior())
|
|
configureFog(*MWBase::Environment::get().getWorld()->getPlayer().getPlayer().getCell());
|
|
}
|
|
else if (it->first == "Video" && (
|
|
it->second == "resolution x"
|
|
|| it->second == "resolution y"
|
|
|| it->second == "fullscreen"))
|
|
changeRes = true;
|
|
else if (it->second == "field of view" && it->first == "General")
|
|
mRendering.setFov(Settings::Manager::getFloat("field of view", "General"));
|
|
else if ((it->second == "texture filtering" && it->first == "General")
|
|
|| (it->second == "anisotropy" && it->first == "General"))
|
|
{
|
|
TextureFilterOptions tfo;
|
|
std::string filter = Settings::Manager::getString("texture filtering", "General");
|
|
if (filter == "anisotropic") tfo = TFO_ANISOTROPIC;
|
|
else if (filter == "trilinear") tfo = TFO_TRILINEAR;
|
|
else if (filter == "bilinear") tfo = TFO_BILINEAR;
|
|
else /*if (filter == "none")*/ tfo = TFO_NONE;
|
|
|
|
MaterialManager::getSingleton().setDefaultTextureFiltering(tfo);
|
|
MaterialManager::getSingleton().setDefaultAnisotropy( (filter == "anisotropic") ? Settings::Manager::getInt("anisotropy", "General") : 1 );
|
|
}
|
|
else if (it->second == "shader" && it->first == "Water")
|
|
{
|
|
applyCompositors();
|
|
sh::Factory::getInstance ().setGlobalSetting ("simple_water", Settings::Manager::getBool("shader", "Water") ? "false" : "true");
|
|
rebuild = true;
|
|
mRendering.getViewport ()->setClearEveryFrame (true);
|
|
}
|
|
else if (it->second == "refraction" && it->first == "Water")
|
|
{
|
|
sh::Factory::getInstance ().setGlobalSetting ("refraction", Settings::Manager::getBool("refraction", "Water") ? "true" : "false");
|
|
rebuild = true;
|
|
}
|
|
else if (it->second == "shaders" && it->first == "Objects")
|
|
{
|
|
sh::Factory::getInstance ().setShadersEnabled (Settings::Manager::getBool("shaders", "Objects"));
|
|
rebuild = true;
|
|
}
|
|
else if (it->second == "shader mode" && it->first == "General")
|
|
{
|
|
sh::Language lang;
|
|
std::string l = Settings::Manager::getString("shader mode", "General");
|
|
if (l == "glsl")
|
|
lang = sh::Language_GLSL;
|
|
else if (l == "hlsl")
|
|
lang = sh::Language_HLSL;
|
|
else
|
|
lang = sh::Language_CG;
|
|
sh::Factory::getInstance ().setCurrentLanguage (lang);
|
|
rebuild = true;
|
|
}
|
|
else if (it->first == "Shadows")
|
|
{
|
|
mShadows->recreate ();
|
|
|
|
rebuild = true;
|
|
}
|
|
}
|
|
|
|
if (changeRes)
|
|
{
|
|
unsigned int x = Settings::Manager::getInt("resolution x", "Video");
|
|
unsigned int y = Settings::Manager::getInt("resolution y", "Video");
|
|
bool fullscreen = Settings::Manager::getBool("fullscreen", "Video");
|
|
|
|
SDL_Window* window = mRendering.getSDLWindow();
|
|
|
|
SDL_SetWindowFullscreen(window, 0);
|
|
|
|
if (SDL_GetWindowFlags(window) & SDL_WINDOW_MAXIMIZED)
|
|
SDL_RestoreWindow(window);
|
|
|
|
if (fullscreen)
|
|
{
|
|
SDL_DisplayMode mode;
|
|
SDL_GetWindowDisplayMode(window, &mode);
|
|
mode.w = x;
|
|
mode.h = y;
|
|
SDL_SetWindowDisplayMode(window, &mode);
|
|
SDL_SetWindowFullscreen(window, fullscreen);
|
|
}
|
|
else
|
|
SDL_SetWindowSize(window, x, y);
|
|
}
|
|
|
|
mWater->processChangedSettings(settings);
|
|
|
|
if (rebuild)
|
|
{
|
|
mObjects.rebuildStaticGeometry();
|
|
if (mTerrain)
|
|
mTerrain->applyMaterials(Settings::Manager::getBool("enabled", "Shadows"),
|
|
Settings::Manager::getBool("split", "Shadows"));
|
|
}
|
|
}
|
|
|
|
void RenderingManager::setMenuTransparency(float val)
|
|
{
|
|
Ogre::TexturePtr tex = Ogre::TextureManager::getSingleton().getByName("transparent.png");
|
|
std::vector<Ogre::uint32> buffer;
|
|
buffer.resize(1);
|
|
buffer[0] = (int(255*val) << 24);
|
|
memcpy(tex->getBuffer()->lock(Ogre::HardwareBuffer::HBL_DISCARD), &buffer[0], 1*4);
|
|
tex->getBuffer()->unlock();
|
|
}
|
|
|
|
void RenderingManager::windowResized(int x, int y)
|
|
{
|
|
mRendering.adjustViewport();
|
|
mCompositors->recreate();
|
|
|
|
mVideoPlayer->setResolution (x, y);
|
|
|
|
MWBase::Environment::get().getWindowManager()->windowResized(x,y);
|
|
}
|
|
|
|
void RenderingManager::applyCompositors()
|
|
{
|
|
}
|
|
|
|
void RenderingManager::getTriangleBatchCount(unsigned int &triangles, unsigned int &batches)
|
|
{
|
|
if (mCompositors->anyCompositorEnabled())
|
|
{
|
|
mCompositors->countTrianglesBatches(triangles, batches);
|
|
}
|
|
else
|
|
{
|
|
triangles = mRendering.getWindow()->getTriangleCount();
|
|
batches = mRendering.getWindow()->getBatchCount();
|
|
}
|
|
}
|
|
|
|
void RenderingManager::setupPlayer(const MWWorld::Ptr &ptr)
|
|
{
|
|
ptr.getRefData().setBaseNode(mRendering.getScene()->getSceneNode("player"));
|
|
mCamera->attachTo(ptr);
|
|
}
|
|
|
|
void RenderingManager::renderPlayer(const MWWorld::Ptr &ptr)
|
|
{
|
|
if(!mPlayerAnimation)
|
|
{
|
|
mPlayerAnimation = new NpcAnimation(ptr, ptr.getRefData().getBaseNode(),
|
|
MWWorld::Class::get(ptr).getInventoryStore(ptr),
|
|
RV_Actors);
|
|
}
|
|
else
|
|
{
|
|
// Reconstruct the NpcAnimation in-place
|
|
mPlayerAnimation->~NpcAnimation();
|
|
new(mPlayerAnimation) NpcAnimation(ptr, ptr.getRefData().getBaseNode(),
|
|
MWWorld::Class::get(ptr).getInventoryStore(ptr),
|
|
RV_Actors);
|
|
}
|
|
mCamera->setAnimation(mPlayerAnimation);
|
|
mWater->removeEmitter(ptr);
|
|
mWater->addEmitter(ptr);
|
|
// apply race height
|
|
MWBase::Environment::get().getWorld()->scaleObject(ptr, 1.f);
|
|
}
|
|
|
|
bool RenderingManager::vanityRotateCamera(const float *rot)
|
|
{
|
|
if(!mCamera->isVanityOrPreviewModeEnabled())
|
|
return false;
|
|
|
|
Ogre::Vector3 vRot(rot);
|
|
mCamera->rotateCamera(vRot, 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::getInteriorMapPosition (Ogre::Vector2 position, float& nX, float& nY, int &x, int& y)
|
|
{
|
|
return mLocalMap->getInteriorMapPosition (position, nX, nY, x, y);
|
|
}
|
|
|
|
bool RenderingManager::isPositionExplored (float nX, float nY, int x, int y, bool interior)
|
|
{
|
|
return mLocalMap->isPositionExplored(nX, nY, x, y, interior);
|
|
}
|
|
|
|
void RenderingManager::setupExternalRendering (MWRender::ExternalRendering& rendering)
|
|
{
|
|
rendering.setup (mRendering.getScene());
|
|
}
|
|
|
|
Animation* RenderingManager::getAnimation(const MWWorld::Ptr &ptr)
|
|
{
|
|
Animation *anim = mActors.getAnimation(ptr);
|
|
if(!anim && ptr.getRefData().getHandle() == "player")
|
|
anim = mPlayerAnimation;
|
|
return anim;
|
|
}
|
|
|
|
|
|
void RenderingManager::playVideo(const std::string& name, bool allowSkipping)
|
|
{
|
|
mVideoPlayer->playVideo ("video/" + name, allowSkipping);
|
|
}
|
|
|
|
void RenderingManager::stopVideo()
|
|
{
|
|
mVideoPlayer->stopVideo ();
|
|
}
|
|
|
|
void RenderingManager::addWaterRippleEmitter (const MWWorld::Ptr& ptr, float scale, float force)
|
|
{
|
|
mWater->addEmitter (ptr, scale, force);
|
|
}
|
|
|
|
void RenderingManager::removeWaterRippleEmitter (const MWWorld::Ptr& ptr)
|
|
{
|
|
mWater->removeEmitter (ptr);
|
|
}
|
|
|
|
void RenderingManager::updateWaterRippleEmitterPtr (const MWWorld::Ptr& old, const MWWorld::Ptr& ptr)
|
|
{
|
|
mWater->updateEmitterPtr(old, ptr);
|
|
}
|
|
|
|
void RenderingManager::frameStarted(float dt, bool paused)
|
|
{
|
|
if (mTerrain)
|
|
mTerrain->update(mRendering.getCamera()->getRealPosition());
|
|
|
|
if (!paused)
|
|
mWater->frameStarted(dt);
|
|
}
|
|
|
|
void RenderingManager::resetCamera()
|
|
{
|
|
mCamera->reset();
|
|
}
|
|
|
|
float RenderingManager::getTerrainHeightAt(Ogre::Vector3 worldPos)
|
|
{
|
|
if (!mTerrain || !mTerrain->getVisible())
|
|
return -std::numeric_limits<float>::max();
|
|
return mTerrain->getHeightAt(worldPos);
|
|
}
|
|
|
|
void RenderingManager::enableTerrain(bool enable)
|
|
{
|
|
if (enable)
|
|
{
|
|
if (!mTerrain)
|
|
{
|
|
Loading::Listener* listener = MWBase::Environment::get().getWindowManager()->getLoadingScreen();
|
|
Loading::ScopedLoad load(listener);
|
|
mTerrain = new Terrain::World(listener, mRendering.getScene(), new MWRender::TerrainStorage(), RV_Terrain,
|
|
Settings::Manager::getBool("distant land", "Terrain"),
|
|
Settings::Manager::getBool("shader", "Terrain"));
|
|
mTerrain->applyMaterials(Settings::Manager::getBool("enabled", "Shadows"),
|
|
Settings::Manager::getBool("split", "Shadows"));
|
|
mTerrain->update(mRendering.getCamera()->getRealPosition());
|
|
mTerrain->setLoadingListener(NULL);
|
|
}
|
|
mTerrain->setVisible(true);
|
|
}
|
|
else
|
|
if (mTerrain)
|
|
mTerrain->setVisible(false);
|
|
}
|
|
|
|
} // namespace
|