forked from mirror/openmw-tes3mp
211 lines
6.9 KiB
C++
211 lines
6.9 KiB
C++
#include "ripplesimulation.hpp"
|
|
|
|
#include <OgreTextureManager.h>
|
|
#include <OgreStringConverter.h>
|
|
#include <OgreHardwarePixelBuffer.h>
|
|
#include <OgreRoot.h>
|
|
|
|
#include <extern/shiny/Main/Factory.hpp>
|
|
|
|
namespace MWRender
|
|
{
|
|
|
|
|
|
RippleSimulation::RippleSimulation(Ogre::SceneManager* mainSceneManager)
|
|
: mMainSceneMgr(mainSceneManager),
|
|
mTime(0),
|
|
mCurrentFrameOffset(0,0),
|
|
mPreviousFrameOffset(0,0),
|
|
mRippleCenter(0,0),
|
|
mTextureSize(512),
|
|
mRippleAreaLength(1000),
|
|
mImpulseSize(20),
|
|
mTexelOffset(0,0),
|
|
mFirstUpdate(true)
|
|
{
|
|
Ogre::AxisAlignedBox aabInf;
|
|
aabInf.setInfinite();
|
|
|
|
mSceneMgr = Ogre::Root::getSingleton().createSceneManager(Ogre::ST_GENERIC);
|
|
|
|
mCamera = mSceneMgr->createCamera("RippleCamera");
|
|
|
|
mRectangle = new Ogre::Rectangle2D(true);
|
|
mRectangle->setBoundingBox(aabInf);
|
|
mRectangle->setCorners(-1.0, 1.0, 1.0, -1.0, false);
|
|
Ogre::SceneNode* node = mSceneMgr->getRootSceneNode()->createChildSceneNode();
|
|
node->attachObject(mRectangle);
|
|
|
|
mImpulse = new Ogre::Rectangle2D(true);
|
|
mImpulse->setCorners(-0.1, 0.1, 0.1, -0.1, false);
|
|
mImpulse->setBoundingBox(aabInf);
|
|
mImpulse->setMaterial("AddImpulse");
|
|
Ogre::SceneNode* impulseNode = mSceneMgr->getRootSceneNode()->createChildSceneNode();
|
|
impulseNode->attachObject(mImpulse);
|
|
|
|
//float w=0.05;
|
|
for (int i=0; i<4; ++i)
|
|
{
|
|
Ogre::TexturePtr texture;
|
|
if (i != 3)
|
|
texture = Ogre::TextureManager::getSingleton().createManual("RippleHeight" + Ogre::StringConverter::toString(i),
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, mTextureSize, mTextureSize, 1, 0, Ogre::PF_R8G8B8, Ogre::TU_RENDERTARGET);
|
|
else
|
|
texture = Ogre::TextureManager::getSingleton().createManual("RippleNormal",
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME, Ogre::TEX_TYPE_2D, mTextureSize, mTextureSize, 1, 0, Ogre::PF_R8G8B8, Ogre::TU_RENDERTARGET);
|
|
|
|
|
|
Ogre::RenderTexture* rt = texture->getBuffer()->getRenderTarget();
|
|
rt->removeAllViewports();
|
|
rt->addViewport(mCamera);
|
|
rt->setAutoUpdated(false);
|
|
rt->getViewport(0)->setClearEveryFrame(false);
|
|
|
|
// debug overlay
|
|
/*
|
|
Ogre::Rectangle2D* debugOverlay = new Ogre::Rectangle2D(true);
|
|
debugOverlay->setCorners(w*2-1, 0.9, (w+0.18)*2-1, 0.4, false);
|
|
w += 0.2;
|
|
debugOverlay->setBoundingBox(aabInf);
|
|
Ogre::SceneNode* debugNode = mMainSceneMgr->getRootSceneNode()->createChildSceneNode();
|
|
debugNode->attachObject(debugOverlay);
|
|
|
|
Ogre::MaterialPtr debugMaterial = Ogre::MaterialManager::getSingleton().create("RippleDebug" + Ogre::StringConverter::toString(i),
|
|
Ogre::ResourceGroupManager::DEFAULT_RESOURCE_GROUP_NAME);
|
|
|
|
if (i != 3)
|
|
debugMaterial->getTechnique(0)->getPass(0)->createTextureUnitState("RippleHeight" + Ogre::StringConverter::toString(i));
|
|
else
|
|
debugMaterial->getTechnique(0)->getPass(0)->createTextureUnitState("RippleNormal");
|
|
debugMaterial->getTechnique(0)->getPass(0)->setLightingEnabled(false);
|
|
|
|
debugOverlay->setMaterial("RippleDebug" + Ogre::StringConverter::toString(i));
|
|
*/
|
|
|
|
mRenderTargets[i] = rt;
|
|
mTextures[i] = texture;
|
|
}
|
|
|
|
sh::Factory::getInstance().setSharedParameter("rippleTextureSize", sh::makeProperty<sh::Vector4>(
|
|
new sh::Vector4(1.0/512, 1.0/512, 512, 512)));
|
|
sh::Factory::getInstance().setSharedParameter("rippleCenter", sh::makeProperty<sh::Vector3>(
|
|
new sh::Vector3(0, 0, 0)));
|
|
sh::Factory::getInstance().setSharedParameter("rippleAreaLength", sh::makeProperty<sh::FloatValue>(
|
|
new sh::FloatValue(mRippleAreaLength)));
|
|
|
|
}
|
|
|
|
RippleSimulation::~RippleSimulation()
|
|
{
|
|
delete mRectangle;
|
|
|
|
Ogre::Root::getSingleton().destroySceneManager(mSceneMgr);
|
|
}
|
|
|
|
void RippleSimulation::update(float dt, Ogre::Vector2 position)
|
|
{
|
|
// try to keep 20 fps
|
|
mTime += dt;
|
|
|
|
while (mTime >= 1/20.0 || mFirstUpdate)
|
|
{
|
|
mPreviousFrameOffset = mCurrentFrameOffset;
|
|
|
|
mCurrentFrameOffset = position - mRippleCenter;
|
|
// add texel offsets from previous frame.
|
|
mCurrentFrameOffset += mTexelOffset;
|
|
|
|
mTexelOffset = Ogre::Vector2(std::fmod(mCurrentFrameOffset.x, 1.0f/mTextureSize),
|
|
std::fmod(mCurrentFrameOffset.y, 1.0f/mTextureSize));
|
|
|
|
// now subtract new offset in order to snap to texels
|
|
mCurrentFrameOffset -= mTexelOffset;
|
|
|
|
// texture coordinate space
|
|
mCurrentFrameOffset /= mRippleAreaLength;
|
|
|
|
mRippleCenter = position;
|
|
|
|
addImpulses();
|
|
waterSimulation();
|
|
heightMapToNormalMap();
|
|
|
|
swapHeightMaps();
|
|
if (!mFirstUpdate)
|
|
mTime -= 1/20.0;
|
|
else
|
|
mFirstUpdate = false;
|
|
}
|
|
|
|
sh::Factory::getInstance().setSharedParameter("rippleCenter", sh::makeProperty<sh::Vector3>(
|
|
new sh::Vector3(mRippleCenter.x + mTexelOffset.x, mRippleCenter.y + mTexelOffset.y, 0)));
|
|
}
|
|
|
|
void RippleSimulation::addImpulse(Ogre::Vector2 position)
|
|
{
|
|
mImpulses.push(position);
|
|
}
|
|
|
|
void RippleSimulation::addImpulses()
|
|
{
|
|
mRectangle->setVisible(false);
|
|
mImpulse->setVisible(true);
|
|
|
|
while (mImpulses.size())
|
|
{
|
|
Ogre::Vector2 pos = mImpulses.front();
|
|
pos -= mRippleCenter;
|
|
pos /= mRippleAreaLength;
|
|
float size = mImpulseSize / mRippleAreaLength;
|
|
mImpulse->setCorners(pos.x-size, pos.y+size, pos.x+size, pos.y-size, false);
|
|
mImpulses.pop();
|
|
|
|
mRenderTargets[1]->update();
|
|
}
|
|
|
|
mImpulse->setVisible(false);
|
|
mRectangle->setVisible(true);
|
|
}
|
|
|
|
void RippleSimulation::waterSimulation()
|
|
{
|
|
mRectangle->setMaterial("HeightmapSimulation");
|
|
|
|
sh::Factory::getInstance().setTextureAlias("Heightmap0", mTextures[0]->getName());
|
|
sh::Factory::getInstance().setTextureAlias("Heightmap1", mTextures[1]->getName());
|
|
|
|
sh::Factory::getInstance().setSharedParameter("currentFrameOffset", sh::makeProperty<sh::Vector3>(
|
|
new sh::Vector3(mCurrentFrameOffset.x, mCurrentFrameOffset.y, 0)));
|
|
sh::Factory::getInstance().setSharedParameter("previousFrameOffset", sh::makeProperty<sh::Vector3>(
|
|
new sh::Vector3(mPreviousFrameOffset.x, mPreviousFrameOffset.y, 0)));
|
|
|
|
mRenderTargets[2]->update();
|
|
}
|
|
|
|
void RippleSimulation::heightMapToNormalMap()
|
|
{
|
|
mRectangle->setMaterial("HeightToNormalMap");
|
|
|
|
sh::Factory::getInstance().setTextureAlias("Heightmap2", mTextures[2]->getName());
|
|
|
|
mRenderTargets[TEX_NORMAL]->update();
|
|
}
|
|
|
|
void RippleSimulation::swapHeightMaps()
|
|
{
|
|
// 0 -> 1 -> 2 to 2 -> 0 ->1
|
|
Ogre::RenderTexture* tmp = mRenderTargets[0];
|
|
Ogre::TexturePtr tmp2 = mTextures[0];
|
|
|
|
mRenderTargets[0] = mRenderTargets[1];
|
|
mTextures[0] = mTextures[1];
|
|
|
|
mRenderTargets[1] = mRenderTargets[2];
|
|
mTextures[1] = mTextures[2];
|
|
|
|
mRenderTargets[2] = tmp;
|
|
mTextures[2] = tmp2;
|
|
}
|
|
|
|
|
|
}
|