2013-02-03 14:46:23 +00:00
|
|
|
#include "refraction.hpp"
|
|
|
|
|
|
|
|
#include <OgreCamera.h>
|
|
|
|
#include <OgreTextureManager.h>
|
|
|
|
#include <OgreSceneManager.h>
|
|
|
|
#include <OgreHardwarePixelBuffer.h>
|
|
|
|
#include <OgreRenderTarget.h>
|
|
|
|
#include <OgreViewport.h>
|
2013-02-12 19:56:00 +00:00
|
|
|
#include <OgreRoot.h>
|
2014-02-19 17:40:29 +00:00
|
|
|
#include <OgreRenderTexture.h>
|
|
|
|
#include <OgreSceneNode.h>
|
2013-02-03 14:46:23 +00:00
|
|
|
|
2013-02-19 02:08:00 +00:00
|
|
|
#include <extern/shiny/Main/Factory.hpp>
|
|
|
|
|
2013-02-03 14:46:23 +00:00
|
|
|
#include "renderconst.hpp"
|
|
|
|
|
|
|
|
namespace MWRender
|
|
|
|
{
|
|
|
|
|
|
|
|
Refraction::Refraction(Ogre::Camera *parentCamera)
|
|
|
|
: mParentCamera(parentCamera)
|
2013-02-12 19:56:00 +00:00
|
|
|
, mRenderActive(false)
|
|
|
|
, mIsUnderwater(false)
|
2013-02-03 14:46:23 +00:00
|
|
|
{
|
|
|
|
mCamera = mParentCamera->getSceneManager()->createCamera("RefractionCamera");
|
|
|
|
|
2013-02-12 19:56:00 +00:00
|
|
|
mParentCamera->getSceneManager()->addRenderQueueListener(this);
|
|
|
|
|
2013-02-03 14:46:23 +00:00
|
|
|
Ogre::TexturePtr texture = Ogre::TextureManager::getSingleton().createManual("WaterRefraction",
|
|
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, 512, 512, 0, Ogre::PF_R8G8B8, Ogre::TU_RENDERTARGET);
|
|
|
|
|
|
|
|
mRenderTarget = texture->getBuffer()->getRenderTarget();
|
|
|
|
Ogre::Viewport* vp = mRenderTarget->addViewport(mCamera);
|
|
|
|
vp->setOverlaysEnabled(false);
|
|
|
|
vp->setShadowsEnabled(false);
|
2013-02-12 19:56:00 +00:00
|
|
|
vp->setVisibilityMask(RV_Actors + RV_Misc + RV_Statics + RV_StaticsSmall + RV_Terrain + RV_Sky);
|
2013-02-19 02:08:00 +00:00
|
|
|
vp->setMaterialScheme("water_refraction");
|
2013-03-03 18:52:20 +00:00
|
|
|
vp->setBackgroundColour (Ogre::ColourValue(0.18039, 0.23137, 0.25490));
|
2013-02-03 14:46:23 +00:00
|
|
|
mRenderTarget->setAutoUpdated(true);
|
|
|
|
mRenderTarget->addListener(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
Refraction::~Refraction()
|
|
|
|
{
|
2013-02-03 17:03:09 +00:00
|
|
|
mRenderTarget->removeListener(this);
|
2013-02-03 14:46:23 +00:00
|
|
|
Ogre::TextureManager::getSingleton().remove("WaterRefraction");
|
|
|
|
mParentCamera->getSceneManager()->destroyCamera(mCamera);
|
2013-02-17 17:12:38 +00:00
|
|
|
mParentCamera->getSceneManager()->removeRenderQueueListener(this);
|
2013-02-03 14:46:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Refraction::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());
|
|
|
|
|
2013-02-19 02:08:00 +00:00
|
|
|
// 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));
|
|
|
|
|
2013-02-17 17:12:38 +00:00
|
|
|
// enable clip plane here to take advantage of CPU culling for overwater or underwater objects
|
|
|
|
mCamera->enableCustomNearClipPlane(mIsUnderwater ? mNearClipPlaneUnderwater : mNearClipPlane);
|
|
|
|
|
2013-02-12 19:56:00 +00:00
|
|
|
mRenderActive = true;
|
2013-02-03 14:46:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Refraction::postRenderTargetUpdate(const Ogre::RenderTargetEvent& evt)
|
|
|
|
{
|
2013-02-19 02:08:00 +00:00
|
|
|
mCamera->disableCustomNearClipPlane ();
|
2013-02-12 19:56:00 +00:00
|
|
|
mRenderActive = false;
|
2013-02-03 14:46:23 +00:00
|
|
|
}
|
|
|
|
|
2013-02-03 18:01:59 +00:00
|
|
|
void Refraction::setHeight(float height)
|
2013-02-03 14:46:23 +00:00
|
|
|
{
|
2013-02-26 13:54:53 +00:00
|
|
|
mNearClipPlane = Ogre::Plane( -Ogre::Vector3(0,0,1), -(height + 5));
|
|
|
|
mNearClipPlaneUnderwater = Ogre::Plane( Ogre::Vector3(0,0,1), height - 5);
|
2013-02-12 19:56:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Refraction::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();
|
|
|
|
Ogre::Root::getSingleton().getRenderSystem()->_setProjectionMatrix(mCamera->getProjectionMatrixRS());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Refraction::renderQueueEnded (Ogre::uint8 queueGroupId, const Ogre::String &invocation, bool &repeatThisInvocation)
|
|
|
|
{
|
|
|
|
if (queueGroupId < 20 && mRenderActive)
|
|
|
|
{
|
|
|
|
mCamera->enableCustomNearClipPlane(mIsUnderwater ? mNearClipPlaneUnderwater : mNearClipPlane);
|
|
|
|
Ogre::Root::getSingleton().getRenderSystem()->_setProjectionMatrix(mCamera->getProjectionMatrixRS());
|
|
|
|
}
|
2013-02-03 14:46:23 +00:00
|
|
|
}
|
|
|
|
|
2013-03-03 14:11:45 +00:00
|
|
|
void Refraction::setActive(bool active)
|
|
|
|
{
|
|
|
|
mRenderTarget->setActive(active);
|
|
|
|
}
|
|
|
|
|
2013-02-03 14:46:23 +00:00
|
|
|
}
|