]> git.mxchange.org Git - flightgear.git/blobdiff - src/Instrumentation/od_gauge.cxx
NavDisplay enhancements for Syd.
[flightgear.git] / src / Instrumentation / od_gauge.cxx
index d02cc354d0fdf5127336c980e8ff4b9ea26d7cea..6760bcd2244e17562d3aea218f89c3afda3a32ed 100644 (file)
@@ -4,6 +4,8 @@
 //
 // Copyright (C) 2005  Harald JOHNSEN
 //
+// Ported to OSG by Tim Moore - Jun 2007
+//
 // This program is free software; you can redistribute it and/or
 // modify it under the terms of the GNU General Public License as
 // published by the Free Software Foundation; either version 2 of the
 #  include "config.h"
 #endif
 
-#include <plib/ssg.h>
-#include <simgear/screen/extensions.hxx>
-#include <simgear/screen/RenderTexture.h>
+#include <osg/Texture2D>
+#include <osg/AlphaFunc>
+#include <osg/BlendFunc>
+#include <osg/Camera>
+#include <osg/Geode>
+#include <osg/NodeVisitor>
+#include <osg/Matrix>
+#include <osg/PolygonMode>
+#include <osg/ShadeModel>
+#include <osg/StateSet>
+#include <osgDB/FileNameUtils>
+
+#include <simgear/scene/util/RenderConstants.hxx>
 #include <simgear/debug/logstream.hxx>
-#include SG_GLU_H
 
 #include <Main/globals.hxx>
+#include <Viewer/renderer.hxx>
 #include <Scenery/scenery.hxx>
 #include "od_gauge.hxx"
 
 FGODGauge::FGODGauge() :
-    rtAvailable( false ),
-    rt( 0 )
+  rtAvailable( false )
 {
 }
 
-// done here and not in init() so we don't allocate a rendering context if it is
-// never used
-void FGODGauge::allocRT () {
-    GLint colorBits = 0;
-    glGetIntegerv( GL_BLUE_BITS, &colorBits );
-    textureWH = 256;
-    rt = new RenderTexture();
-    if( colorBits < 8 )
-        rt->Reset("rgba=5,5,5,1 ctt");
-    else
-        rt->Reset("rgba ctt");
-
-    if( rt->Initialize(256, 256, true) ) {
-        SG_LOG(SG_ALL, SG_INFO, "FGODGauge:Initialize sucessfull");
-        if (rt->BeginCapture())
-        {
-            SG_LOG(SG_ALL, SG_INFO, "FGODGauge:BeginCapture sucessfull, RTT available");
-            rtAvailable = true;
-            glViewport(0, 0, textureWH, textureWH);
-            glMatrixMode(GL_PROJECTION);
-            glLoadIdentity();
-            gluOrtho2D( -256.0, 256.0, -256.0, 256.0 );
-            glMatrixMode(GL_MODELVIEW);
-            glLoadIdentity();
-            glDisable(GL_LIGHTING);
-            glEnable(GL_COLOR_MATERIAL);
-            glDisable(GL_CULL_FACE);
-            glDisable(GL_FOG);
-            glDisable(GL_DEPTH_TEST);
-            glClearColor(0.0, 0.0, 0.0, 0.0);
-            glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
-            glBindTexture(GL_TEXTURE_2D, 0);
-            glEnable(GL_TEXTURE_2D);
-            glEnable(GL_ALPHA_TEST);
-            glAlphaFunc(GL_GREATER, 0.0f);
-            glDisable(GL_SMOOTH);
-            glEnable(GL_BLEND);
-            glBlendFunc( GL_ONE, GL_ONE_MINUS_SRC_ALPHA );
-            rt->EndCapture();
-        } else
-            SG_LOG(SG_ALL, SG_WARN, "FGODGauge:BeginCapture failed, RTT not available, using backbuffer");
-    } else
-        SG_LOG(SG_ALL, SG_WARN, "FGODGauge:Initialize failed, RTT not available, using backbuffer");
-}
-
-FGODGauge::~FGODGauge() {
-    delete rt;
-}
-
-void FGODGauge::init () {
-}
-
-void FGODGauge::update (double dt) {
-}
-
-void FGODGauge::beginCapture(int viewSize) {
-    if( ! rt )
-        allocRT();
-    if(rtAvailable) {
-        rt->BeginCapture();
-    }
-    else
-        set2D();
-     textureWH = viewSize;
-    glViewport(0, 0, textureWH, textureWH);
-}
-
-void FGODGauge::beginCapture(void) {
-    if( ! rt )
-        allocRT();
-    if(rtAvailable) {
-        rt->BeginCapture();
+void FGODGauge::allocRT ()
+{
+    camera = new osg::Camera;
+    // Only the far camera should trigger this texture to be rendered.
+    camera->setNodeMask(simgear::BACKGROUND_BIT);
+    camera->setProjectionMatrix(osg::Matrix::ortho2D(-textureWH/2.0, textureWH/2.0, -textureWH/2.0, textureWH/2.0));
+    camera->setViewport(0, 0, textureWH, textureWH);
+    camera->setReferenceFrame(osg::Transform::ABSOLUTE_RF);
+    camera->setRenderOrder(osg::Camera::PRE_RENDER);
+    camera->setClearMask(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
+    camera->setClearColor(osg::Vec4(0.0f, 0.0f, 0.0f , 0.0f));
+    camera->setRenderTargetImplementation(osg::Camera::FRAME_BUFFER_OBJECT, osg::Camera::FRAME_BUFFER);
+    osg::StateSet* stateSet = camera->getOrCreateStateSet();
+    stateSet->setMode(GL_LIGHTING, osg::StateAttribute::OFF);
+    stateSet->setMode(GL_CULL_FACE, osg::StateAttribute::OFF);
+    stateSet->setMode(GL_FOG, osg::StateAttribute::OFF);
+    stateSet->setMode(GL_DEPTH_TEST, osg::StateAttribute::OFF);
+    stateSet->setAttributeAndModes(new osg::PolygonMode(osg::PolygonMode::FRONT_AND_BACK,
+            osg::PolygonMode::FILL),
+            osg::StateAttribute::ON);
+    stateSet->setAttributeAndModes(new osg::AlphaFunc(osg::AlphaFunc::GREATER,
+            0.0f),
+            osg::StateAttribute::ON);
+    stateSet->setAttribute(new osg::ShadeModel(osg::ShadeModel::FLAT));
+    stateSet->setAttributeAndModes(new osg::BlendFunc(osg::BlendFunc::SRC_ALPHA,
+            osg::BlendFunc::ONE_MINUS_SRC_ALPHA),
+            osg::StateAttribute::ON);
+    if (!texture.valid()) {
+        texture = new osg::Texture2D;
+        texture->setTextureSize(textureWH, textureWH);
+        texture->setInternalFormat(GL_RGBA);
+        texture->setFilter(osg::Texture2D::MIN_FILTER,osg::Texture2D::LINEAR);
+        texture->setFilter(osg::Texture2D::MAG_FILTER,osg::Texture2D::LINEAR);
     }
-    else
-        set2D();
+    camera->attach(osg::Camera::COLOR_BUFFER, texture.get());
+    globals->get_renderer()->addCamera(camera.get(), false);
+    rtAvailable = true;
 }
 
-void FGODGauge::Clear(void) {
-    if(rtAvailable) {
-        glClear(GL_COLOR_BUFFER_BIT);
-    }
-    else {
-        glDisable(GL_BLEND);
-        glDisable(GL_ALPHA_TEST);
-          glColor4f(0.0f, 0.0f, 0.0f, 0.0f);
-        glRectf(-256.0, -256.0, 256.0, 256.0);
-        glEnable(GL_BLEND);
-        glBlendFunc( GL_ONE, GL_ONE_MINUS_SRC_ALPHA );
-        glEnable(GL_ALPHA_TEST);
+FGODGauge::~FGODGauge()
+{
+    if (camera.valid()) {
+        globals->get_renderer()->removeCamera(camera.get());
     }
 }
 
-void FGODGauge::endCapture(GLuint texID) {
-    glBindTexture(GL_TEXTURE_2D, texID);
-    // don't use mimaps if we don't update them
-    glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
-
-    glCopyTexSubImage2D(GL_TEXTURE_2D, 0, 0, 0, 0, 0, textureWH, textureWH);
-    if(rtAvailable)
-        rt->EndCapture();
-    else
-        set3D();
-    glBindTexture(GL_TEXTURE_2D, 0);
-}
-
-void FGODGauge::setSize(int viewSize) {
+void FGODGauge::setSize(int viewSize)
+{
     textureWH = viewSize;
-    glViewport(0, 0, textureWH, textureWH);
+    if (texture.valid()) {
+        texture->setTextureSize(textureWH, textureWH);
+    }
 }
 
-bool FGODGauge::serviceable(void) {
+bool FGODGauge::serviceable(void) 
+{
     return rtAvailable;
 }
 
 /**
- * Locate a texture SSG node in a branch.
+ * Replace a texture in the airplane model with the gauge texture.
  */
-static const char *strip_path(const char *name) {
-    /* Remove all leading path information. */
-    const char* seps = "\\/" ;
-    const char* fn = & name [ strlen ( name ) - 1 ] ;
-    for ( ; fn != name && strchr(seps,*fn) == NULL ; fn-- )
-        /* Search back for a seperator */ ;
-    if ( strchr(seps,*fn) != NULL )
-        fn++ ;
-    return fn ;
-}
 
-static ssgSimpleState *
-find_texture_node (ssgEntity * node, const char * name)
+class ReplaceStaticTextureVisitor : public osg::NodeVisitor
 {
-  if( node->isAKindOf( ssgTypeLeaf() ) ) {
-    ssgLeaf *leaf = (ssgLeaf *) node;
-    ssgSimpleState *state = (ssgSimpleState *) leaf->getState();
-    if( state ) {
-        ssgTexture *tex = state->getTexture();
-        if( tex ) {
-            const char * texture_name = tex->getFilename();
-            if (texture_name) {
-                texture_name = strip_path( texture_name );
-                if ( !strcmp(name, texture_name) )
-                    return state;
-            }
-        }
-    }
-  }
-  else {
-    int nKids = node->getNumKids();
-    for (int i = 0; i < nKids; i++) {
-      ssgSimpleState * result =
-        find_texture_node(((ssgBranch*)node)->getKid(i), name);
-      if (result != 0)
-        return result;
+public:
+    ReplaceStaticTextureVisitor(const std::string& name,
+        osg::Texture2D* _newTexture) :
+        osg::NodeVisitor(osg::NodeVisitor::TRAVERSE_ALL_CHILDREN),
+        newTexture(_newTexture)
+    {
+        textureFileName = osgDB::getSimpleFileName(name);
     }
-  } 
-  return 0;
-}
 
-void FGODGauge::set_texture(const char * name, GLuint new_texture) {
-    ssgEntity * root = globals->get_scenery()->get_aircraft_branch();
-    name = strip_path( name );
-    ssgSimpleState * node = find_texture_node( root, name );
-    if( node )
-        node->setTexture( new_texture );
-}
-
-void FGODGauge::set2D() {
-    glPushAttrib ( GL_ENABLE_BIT | GL_VIEWPORT_BIT  | GL_TRANSFORM_BIT | GL_LIGHTING_BIT ) ;
-
-    glDisable(GL_LIGHTING);
-    glEnable(GL_COLOR_MATERIAL);
-    glDisable(GL_CULL_FACE);
-    glDisable(GL_FOG);
-    glDisable(GL_DEPTH_TEST);
-    glClearColor(0.0, 0.0, 0.0, 0.0);
-    glEnable(GL_TEXTURE_2D);
-    glDisable(GL_SMOOTH);
-    glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
-    glBindTexture(GL_TEXTURE_2D, 0);
-
-    glViewport ( 0, 0, textureWH, textureWH ) ;
-    glMatrixMode   ( GL_PROJECTION ) ;
-    glPushMatrix   () ;
-    glLoadIdentity () ;
-    gluOrtho2D( -256.0, 256.0, -256.0, 256.0 );
-    glMatrixMode   ( GL_MODELVIEW ) ;
-    glPushMatrix   () ;
-    glLoadIdentity () ;
+    virtual void apply(osg::Node& node)
+    {
+        osg::StateSet* ss = node.getStateSet();
+        if (ss)
+            changeStateSetTexture(ss);
+        traverse(node);
+    }
 
-    glAlphaFunc(GL_GREATER, 0.0f);
+    virtual void apply(osg::Geode& node)
+    {
+        int numDrawables = node.getNumDrawables();
+        for (int i = 0; i < numDrawables; i++) {
+            osg::Drawable* drawable = node.getDrawable(i);
+            osg::StateSet* ss = drawable->getStateSet();
+            if (ss)
+                changeStateSetTexture(ss);
+        }
+        traverse(node);
+}
+protected:
+    void changeStateSetTexture(osg::StateSet *ss)
+    {
+        osg::Texture2D* tex
+                = dynamic_cast<osg::Texture2D*>(ss->getTextureAttribute(0,
+                osg::StateAttribute::TEXTURE));
+        if (!tex || tex == newTexture || !tex->getImage())
+            return;
+        std::string fileName = tex->getImage()->getFileName();
+        std::string simpleName = osgDB::getSimpleFileName(fileName);
+        if (osgDB::equalCaseInsensitive(textureFileName, simpleName))
+            ss->setTextureAttribute(0, newTexture);
+    }
+    std::string textureFileName;
+    osg::Texture2D* newTexture;
+};
 
+void FGODGauge::set_texture(const char * name, osg::Texture2D* new_texture)
+{
+    osg::Group* root = globals->get_scenery()->get_aircraft_branch();
+    ReplaceStaticTextureVisitor visitor(name, new_texture);
+    root->accept(visitor);
 }
 
-void FGODGauge::set3D() {
-    glMatrixMode   ( GL_PROJECTION ) ;
-    glPopMatrix    () ;
-    glMatrixMode   ( GL_MODELVIEW ) ;
-    glPopMatrix    () ;
-    glBlendFunc ( GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA ) ;
-    glPopAttrib    () ;
-}