1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-18 13:12:50 +00:00

501 lines
16 KiB
C++
Raw Normal View History

2012-01-14 18:34:14 -05:00
#include "water.hpp"
2012-05-29 06:45:44 +02:00
#include <OgreRenderTexture.h>
#include <OgreEntity.h>
#include <OgreMeshManager.h>
#include <OgreMaterialManager.h>
#include <OgreHardwarePixelBuffer.h>
#include <OgreRoot.h>
#include <OgreCamera.h>
#include <OgreTextureManager.h>
#include <OgreViewport.h>
#include <OgreSceneNode.h>
#include <OgreTechnique.h>
2012-04-03 21:16:43 +02:00
#include "sky.hpp"
#include "renderingmanager.hpp"
2013-02-01 23:43:23 +01:00
#include "ripplesimulation.hpp"
#include "refraction.hpp"
#include <extern/shiny/Main/Factory.hpp>
2012-07-20 14:45:42 +02:00
#include <extern/shiny/Platforms/Ogre/OgreMaterial.hpp>
2012-07-19 23:30:41 +02:00
#include "../mwbase/environment.hpp"
#include "../mwbase/world.hpp"
using namespace Ogre;
2012-01-14 18:34:14 -05:00
2012-03-29 15:49:24 +02:00
namespace MWRender
{
CubeReflection::CubeReflection(Ogre::SceneManager* sceneManager)
: Reflection(sceneManager)
{
Ogre::TexturePtr texture = Ogre::TextureManager::getSingleton ().createManual("CubeReflection",
ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, TEX_TYPE_CUBE_MAP,
512,512, 0, PF_R8G8B8, TU_RENDERTARGET);
mCamera = mSceneMgr->createCamera ("CubeCamera");
mCamera->setNearClipDistance (5);
mCamera->setFarClipDistance (1000);
for (int face = 0; face < 6; ++face)
{
mRenderTargets[face] = texture->getBuffer (face)->getRenderTarget();
mRenderTargets[face]->removeAllViewports ();
Viewport* vp = mRenderTargets[face]->addViewport (mCamera);
vp->setOverlaysEnabled(false);
vp->setShadowsEnabled(false);
vp->setMaterialScheme ("water_reflection");
mRenderTargets[face]->setAutoUpdated(false);
/*
Vector3 lookAt(0,0,0), up(0,0,0), right(0,0,0);
switch(face)
{
case 0: lookAt.x =-1; up.y = 1; right.z = 1; break; // +X
case 1: lookAt.x = 1; up.y = 1; right.z =-1; break; // -X
case 2: lookAt.y =-1; up.z = 1; right.x = 1; break; // +Y
case 3: lookAt.y = 1; up.z =-1; right.x = 1; break; // -Y
case 4: lookAt.z = 1; up.y = 1; right.x =-1; break; // +Z
case 5: lookAt.z =-1; up.y = 1; right.x =-1; break; // -Z
}
Quaternion orient(right, up, lookAt);
mCamera->setOrientation(orient);
*/
}
}
CubeReflection::~CubeReflection ()
{
Ogre::TextureManager::getSingleton ().remove("CubeReflection");
mSceneMgr->destroyCamera (mCamera);
}
void CubeReflection::update ()
{
mParentCamera->getParentSceneNode ()->needUpdate ();
mCamera->setPosition(mParentCamera->getDerivedPosition());
}
// --------------------------------------------------------------------------------------------------------------------------------
PlaneReflection::PlaneReflection(Ogre::SceneManager* sceneManager, SkyManager* sky)
: Reflection(sceneManager)
, mSky(sky)
2013-02-27 09:20:42 +01:00
, mRenderActive(false)
{
mCamera = mSceneMgr->createCamera ("PlaneReflectionCamera");
mSceneMgr->addRenderQueueListener(this);
mTexture = TextureManager::getSingleton().createManual("WaterReflection",
ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, TEX_TYPE_2D, 512, 512, 0, PF_R8G8B8, TU_RENDERTARGET);
mRenderTarget = mTexture->getBuffer()->getRenderTarget();
Viewport* vp = mRenderTarget->addViewport(mCamera);
vp->setOverlaysEnabled(false);
vp->setBackgroundColour(ColourValue(0.8f, 0.9f, 1.0f));
vp->setShadowsEnabled(false);
vp->setMaterialScheme("water_reflection");
mRenderTarget->addListener(this);
mRenderTarget->setActive(true);
mRenderTarget->setAutoUpdated(true);
sh::Factory::getInstance ().setTextureAlias ("WaterReflection", mTexture->getName());
}
PlaneReflection::~PlaneReflection ()
{
mRenderTarget->removeListener (this);
mSceneMgr->destroyCamera (mCamera);
mSceneMgr->removeRenderQueueListener(this);
TextureManager::getSingleton ().remove("WaterReflection");
}
void PlaneReflection::renderQueueStarted (Ogre::uint8 queueGroupId, const Ogre::String &invocation, bool &skipThisInvocation)
{
// We don't want the sky to get clipped by custom near clip plane (the water plane)
if (queueGroupId < 20 && mRenderActive)
{
mCamera->disableCustomNearClipPlane();
Root::getSingleton().getRenderSystem()->_setProjectionMatrix(mCamera->getProjectionMatrixRS());
}
}
void PlaneReflection::renderQueueEnded (Ogre::uint8 queueGroupId, const Ogre::String &invocation, bool &repeatThisInvocation)
{
if (queueGroupId < 20 && mRenderActive)
{
2013-02-12 20:56:00 +01:00
mCamera->enableCustomNearClipPlane(mIsUnderwater ? mErrorPlaneUnderwater : mErrorPlane);
Root::getSingleton().getRenderSystem()->_setProjectionMatrix(mCamera->getProjectionMatrixRS());
}
}
void PlaneReflection::preRenderTargetUpdate(const Ogre::RenderTargetEvent& evt)
{
mParentCamera->getParentSceneNode ()->needUpdate ();
mCamera->setOrientation(mParentCamera->getDerivedOrientation());
mCamera->setPosition(mParentCamera->getDerivedPosition());
mCamera->setNearClipDistance(mParentCamera->getNearClipDistance());
mCamera->setFarClipDistance(mParentCamera->getFarClipDistance());
mCamera->setAspectRatio(mParentCamera->getAspectRatio());
mCamera->setFOVy(mParentCamera->getFOVy());
mRenderActive = true;
mCamera->enableReflection(mWaterPlane);
// for depth calculation, we want the original viewproj matrix _without_ the custom near clip plane.
// since all we are interested in is depth, we only need the third row of the matrix.
Ogre::Matrix4 projMatrix = mCamera->getProjectionMatrixWithRSDepth () * mCamera->getViewMatrix ();
sh::Vector4* row3 = new sh::Vector4(projMatrix[2][0], projMatrix[2][1], projMatrix[2][2], projMatrix[2][3]);
sh::Factory::getInstance ().setSharedParameter ("vpRow2Fix", sh::makeProperty<sh::Vector4> (row3));
// enable clip plane here to take advantage of CPU culling for overwater or underwater objects
mCamera->enableCustomNearClipPlane(mIsUnderwater ? mErrorPlaneUnderwater : mErrorPlane);
}
void PlaneReflection::postRenderTargetUpdate(const Ogre::RenderTargetEvent& evt)
{
mCamera->disableReflection();
mCamera->disableCustomNearClipPlane();
mRenderActive = false;
}
2013-02-03 19:01:59 +01:00
void PlaneReflection::setHeight (float height)
{
mWaterPlane = Plane(Ogre::Vector3(0,0,1), height);
mErrorPlane = Plane(Ogre::Vector3(0,0,1), height - 5);
mErrorPlaneUnderwater = Plane(Ogre::Vector3(0,0,-1), -height - 5);
}
void PlaneReflection::setActive (bool active)
{
mRenderTarget->setActive(active);
}
void PlaneReflection::setViewportBackground(Ogre::ColourValue colour)
{
mRenderTarget->getViewport (0)->setBackgroundColour (colour);
}
void PlaneReflection::setVisibilityMask (int flags)
{
mRenderTarget->getViewport (0)->setVisibilityMask (flags);
}
// --------------------------------------------------------------------------------------------------------------------------------
2013-02-27 09:20:42 +01:00
Water::Water (Ogre::Camera *camera, RenderingManager* rend) :
mCamera (camera), mSceneMgr (camera->getSceneManager()),
mIsUnderwater(false), mVisibilityFlags(0),
mActive(1), mToggled(1),
mRendering(rend),
mWaterTimer(0.f),
2013-02-01 23:43:23 +01:00
mReflection(NULL),
mRefraction(NULL),
mSimulation(NULL),
mPlayer(0,0)
{
2013-02-01 23:43:23 +01:00
mSimulation = new RippleSimulation(mSceneMgr);
mSky = rend->getSkyManager();
2012-03-29 15:49:24 +02:00
2012-07-20 14:45:42 +02:00
mMaterial = MaterialManager::getSingleton().getByName("Water");
2013-02-27 09:20:42 +01:00
mTop = 0;
2012-03-29 15:49:24 +02:00
mIsUnderwater = false;
2012-03-29 15:49:24 +02:00
2013-02-26 10:38:48 +01:00
mWaterPlane = Plane(Vector3::UNIT_Z, 0);
2012-03-29 15:49:24 +02:00
int waterScale = 300;
MeshManager::getSingleton().createPlane("water", ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, mWaterPlane,
CELL_SIZE*5*waterScale, CELL_SIZE*5*waterScale, 10, 10, true, 1, 3*waterScale,3*waterScale, Vector3::UNIT_Y);
2012-03-29 15:49:24 +02:00
mWater = mSceneMgr->createEntity("water");
2012-04-03 15:13:47 +02:00
mWater->setVisibilityFlags(RV_Water);
2012-04-11 18:53:13 +02:00
mWater->setCastShadows(false);
2013-03-03 19:52:20 +01:00
mWater->setRenderQueueGroup(RQG_Alpha);
mWaterNode = mSceneMgr->getRootSceneNode()->createChildSceneNode();
2012-04-11 18:53:13 +02:00
mWaterNode->attachObject(mWater);
2012-05-29 06:45:44 +02:00
applyRTT();
applyVisibilityMask();
2012-04-06 15:51:57 +02:00
mWater->setMaterial(mMaterial);
setHeight(mTop);
2012-07-20 14:45:42 +02:00
sh::MaterialInstance* m = sh::Factory::getInstance ().getMaterialInstance ("Water");
m->setListener (this);
2012-04-11 18:53:13 +02:00
// ----------------------------------------------------------------------------------------------
// ---------------------------------- reflection debug overlay ----------------------------------
// ----------------------------------------------------------------------------------------------
/*
2012-04-11 18:53:13 +02:00
if (Settings::Manager::getBool("shader", "Water"))
{
OverlayManager& mgr = OverlayManager::getSingleton();
Overlay* overlay;
// destroy if already exists
if ((overlay = mgr.getByName("ReflectionDebugOverlay")))
2012-04-11 18:53:13 +02:00
mgr.destroy(overlay);
2012-04-12 16:46:56 +02:00
2012-04-11 18:53:13 +02:00
overlay = mgr.create("ReflectionDebugOverlay");
2012-04-12 16:46:56 +02:00
2012-04-11 18:53:13 +02:00
if (MaterialManager::getSingleton().resourceExists("Ogre/ReflectionDebugTexture"))
MaterialManager::getSingleton().remove("Ogre/ReflectionDebugTexture");
MaterialPtr debugMat = MaterialManager::getSingleton().create(
2012-04-12 16:46:56 +02:00
"Ogre/ReflectionDebugTexture",
2012-04-11 18:53:13 +02:00
ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME);
2012-04-12 16:46:56 +02:00
2012-04-11 18:53:13 +02:00
debugMat->getTechnique(0)->getPass(0)->setLightingEnabled(false);
2012-07-20 14:45:42 +02:00
debugMat->getTechnique(0)->getPass(0)->createTextureUnitState(mReflectionTexture->getName());
2012-04-11 18:53:13 +02:00
OverlayContainer* debugPanel;
2012-04-12 16:46:56 +02:00
2012-04-11 18:53:13 +02:00
// destroy container if exists
try
{
if ((debugPanel =
2012-04-11 18:53:13 +02:00
static_cast<OverlayContainer*>(
mgr.getOverlayElement("Ogre/ReflectionDebugTexPanel"
))))
2012-04-11 18:53:13 +02:00
mgr.destroyOverlayElement(debugPanel);
}
catch (Ogre::Exception&) {}
2012-04-12 16:46:56 +02:00
2012-04-11 18:53:13 +02:00
debugPanel = (OverlayContainer*)
(OverlayManager::getSingleton().createOverlayElement("Panel", "Ogre/ReflectionDebugTexPanel"));
debugPanel->_setPosition(0, 0.55);
debugPanel->_setDimensions(0.3, 0.3);
debugPanel->setMaterialName(debugMat->getName());
debugPanel->show();
overlay->add2D(debugPanel);
overlay->show();
}
*/
}
2012-03-29 15:49:24 +02:00
2012-04-03 15:13:47 +02:00
void Water::setActive(bool active)
{
mActive = active;
2012-04-12 16:46:56 +02:00
updateVisible();
sh::Factory::getInstance ().setSharedParameter ("waterEnabled", sh::makeProperty<sh::FloatValue> (new sh::FloatValue(active ? 1.0 : 0.0)));
2012-04-03 15:13:47 +02:00
}
2012-01-22 16:38:10 -05:00
Water::~Water()
{
MeshManager::getSingleton().remove("water");
2012-06-09 19:15:11 +02:00
mWaterNode->detachObject(mWater);
mSceneMgr->destroyEntity(mWater);
mSceneMgr->destroySceneNode(mWaterNode);
delete mReflection;
delete mRefraction;
delete mSimulation;
}
2012-01-20 17:59:56 -05:00
void Water::changeCell(const ESM::Cell* cell)
{
2012-09-17 11:37:50 +04:00
mTop = cell->mWater;
2012-01-14 18:34:14 -05:00
setHeight(mTop);
2012-09-17 11:37:50 +04:00
if(!(cell->mData.mFlags & cell->Interior))
mWaterNode->setPosition(getSceneNodeCoordinates(cell->mData.mX, cell->mData.mY));
}
2012-03-29 15:49:24 +02:00
void Water::setHeight(const float height)
{
mTop = height;
2012-07-19 22:23:07 +02:00
mWaterPlane = Plane(Vector3::UNIT_Z, -height);
2012-07-19 22:23:07 +02:00
if (mReflection)
2013-02-03 19:01:59 +01:00
mReflection->setHeight(height);
if (mRefraction)
2013-02-03 19:01:59 +01:00
mRefraction->setHeight(height);
2012-07-19 22:23:07 +02:00
2013-02-26 10:38:48 +01:00
mWaterNode->setPosition(0, 0, height);
sh::Factory::getInstance ().setSharedParameter ("waterLevel", sh::makeProperty<sh::FloatValue>(new sh::FloatValue(height)));
}
2012-01-14 18:34:14 -05:00
bool Water::toggle()
{
2012-04-12 16:46:56 +02:00
mToggled = !mToggled;
updateVisible();
return mToggled;
}
2012-01-14 18:34:14 -05:00
void
Water::updateUnderwater(bool underwater)
{
if (!mActive) {
return;
}
mIsUnderwater =
underwater &&
mWater->isVisible() &&
mCamera->getPolygonMode() == Ogre::PM_SOLID;
2012-04-12 16:46:56 +02:00
if (mReflection)
mReflection->setUnderwater (mIsUnderwater);
2013-02-12 20:56:00 +01:00
if (mRefraction)
mRefraction->setUnderwater (mIsUnderwater);
2012-04-12 16:46:56 +02:00
updateVisible();
2012-01-14 18:34:14 -05:00
}
Vector3 Water::getSceneNodeCoordinates(int gridX, int gridY)
{
2013-02-26 10:38:48 +01:00
return Vector3(gridX * CELL_SIZE + (CELL_SIZE / 2), gridY * CELL_SIZE + (CELL_SIZE / 2), mTop);
}
2012-04-05 15:30:55 +02:00
void Water::setViewportBackground(const ColourValue& bg)
{
if (mReflection)
mReflection->setViewportBackground(bg);
2012-04-05 15:30:55 +02:00
}
2012-04-12 16:46:56 +02:00
void Water::updateVisible()
{
mWater->setVisible(mToggled && mActive);
if (mReflection)
mReflection->setActive(mToggled && mActive);
if (mRefraction)
mRefraction->setActive(mToggled && mActive);
2012-04-19 01:08:26 +02:00
}
2013-02-01 23:43:23 +01:00
void Water::update(float dt, Ogre::Vector3 player)
2012-04-19 01:08:26 +02:00
{
mWaterTimer += dt;
sh::Factory::getInstance ().setSharedParameter ("waterTimer", sh::makeProperty<sh::FloatValue>(new sh::FloatValue(mWaterTimer)));
2012-07-19 22:23:07 +02:00
mRendering->getSkyManager ()->setGlareEnabled (!mIsUnderwater);
2013-02-01 23:43:23 +01:00
mPlayer = Ogre::Vector2(player.x, player.y);
}
void Water::frameStarted(float dt)
{
mSimulation->update(dt, mPlayer);
if (mReflection)
mReflection->update();
2012-04-19 01:08:26 +02:00
}
2012-05-29 06:45:44 +02:00
void Water::applyRTT()
{
delete mReflection;
mReflection = NULL;
2013-03-03 19:52:20 +01:00
delete mRefraction;
mRefraction = NULL;
2012-05-29 06:45:44 +02:00
// Create rendertarget for reflection
//int rttsize = Settings::Manager::getInt("rtt size", "Water");
2012-05-29 06:45:44 +02:00
if (Settings::Manager::getBool("shader", "Water"))
{
mReflection = new PlaneReflection(mSceneMgr, mSky);
mReflection->setParentCamera (mCamera);
2013-02-03 19:01:59 +01:00
mReflection->setHeight(mTop);
2013-03-03 19:52:20 +01:00
if (Settings::Manager::getBool("refraction", "Water"))
{
mRefraction = new Refraction(mCamera);
mRefraction->setHeight(mTop);
}
}
updateVisible();
2012-05-29 06:45:44 +02:00
}
void Water::applyVisibilityMask()
{
mVisibilityFlags = RV_Terrain * Settings::Manager::getBool("reflect terrain", "Water")
2014-03-22 19:02:14 +01:00
+ (RV_Statics + RV_StaticsSmall + RV_Misc) * Settings::Manager::getBool("reflect statics", "Water")
2012-09-13 19:03:31 +02:00
+ RV_Actors * Settings::Manager::getBool("reflect actors", "Water")
2012-05-29 06:45:44 +02:00
+ RV_Sky;
if (mReflection)
mReflection->setVisibilityMask(mVisibilityFlags);
2012-05-29 06:45:44 +02:00
}
void Water::processChangedSettings(const Settings::CategorySettingVector& settings)
{
bool applyRT = false;
bool applyVisMask = false;
for (Settings::CategorySettingVector::const_iterator it=settings.begin();
it != settings.end(); ++it)
{
if ( it->first == "Water" && (
it->second == "shader"
|| it->second == "refraction"
2012-05-29 06:45:44 +02:00
|| it->second == "rtt size"))
applyRT = true;
if ( it->first == "Water" && (
it->second == "reflect actors"
|| it->second == "reflect terrain"
|| it->second == "reflect statics"))
applyVisMask = true;
}
if(applyRT)
{
applyRTT();
applyVisibilityMask();
mWater->setMaterial(mMaterial);
}
if (applyVisMask)
applyVisibilityMask();
}
2012-07-20 14:45:42 +02:00
void Water::requestedConfiguration (sh::MaterialInstance* m, const std::string& configuration)
{
}
void Water::createdConfiguration (sh::MaterialInstance* m, const std::string& configuration)
{
if (configuration == "local_map" || !Settings::Manager::getBool("shader", "Water"))
{
// for simple water, set animated texture names
// these have to be set in code
std::string textureNames[32];
for (int i=0; i<32; ++i)
{
textureNames[i] = "textures\\water\\water" + StringConverter::toString(i, 2, '0') + ".dds";
}
Ogre::Technique* t = static_cast<sh::OgreMaterial*>(m->getMaterial())->getOgreTechniqueForConfiguration(configuration);
if (t->getPass(0)->getNumTextureUnitStates () == 0)
return;
2012-07-20 14:45:42 +02:00
t->getPass(0)->getTextureUnitState(0)->setAnimatedTextureName(textureNames, 32, 2);
t->getPass(0)->setDepthWriteEnabled (false);
t->getPass(0)->setSceneBlending (Ogre::SBT_TRANSPARENT_ALPHA);
}
}
2013-02-27 09:20:42 +01:00
void Water::addEmitter (const MWWorld::Ptr& ptr, float scale, float force)
{
mSimulation->addEmitter (ptr, scale, force);
}
void Water::removeEmitter (const MWWorld::Ptr& ptr)
{
mSimulation->removeEmitter (ptr);
}
void Water::updateEmitterPtr (const MWWorld::Ptr& old, const MWWorld::Ptr& ptr)
{
2013-02-27 09:20:42 +01:00
mSimulation->updateEmitterPtr(old, ptr);
}
} // namespace