From ebdf25ccb942b0128facd18a06d36fb120d98ecf Mon Sep 17 00:00:00 2001
From: scrawl <scrawl@baseoftrash.de>
Date: Wed, 28 Oct 2015 19:57:58 +0100
Subject: [PATCH] Water: move refraction code to a new class

---
 apps/openmw/mwrender/water.cpp | 136 +++++++++++++++++++++------------
 apps/openmw/mwrender/water.hpp |   5 ++
 2 files changed, 94 insertions(+), 47 deletions(-)

diff --git a/apps/openmw/mwrender/water.cpp b/apps/openmw/mwrender/water.cpp
index 58e80a2714..17e804052e 100644
--- a/apps/openmw/mwrender/water.cpp
+++ b/apps/openmw/mwrender/water.cpp
@@ -273,6 +273,88 @@ osg::ref_ptr<osg::Image> readPngImage (const std::string& file)
 }
 
 
+class Refraction : public osg::Camera
+{
+public:
+    Refraction()
+    {
+        unsigned int rttSize = Settings::Manager::getInt("rtt size", "Water");
+        setRenderOrder(osg::Camera::PRE_RENDER);
+        setClearMask(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+        setRenderTargetImplementation(osg::Camera::FRAME_BUFFER_OBJECT);
+        setReferenceFrame(osg::Camera::RELATIVE_RF);
+
+        setCullMask(Mask_Effect|Mask_Scene|Mask_Terrain|Mask_Actor|Mask_ParticleSystem|Mask_Sky|Mask_Player|(1<<16));
+        setNodeMask(Mask_RenderToTexture);
+        setViewport(0, 0, rttSize, rttSize);
+
+        // No need for Update traversal since the scene is already updated as part of the main scene graph
+        // A double update would mess with the light collection (in addition to being plain redundant)
+        setUpdateCallback(new NoTraverseCallback);
+
+        // No need for fog here, we are already applying fog on the water surface itself as well as underwater fog
+        getOrCreateStateSet()->setMode(GL_FOG, osg::StateAttribute::OFF|osg::StateAttribute::OVERRIDE);
+
+        mClipCullNode = new ClipCullNode;
+        addChild(mClipCullNode);
+
+        mRefractionTexture = new osg::Texture2D;
+        mRefractionTexture->setTextureSize(rttSize, rttSize);
+        mRefractionTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
+        mRefractionTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
+        mRefractionTexture->setInternalFormat(GL_RGB);
+        mRefractionTexture->setFilter(osg::Texture::MIN_FILTER, osg::Texture::LINEAR);
+        mRefractionTexture->setFilter(osg::Texture::MAG_FILTER, osg::Texture::LINEAR);
+
+        attach(osg::Camera::COLOR_BUFFER, mRefractionTexture);
+
+        mRefractionDepthTexture = new osg::Texture2D;
+        mRefractionDepthTexture->setSourceFormat(GL_DEPTH_COMPONENT);
+        mRefractionDepthTexture->setInternalFormat(GL_DEPTH_COMPONENT24_ARB);
+        mRefractionDepthTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
+        mRefractionDepthTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
+        mRefractionDepthTexture->setSourceType(GL_UNSIGNED_INT);
+        mRefractionDepthTexture->setFilter(osg::Texture::MIN_FILTER, osg::Texture::LINEAR);
+        mRefractionDepthTexture->setFilter(osg::Texture::MAG_FILTER, osg::Texture::LINEAR);
+
+        attach(osg::Camera::DEPTH_BUFFER, mRefractionDepthTexture);
+    }
+
+    void setScene(osg::Node* scene)
+    {
+        if (mScene)
+            mClipCullNode->removeChild(mScene);
+        mScene = scene;
+        mClipCullNode->addChild(scene);
+    }
+
+    void setWaterLevel(float waterLevel)
+    {
+        mClipCullNode->setPlane(osg::Plane(osg::Vec3d(0,0,-1), osg::Vec3d(0,0, waterLevel)));
+    }
+
+    osg::Texture2D* getRefractionTexture() const
+    {
+        return mRefractionTexture.get();
+    }
+
+    osg::Texture2D* getRefractionDepthTexture() const
+    {
+        return mRefractionDepthTexture.get();
+    }
+
+private:
+    osg::ref_ptr<ClipCullNode> mClipCullNode;
+    osg::ref_ptr<osg::Texture2D> mRefractionTexture;
+    osg::ref_ptr<osg::Texture2D> mRefractionDepthTexture;
+    osg::ref_ptr<osg::Node> mScene;
+};
+
+class Reflection : public osg::Camera
+{
+
+};
+
 Water::Water(osg::Group *parent, osg::Group* sceneRoot, Resource::ResourceSystem *resourceSystem, osgUtil::IncrementalCompileOperation *ico,
              const MWWorld::Fallback* fallback, const std::string& resourcePath)
     : mParent(parent)
@@ -309,54 +391,13 @@ Water::Water(osg::Group *parent, osg::Group* sceneRoot, Resource::ResourceSystem
     const float waterLevel = -1;
 
     // refraction
-    unsigned int rttSize = Settings::Manager::getInt("rtt size", "Water");
-    osg::ref_ptr<osg::Camera> refractionCamera (new osg::Camera);
-    refractionCamera->setRenderOrder(osg::Camera::PRE_RENDER);
-    refractionCamera->setClearMask(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
-    refractionCamera->setRenderTargetImplementation(osg::Camera::FRAME_BUFFER_OBJECT);
-    refractionCamera->setReferenceFrame(osg::Camera::RELATIVE_RF);
-
-    refractionCamera->setCullMask(Mask_Effect|Mask_Scene|Mask_Terrain|Mask_Actor|Mask_ParticleSystem|Mask_Sky|Mask_Player|(1<<16));
-    refractionCamera->setNodeMask(Mask_RenderToTexture);
-    refractionCamera->setViewport(0, 0, rttSize, rttSize);
-
-    // No need for Update traversal since the mSceneRoot is already updated as part of the main scene graph
-    // A double update would mess with the light collection (in addition to being plain redundant)
-    refractionCamera->setUpdateCallback(new NoTraverseCallback);
-
-    // No need for fog here, we are already applying fog on the water surface itself as well as underwater fog
-    refractionCamera->getOrCreateStateSet()->setMode(GL_FOG, osg::StateAttribute::OFF|osg::StateAttribute::OVERRIDE);
-
-    osg::ref_ptr<ClipCullNode> clipNode (new ClipCullNode);
-    clipNode->setPlane(osg::Plane(osg::Vec3d(0,0,-1), osg::Vec3d(0,0, waterLevel)));
-
-    refractionCamera->addChild(clipNode);
-    clipNode->addChild(mSceneRoot);
+    mRefraction = new Refraction();
+    mRefraction->setWaterLevel(waterLevel);
+    mRefraction->setScene(mSceneRoot);
 
     // TODO: add ingame setting for texture quality
 
-    osg::ref_ptr<osg::Texture2D> refractionTexture = new osg::Texture2D;
-    refractionTexture->setTextureSize(rttSize, rttSize);
-    refractionTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
-    refractionTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
-    refractionTexture->setInternalFormat(GL_RGB);
-    refractionTexture->setFilter(osg::Texture::MIN_FILTER, osg::Texture::LINEAR);
-    refractionTexture->setFilter(osg::Texture::MAG_FILTER, osg::Texture::LINEAR);
-
-    refractionCamera->attach(osg::Camera::COLOR_BUFFER, refractionTexture);
-
-    osg::ref_ptr<osg::Texture2D> refractionDepthTexture = new osg::Texture2D;
-    refractionDepthTexture->setSourceFormat(GL_DEPTH_COMPONENT);
-    refractionDepthTexture->setInternalFormat(GL_DEPTH_COMPONENT24_ARB);
-    refractionDepthTexture->setWrap(osg::Texture::WRAP_S, osg::Texture::CLAMP_TO_EDGE);
-    refractionDepthTexture->setWrap(osg::Texture::WRAP_T, osg::Texture::CLAMP_TO_EDGE);
-    refractionDepthTexture->setSourceType(GL_UNSIGNED_INT);
-    refractionDepthTexture->setFilter(osg::Texture::MIN_FILTER, osg::Texture::LINEAR);
-    refractionDepthTexture->setFilter(osg::Texture::MAG_FILTER, osg::Texture::LINEAR);
-
-    refractionCamera->attach(osg::Camera::DEPTH_BUFFER, refractionDepthTexture);
-
-    mParent->addChild(refractionCamera);
+    mParent->addChild(mRefraction);
 
     // reflection
     osg::ref_ptr<osg::Camera> reflectionCamera (new osg::Camera);
@@ -368,6 +409,7 @@ Water::Water(osg::Group *parent, osg::Group* sceneRoot, Resource::ResourceSystem
     reflectionCamera->setCullMask(Mask_Effect|Mask_Scene|Mask_Terrain|Mask_Actor|Mask_ParticleSystem|Mask_Sky|Mask_Player|(1<<16));
     reflectionCamera->setNodeMask(Mask_RenderToTexture);
 
+    unsigned int rttSize = Settings::Manager::getInt("rtt size", "Water");
     reflectionCamera->setViewport(0, 0, rttSize, rttSize);
 
     // No need for Update traversal since the mSceneRoot is already updated as part of the main scene graph
@@ -430,8 +472,8 @@ Water::Water(osg::Group *parent, osg::Group* sceneRoot, Resource::ResourceSystem
     shaderStateset->addUniform(new osg::Uniform("normalMap", 3));
 
     shaderStateset->setTextureAttributeAndModes(0, reflectionTexture, osg::StateAttribute::ON);
-    shaderStateset->setTextureAttributeAndModes(1, refractionTexture, osg::StateAttribute::ON);
-    shaderStateset->setTextureAttributeAndModes(2, refractionDepthTexture, osg::StateAttribute::ON);
+    shaderStateset->setTextureAttributeAndModes(1, mRefraction->getRefractionTexture(), osg::StateAttribute::ON);
+    shaderStateset->setTextureAttributeAndModes(2, mRefraction->getRefractionDepthTexture(), osg::StateAttribute::ON);
     shaderStateset->setTextureAttributeAndModes(3, normalMap, osg::StateAttribute::ON);
     shaderStateset->setMode(GL_BLEND, osg::StateAttribute::ON); // TODO: set Off when refraction is on
     shaderStateset->setMode(GL_CULL_FACE, osg::StateAttribute::OFF);
diff --git a/apps/openmw/mwrender/water.hpp b/apps/openmw/mwrender/water.hpp
index a754646128..46b6382e7b 100644
--- a/apps/openmw/mwrender/water.hpp
+++ b/apps/openmw/mwrender/water.hpp
@@ -29,6 +29,8 @@ namespace MWWorld
 namespace MWRender
 {
 
+    class Refraction;
+    class Reflection;
     class RippleSimulation;
 
     /// Water rendering
@@ -44,6 +46,9 @@ namespace MWRender
 
         std::auto_ptr<RippleSimulation> mSimulation;
 
+        osg::ref_ptr<Refraction> mRefraction;
+        osg::ref_ptr<Reflection> mReflection;
+
         bool mEnabled;
         bool mToggled;
         float mTop;