2000-06-23 02:59:03 +00:00
|
|
|
|
// newmat.cxx -- class to handle material properties
|
|
|
|
|
//
|
|
|
|
|
// Written by Curtis Olson, started May 1998.
|
|
|
|
|
//
|
|
|
|
|
// Copyright (C) 1998 - 2000 Curtis L. Olson - curt@flightgear.org
|
|
|
|
|
//
|
|
|
|
|
// 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
|
|
|
|
|
// License, or (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful, but
|
|
|
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
|
// General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
|
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
|
|
|
//
|
|
|
|
|
// $Id$
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
|
# include <config.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <simgear/compiler.h>
|
|
|
|
|
|
2001-03-23 22:42:49 +00:00
|
|
|
|
#ifdef SG_MATH_EXCEPTION_CLASH
|
2000-06-23 02:59:03 +00:00
|
|
|
|
# include <math.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <simgear/debug/logstream.hxx>
|
2001-03-25 14:20:12 +00:00
|
|
|
|
#include <simgear/misc/sg_path.hxx>
|
|
|
|
|
#include <simgear/misc/sgstream.hxx>
|
2000-06-23 02:59:03 +00:00
|
|
|
|
|
2001-12-28 23:33:32 +00:00
|
|
|
|
#include <Main/globals.hxx>
|
|
|
|
|
#include <Main/fg_props.hxx>
|
|
|
|
|
|
2000-06-23 02:59:03 +00:00
|
|
|
|
#include "newmat.hxx"
|
|
|
|
|
|
|
|
|
|
|
2001-12-30 03:04:39 +00:00
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Local static functions.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Internal method to test whether a file exists.
|
|
|
|
|
*
|
|
|
|
|
* TODO: this should be moved to a SimGear library of local file
|
|
|
|
|
* functions.
|
|
|
|
|
*/
|
|
|
|
|
static inline bool
|
2001-12-28 23:33:32 +00:00
|
|
|
|
local_file_exists( const string& path ) {
|
|
|
|
|
sg_gzifstream in( path );
|
|
|
|
|
if ( ! in.is_open() ) {
|
|
|
|
|
return false;
|
|
|
|
|
} else {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2000-06-23 02:59:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2001-12-30 03:04:39 +00:00
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Constructors and destructor.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
2001-12-28 23:33:32 +00:00
|
|
|
|
|
|
|
|
|
FGNewMat::FGNewMat (const SGPropertyNode * props)
|
2000-06-23 02:59:03 +00:00
|
|
|
|
{
|
2001-12-29 13:19:09 +00:00
|
|
|
|
init();
|
2001-12-28 23:33:32 +00:00
|
|
|
|
read_properties(props);
|
|
|
|
|
build_ssg_state(false);
|
2000-06-23 04:55:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-03-12 05:37:55 +00:00
|
|
|
|
FGNewMat::FGNewMat (const string &texpath)
|
2001-12-28 23:33:32 +00:00
|
|
|
|
{
|
2001-12-29 13:19:09 +00:00
|
|
|
|
init();
|
2002-03-12 05:37:55 +00:00
|
|
|
|
texture_path = texpath;
|
2001-12-28 23:33:32 +00:00
|
|
|
|
build_ssg_state(true);
|
|
|
|
|
}
|
2000-06-23 04:55:55 +00:00
|
|
|
|
|
2001-12-28 23:33:32 +00:00
|
|
|
|
FGNewMat::FGNewMat (ssgSimpleState * s)
|
|
|
|
|
{
|
2001-12-29 13:19:09 +00:00
|
|
|
|
init();
|
2001-12-28 23:33:32 +00:00
|
|
|
|
set_ssg_state(s);
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-30 03:04:39 +00:00
|
|
|
|
FGNewMat::~FGNewMat (void)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Public methods.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
FGNewMat::read_properties (const SGPropertyNode * props)
|
|
|
|
|
{
|
|
|
|
|
// Get the path to the texture
|
|
|
|
|
string tname = props->getStringValue("texture", "unknown.rgb");
|
|
|
|
|
SGPath tpath(globals->get_fg_root());
|
|
|
|
|
tpath.append("Textures.high");
|
|
|
|
|
tpath.append(tname);
|
|
|
|
|
if (!local_file_exists(tpath.str())) {
|
|
|
|
|
tpath = SGPath(globals->get_fg_root());
|
|
|
|
|
tpath.append("Textures");
|
|
|
|
|
tpath.append(tname);
|
|
|
|
|
}
|
|
|
|
|
texture_path = tpath.str();
|
|
|
|
|
|
|
|
|
|
xsize = props->getDoubleValue("xsize", 0.0);
|
|
|
|
|
ysize = props->getDoubleValue("ysize", 0.0);
|
|
|
|
|
wrapu = props->getBoolValue("wrapu", true);
|
|
|
|
|
wrapv = props->getBoolValue("wrapv", true);
|
|
|
|
|
mipmap = props->getBoolValue("mipmap", true);
|
2002-03-26 16:10:53 +00:00
|
|
|
|
light_coverage = props->getDoubleValue("light-coverage", 0.0);
|
2001-12-30 03:04:39 +00:00
|
|
|
|
|
|
|
|
|
ambient[0] = props->getDoubleValue("ambient/r", 0.0);
|
|
|
|
|
ambient[1] = props->getDoubleValue("ambient/g", 0.0);
|
|
|
|
|
ambient[2] = props->getDoubleValue("ambient/b", 0.0);
|
|
|
|
|
ambient[3] = props->getDoubleValue("ambient/a", 0.0);
|
|
|
|
|
|
|
|
|
|
diffuse[0] = props->getDoubleValue("diffuse/r", 0.0);
|
|
|
|
|
diffuse[1] = props->getDoubleValue("diffuse/g", 0.0);
|
|
|
|
|
diffuse[2] = props->getDoubleValue("diffuse/b", 0.0);
|
|
|
|
|
diffuse[3] = props->getDoubleValue("diffuse/a", 0.0);
|
|
|
|
|
|
|
|
|
|
specular[0] = props->getDoubleValue("specular/r", 0.0);
|
|
|
|
|
specular[1] = props->getDoubleValue("specular/g", 0.0);
|
|
|
|
|
specular[2] = props->getDoubleValue("specular/b", 0.0);
|
|
|
|
|
specular[3] = props->getDoubleValue("specular/a", 0.0);
|
|
|
|
|
|
|
|
|
|
emission[0] = props->getDoubleValue("emissive/r", 0.0);
|
|
|
|
|
emission[1] = props->getDoubleValue("emissive/g", 0.0);
|
|
|
|
|
emission[2] = props->getDoubleValue("emissive/b", 0.0);
|
|
|
|
|
emission[3] = props->getDoubleValue("emissive/a", 0.0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
// Private methods.
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
FGNewMat::init ()
|
|
|
|
|
{
|
|
|
|
|
texture_path = "";
|
|
|
|
|
state = 0;
|
|
|
|
|
textured = 0;
|
|
|
|
|
nontextured = 0;
|
|
|
|
|
xsize = 0;
|
|
|
|
|
ysize = 0;
|
|
|
|
|
wrapu = true;
|
|
|
|
|
wrapv = true;
|
|
|
|
|
mipmap = true;
|
2002-03-26 16:10:53 +00:00
|
|
|
|
light_coverage = 0.0;
|
2001-12-30 03:04:39 +00:00
|
|
|
|
texture_loaded = false;
|
|
|
|
|
refcount = 0;
|
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
|
ambient[i] = diffuse[i] = specular[i] = emission[i] = 0.0;
|
|
|
|
|
}
|
2001-12-28 23:33:32 +00:00
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
FGNewMat::load_texture ()
|
2000-06-23 04:55:55 +00:00
|
|
|
|
{
|
2001-12-28 23:33:32 +00:00
|
|
|
|
if (texture_loaded) {
|
|
|
|
|
return false;
|
|
|
|
|
} else {
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Loading deferred texture " << texture_path );
|
|
|
|
|
textured->setTexture((char *)texture_path.c_str(), wrapu, wrapv, mipmap );
|
|
|
|
|
texture_loaded = true;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2000-06-23 02:59:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2001-12-30 03:04:39 +00:00
|
|
|
|
void
|
|
|
|
|
FGNewMat::build_ssg_state (bool defer_tex_load)
|
2000-06-23 02:59:03 +00:00
|
|
|
|
{
|
2001-12-28 23:33:32 +00:00
|
|
|
|
GLenum shade_model =
|
|
|
|
|
(fgGetBool("/sim/rendering/shading") ? GL_SMOOTH : GL_FLAT);
|
|
|
|
|
bool texture_default = fgGetBool("/sim/rendering/textures");
|
|
|
|
|
|
2000-06-23 02:59:03 +00:00
|
|
|
|
state = new ssgStateSelector(2);
|
2000-10-18 07:03:49 +00:00
|
|
|
|
state->ref();
|
2000-10-18 00:48:12 +00:00
|
|
|
|
|
2000-06-23 02:59:03 +00:00
|
|
|
|
textured = new ssgSimpleState();
|
2000-10-18 00:48:12 +00:00
|
|
|
|
textured->ref();
|
|
|
|
|
|
2000-06-23 02:59:03 +00:00
|
|
|
|
nontextured = new ssgSimpleState();
|
2000-10-18 00:48:12 +00:00
|
|
|
|
nontextured->ref();
|
2000-06-23 02:59:03 +00:00
|
|
|
|
|
|
|
|
|
// Set up the textured state
|
|
|
|
|
textured->setShadeModel( shade_model );
|
2000-12-04 05:23:06 +00:00
|
|
|
|
textured->enable( GL_LIGHTING );
|
2000-06-23 02:59:03 +00:00
|
|
|
|
textured->enable ( GL_CULL_FACE ) ;
|
|
|
|
|
textured->enable( GL_TEXTURE_2D );
|
|
|
|
|
textured->disable( GL_BLEND );
|
|
|
|
|
textured->disable( GL_ALPHA_TEST );
|
2002-01-20 03:52:36 +00:00
|
|
|
|
#if 0
|
|
|
|
|
# ifdef GL_EXT_texture_filter_anisotropic
|
|
|
|
|
float max_anisotropy;
|
|
|
|
|
glGetFloatv( GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT, &max_anisotropy );
|
|
|
|
|
glTexParameterf( GL_TEXTURE_2D, GL_TEXTURE_MAX_ANISOTROPY_EXT,
|
|
|
|
|
max_anisotropy );
|
|
|
|
|
cout << "Max anisotropy = " << max_anisotropy << endl;
|
|
|
|
|
# endif
|
|
|
|
|
#endif
|
2001-04-11 02:47:15 +00:00
|
|
|
|
if ( !defer_tex_load ) {
|
2001-12-28 23:33:32 +00:00
|
|
|
|
textured->setTexture( (char *)texture_path.c_str(), wrapu, wrapv );
|
2001-04-11 02:47:15 +00:00
|
|
|
|
texture_loaded = true;
|
|
|
|
|
} else {
|
|
|
|
|
texture_loaded = false;
|
|
|
|
|
}
|
2000-06-23 02:59:03 +00:00
|
|
|
|
textured->enable( GL_COLOR_MATERIAL );
|
|
|
|
|
textured->setColourMaterial( GL_AMBIENT_AND_DIFFUSE );
|
|
|
|
|
textured->setMaterial( GL_EMISSION, 0, 0, 0, 1 );
|
|
|
|
|
textured->setMaterial( GL_SPECULAR, 0, 0, 0, 1 );
|
|
|
|
|
|
|
|
|
|
// Set up the coloured state
|
|
|
|
|
nontextured->enable( GL_LIGHTING );
|
|
|
|
|
nontextured->setShadeModel( shade_model );
|
|
|
|
|
nontextured->enable ( GL_CULL_FACE ) ;
|
|
|
|
|
nontextured->disable( GL_TEXTURE_2D );
|
|
|
|
|
nontextured->disable( GL_BLEND );
|
|
|
|
|
nontextured->disable( GL_ALPHA_TEST );
|
|
|
|
|
nontextured->disable( GL_COLOR_MATERIAL );
|
|
|
|
|
|
|
|
|
|
nontextured->setMaterial ( GL_AMBIENT,
|
|
|
|
|
ambient[0], ambient[1],
|
|
|
|
|
ambient[2], ambient[3] ) ;
|
|
|
|
|
nontextured->setMaterial ( GL_DIFFUSE,
|
|
|
|
|
diffuse[0], diffuse[1],
|
|
|
|
|
diffuse[2], diffuse[3] ) ;
|
|
|
|
|
nontextured->setMaterial ( GL_SPECULAR,
|
|
|
|
|
specular[0], specular[1],
|
|
|
|
|
specular[2], specular[3] ) ;
|
|
|
|
|
nontextured->setMaterial ( GL_EMISSION,
|
|
|
|
|
emission[0], emission[1],
|
|
|
|
|
emission[2], emission[3] ) ;
|
|
|
|
|
|
|
|
|
|
state->setStep( 0, textured ); // textured
|
|
|
|
|
state->setStep( 1, nontextured ); // untextured
|
|
|
|
|
|
|
|
|
|
// Choose the appropriate starting state.
|
|
|
|
|
if ( texture_default ) {
|
|
|
|
|
state->selectStep(0);
|
|
|
|
|
} else {
|
|
|
|
|
state->selectStep(1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2001-12-30 03:04:39 +00:00
|
|
|
|
void FGNewMat::set_ssg_state( ssgSimpleState *s )
|
|
|
|
|
{
|
2000-06-23 04:55:55 +00:00
|
|
|
|
state = new ssgStateSelector(2);
|
2000-10-18 07:03:49 +00:00
|
|
|
|
state->ref();
|
|
|
|
|
|
2000-06-23 04:55:55 +00:00
|
|
|
|
textured = s;
|
2000-10-18 00:48:12 +00:00
|
|
|
|
|
2000-06-23 04:55:55 +00:00
|
|
|
|
nontextured = new ssgSimpleState();
|
2000-10-18 00:48:12 +00:00
|
|
|
|
nontextured->ref();
|
2000-06-23 04:55:55 +00:00
|
|
|
|
|
|
|
|
|
// Set up the coloured state
|
|
|
|
|
nontextured->enable( GL_LIGHTING );
|
|
|
|
|
nontextured->setShadeModel( GL_FLAT );
|
|
|
|
|
nontextured->enable ( GL_CULL_FACE ) ;
|
|
|
|
|
nontextured->disable( GL_TEXTURE_2D );
|
|
|
|
|
nontextured->disable( GL_BLEND );
|
|
|
|
|
nontextured->disable( GL_ALPHA_TEST );
|
|
|
|
|
nontextured->disable( GL_COLOR_MATERIAL );
|
|
|
|
|
|
|
|
|
|
/* cout << "ambient = " << ambient[0] << "," << ambient[1]
|
|
|
|
|
<< "," << ambient[2] << endl; */
|
|
|
|
|
nontextured->setMaterial ( GL_AMBIENT,
|
|
|
|
|
ambient[0], ambient[1],
|
|
|
|
|
ambient[2], ambient[3] ) ;
|
|
|
|
|
nontextured->setMaterial ( GL_DIFFUSE,
|
|
|
|
|
diffuse[0], diffuse[1],
|
|
|
|
|
diffuse[2], diffuse[3] ) ;
|
|
|
|
|
nontextured->setMaterial ( GL_SPECULAR,
|
|
|
|
|
specular[0], specular[1],
|
|
|
|
|
specular[2], specular[3] ) ;
|
|
|
|
|
nontextured->setMaterial ( GL_EMISSION,
|
|
|
|
|
emission[0], emission[1],
|
|
|
|
|
emission[2], emission[3] ) ;
|
|
|
|
|
|
|
|
|
|
state->setStep( 0, textured ); // textured
|
|
|
|
|
state->setStep( 1, nontextured ); // untextured
|
|
|
|
|
|
|
|
|
|
// Choose the appropriate starting state.
|
|
|
|
|
state->selectStep(0);
|
|
|
|
|
}
|
|
|
|
|
|
2001-12-30 03:04:39 +00:00
|
|
|
|
// end of newmat.cxx
|