2001-05-19 16:59:43 +00:00
|
|
|
// tileentry.cxx -- routines to handle a scenery tile
|
1999-06-12 21:15:27 +00:00
|
|
|
//
|
|
|
|
// Written by Curtis Olson, started May 1998.
|
|
|
|
//
|
2004-11-19 22:10:41 +00:00
|
|
|
// Copyright (C) 1998 - 2001 Curtis L. Olson - http://www.flightgear.org/~curt
|
1999-06-12 21:15:27 +00:00
|
|
|
//
|
|
|
|
// 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
|
2006-02-21 01:16:04 +00:00
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
1999-06-12 21:15:27 +00:00
|
|
|
//
|
|
|
|
// $Id$
|
|
|
|
|
|
|
|
|
2001-05-15 22:30:39 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
2000-02-15 03:30:01 +00:00
|
|
|
#include <simgear/compiler.h>
|
2004-10-10 19:19:23 +00:00
|
|
|
#include <plib/ul.h>
|
2003-08-29 16:46:21 +00:00
|
|
|
#include <Main/main.hxx>
|
|
|
|
|
|
|
|
|
2003-05-14 18:33:56 +00:00
|
|
|
#include STL_STRING
|
2007-11-29 23:59:14 +00:00
|
|
|
#include <sstream>
|
2003-05-14 18:33:56 +00:00
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
#include <osg/Array>
|
|
|
|
#include <osg/Geometry>
|
|
|
|
#include <osg/Geode>
|
|
|
|
#include <osg/LOD>
|
|
|
|
#include <osg/MatrixTransform>
|
2007-12-14 22:51:56 +00:00
|
|
|
#include <osg/Math>
|
2006-10-29 19:30:21 +00:00
|
|
|
#include <osg/NodeCallback>
|
|
|
|
#include <osg/Switch>
|
|
|
|
|
2007-11-29 23:59:47 +00:00
|
|
|
#include <osgDB/FileNameUtils>
|
|
|
|
#include <osgDB/ReaderWriter>
|
2007-07-29 22:34:15 +00:00
|
|
|
#include <osgDB/ReadFile>
|
2007-11-29 23:59:47 +00:00
|
|
|
#include <osgDB/Registry>
|
2007-07-29 22:34:15 +00:00
|
|
|
|
2000-02-16 23:01:03 +00:00
|
|
|
#include <simgear/bucket/newbucket.hxx>
|
|
|
|
#include <simgear/debug/logstream.hxx>
|
2003-12-30 05:57:25 +00:00
|
|
|
#include <simgear/math/polar3d.hxx>
|
2001-05-16 06:00:02 +00:00
|
|
|
#include <simgear/math/sg_geodesy.hxx>
|
2001-03-29 01:42:31 +00:00
|
|
|
#include <simgear/math/sg_random.h>
|
|
|
|
#include <simgear/misc/sgstream.hxx>
|
2003-05-12 21:34:29 +00:00
|
|
|
#include <simgear/scene/material/mat.hxx>
|
|
|
|
#include <simgear/scene/material/matlib.hxx>
|
2007-11-29 23:59:47 +00:00
|
|
|
#include <simgear/scene/model/ModelRegistry.hxx>
|
2003-05-14 19:22:24 +00:00
|
|
|
#include <simgear/scene/tgdb/apt_signs.hxx>
|
2003-05-28 21:01:55 +00:00
|
|
|
#include <simgear/scene/tgdb/obj.hxx>
|
2007-07-29 22:34:15 +00:00
|
|
|
#include <simgear/scene/tgdb/SGReaderWriterBTGOptions.hxx>
|
Mathias:
I have done a patch to eliminate the jitter of 3D-objects near the viewpoint
(for example 3D cockpit objects).
The problem is the roundoff accuracy of the float values used in the
scenegraph together with the transforms of the eyepoint relative to the
scenery center.
The solution will be to move the scenery center near the view point.
This way floats relative accuracy is enough to show a stable picture.
To get that right I have introduced a transform node for the scenegraph which
is responsible for that shift and uses double values as long as possible.
The scenery subsystem now has a list of all those transforms required to place
objects in the world and will tell all those transforms that the scenery
center has changed when the set_scenery_center() of the scenery subsystem is
called.
The problem was not solvable by SGModelPlacement and SGLocation, since not all
objects, especially the scenery, are placed using these classes.
The first approach was to have the scenery center exactly at the eyepoint.
This works well for the cockpit.
But then the ground jitters a bit below the aircraft. With our default views
you can't see that, but that F-18 has a camera view below the left engine
intake with the nose gear and the ground in its field of view, here I could
see that.
Having the scenery center constant will still have this roundoff problems, but
like it is now too, the roundoff error here is exactly the same in each
frame, so you will not notice any jitter.
The real solution is now to keep the scenery center constant as long as it is
in a ball of 30m radius around the view point. If the scenery center is
outside this ball, just put it at the view point.
As a sideeffect of now beeing able to switch the scenery center in the whole
scenegraph with one function call, I was able to remove a one half of a
problem when switching views, where the scenery center was far off for one or
two frames past switching from one view to the next. Also included is a fix
to the other half of this problem, where the view position was not yet copied
into a view when it is switched (at least under glut). This was responsible
for the 'Error: ...' messages of the cloud subsystem when views were
switched.
2005-04-29 14:38:24 +00:00
|
|
|
#include <simgear/scene/model/placementtrans.hxx>
|
2007-05-08 06:12:26 +00:00
|
|
|
#include <simgear/scene/util/SGUpdateVisitor.hxx>
|
1999-06-12 21:15:27 +00:00
|
|
|
|
2000-12-04 05:24:38 +00:00
|
|
|
#include <Aircraft/aircraft.hxx>
|
2000-12-05 14:27:27 +00:00
|
|
|
#include <Include/general.hxx>
|
2003-05-14 20:48:31 +00:00
|
|
|
#include <Main/fg_props.hxx>
|
2000-12-04 05:24:38 +00:00
|
|
|
#include <Main/globals.hxx>
|
2001-07-22 19:51:16 +00:00
|
|
|
#include <Main/viewer.hxx>
|
2000-12-04 05:24:38 +00:00
|
|
|
#include <Scenery/scenery.hxx>
|
2000-12-04 23:25:05 +00:00
|
|
|
#include <Time/light.hxx>
|
2000-12-04 05:24:38 +00:00
|
|
|
|
1999-06-12 21:15:27 +00:00
|
|
|
#include "tileentry.hxx"
|
2001-05-19 16:59:43 +00:00
|
|
|
#include "tilemgr.hxx"
|
1999-06-12 21:15:27 +00:00
|
|
|
|
2003-05-14 18:33:56 +00:00
|
|
|
SG_USING_STD(string);
|
2007-11-29 23:59:47 +00:00
|
|
|
using namespace simgear;
|
2003-05-14 18:33:56 +00:00
|
|
|
|
2007-05-26 13:51:23 +00:00
|
|
|
// FIXME: investigate what huge update flood is clamped away here ...
|
2007-05-08 06:12:26 +00:00
|
|
|
class FGTileUpdateCallback : public osg::NodeCallback {
|
|
|
|
public:
|
|
|
|
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv)
|
|
|
|
{
|
|
|
|
assert(dynamic_cast<SGUpdateVisitor*>(nv));
|
|
|
|
SGUpdateVisitor* updateVisitor = static_cast<SGUpdateVisitor*>(nv);
|
|
|
|
|
|
|
|
osg::Vec3 center = node->getBound().center();
|
2007-05-26 13:51:23 +00:00
|
|
|
double distance = dist(updateVisitor->getGlobalEyePos(),
|
2007-05-08 06:12:26 +00:00
|
|
|
SGVec3d(center[0], center[1], center[2]));
|
2007-05-26 13:51:23 +00:00
|
|
|
if (updateVisitor->getVisibility() + node->getBound().radius() < distance)
|
2007-05-08 06:12:26 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
traverse(node, nv);
|
|
|
|
}
|
|
|
|
};
|
1999-06-12 21:15:27 +00:00
|
|
|
|
2007-12-14 22:51:56 +00:00
|
|
|
namespace
|
|
|
|
{
|
2008-02-21 22:50:05 +00:00
|
|
|
// Update the timestamp on a tile whenever it is in view.
|
|
|
|
|
2007-12-14 22:51:56 +00:00
|
|
|
class TileCullCallback : public osg::NodeCallback
|
|
|
|
{
|
|
|
|
public:
|
2008-02-21 21:36:20 +00:00
|
|
|
TileCullCallback() : _timeStamp(0) {}
|
2007-12-14 22:51:56 +00:00
|
|
|
TileCullCallback(const TileCullCallback& tc, const osg::CopyOp& copyOp) :
|
|
|
|
NodeCallback(tc, copyOp), _timeStamp(tc._timeStamp)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void operator()(osg::Node* node, osg::NodeVisitor* nv);
|
|
|
|
double getTimeStamp() const { return _timeStamp; }
|
|
|
|
void setTimeStamp(double timeStamp) { _timeStamp = timeStamp; }
|
|
|
|
protected:
|
|
|
|
double _timeStamp;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void TileCullCallback::operator()(osg::Node* node, osg::NodeVisitor* nv)
|
|
|
|
{
|
|
|
|
if (nv->getFrameStamp())
|
|
|
|
_timeStamp = nv->getFrameStamp()->getReferenceTime();
|
|
|
|
traverse(node, nv);
|
|
|
|
}
|
|
|
|
|
|
|
|
double FGTileEntry::get_timestamp() const
|
|
|
|
{
|
|
|
|
if (_node.valid()) {
|
|
|
|
return (dynamic_cast<TileCullCallback*>(_node->getCullCallback()))
|
|
|
|
->getTimeStamp();
|
|
|
|
} else
|
|
|
|
return DBL_MAX;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FGTileEntry::set_timestamp(double time_ms)
|
|
|
|
{
|
|
|
|
if (_node.valid()) {
|
|
|
|
TileCullCallback* cb
|
|
|
|
= dynamic_cast<TileCullCallback*>(_node->getCullCallback());
|
|
|
|
if (cb)
|
|
|
|
cb->setTimeStamp(time_ms);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1999-06-12 21:15:27 +00:00
|
|
|
// Constructor
|
2001-03-29 01:42:31 +00:00
|
|
|
FGTileEntry::FGTileEntry ( const SGBucket& b )
|
2007-05-08 06:12:26 +00:00
|
|
|
: tile_bucket( b ),
|
2007-12-14 22:51:56 +00:00
|
|
|
_node( new osg::LOD ),
|
2003-12-30 15:12:04 +00:00
|
|
|
is_inner_ring(false),
|
2007-12-14 22:51:56 +00:00
|
|
|
free_tracker(0),
|
|
|
|
tileFileName(b.gen_index_str())
|
1999-06-12 21:15:27 +00:00
|
|
|
{
|
2007-12-14 22:51:56 +00:00
|
|
|
_node->setUpdateCallback(new FGTileUpdateCallback);
|
|
|
|
_node->setCullCallback(new TileCullCallback);
|
|
|
|
tileFileName += ".stg";
|
|
|
|
_node->setName(tileFileName);
|
2007-12-20 23:20:51 +00:00
|
|
|
// Give a default LOD range so that traversals that traverse
|
|
|
|
// active children (like the groundcache lookup) will work before
|
|
|
|
// tile manager has had a chance to update this node.
|
|
|
|
_node->setRange(0, 0.0, 10000.0);
|
1999-06-12 21:15:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Destructor
|
2007-12-14 22:51:56 +00:00
|
|
|
FGTileEntry::~FGTileEntry ()
|
|
|
|
{
|
1999-06-12 21:15:27 +00:00
|
|
|
}
|
|
|
|
|
2008-03-04 09:02:24 +00:00
|
|
|
void WorldCoordinate( osg::Matrix& obj_pos, double lat,
|
2001-05-23 22:28:38 +00:00
|
|
|
double lon, double elev, double hdg )
|
|
|
|
{
|
|
|
|
double lon_rad = lon * SGD_DEGREES_TO_RADIANS;
|
|
|
|
double lat_rad = lat * SGD_DEGREES_TO_RADIANS;
|
|
|
|
double hdg_rad = hdg * SGD_DEGREES_TO_RADIANS;
|
|
|
|
|
|
|
|
// setup transforms
|
|
|
|
Point3D geod( lon_rad, lat_rad, elev );
|
|
|
|
Point3D world_pos = sgGeodToCart( geod );
|
|
|
|
|
2007-05-08 06:12:26 +00:00
|
|
|
double sin_lat = sin( lat_rad );
|
|
|
|
double cos_lat = cos( lat_rad );
|
|
|
|
double cos_lon = cos( lon_rad );
|
|
|
|
double sin_lon = sin( lon_rad );
|
|
|
|
double sin_hdg = sin( hdg_rad ) ;
|
|
|
|
double cos_hdg = cos( hdg_rad ) ;
|
2001-05-23 22:28:38 +00:00
|
|
|
|
2007-11-29 23:59:14 +00:00
|
|
|
obj_pos(0, 0) = cos_hdg * sin_lat * cos_lon - sin_hdg * sin_lon;
|
|
|
|
obj_pos(0, 1) = cos_hdg * sin_lat * sin_lon + sin_hdg * cos_lon;
|
|
|
|
obj_pos(0, 2) = -cos_hdg * cos_lat;
|
|
|
|
obj_pos(0, 3) = SG_ZERO;
|
|
|
|
|
|
|
|
obj_pos(1, 0) = -sin_hdg * sin_lat * cos_lon - cos_hdg * sin_lon;
|
|
|
|
obj_pos(1, 1) = -sin_hdg * sin_lat * sin_lon + cos_hdg * cos_lon;
|
|
|
|
obj_pos(1, 2) = sin_hdg * cos_lat;
|
|
|
|
obj_pos(1, 3) = SG_ZERO;
|
|
|
|
|
|
|
|
obj_pos(2, 0) = cos_lat * cos_lon;
|
|
|
|
obj_pos(2, 1) = cos_lat * sin_lon;
|
|
|
|
obj_pos(2, 2) = sin_lat;
|
|
|
|
obj_pos(2, 3) = SG_ZERO;
|
|
|
|
|
|
|
|
obj_pos(3, 0) = world_pos.x();
|
|
|
|
obj_pos(3, 1) = world_pos.y();
|
|
|
|
obj_pos(3, 2) = world_pos.z();
|
|
|
|
obj_pos(3, 3) = SG_ONE ;
|
2001-05-23 22:28:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-25 23:59:04 +00:00
|
|
|
// Free "n" leaf elements of an ssg tree. returns the number of
|
|
|
|
// elements freed. An empty branch node is considered a leaf. This
|
|
|
|
// is intended to spread the load of freeing a complex tile out over
|
|
|
|
// several frames.
|
2006-10-29 19:30:21 +00:00
|
|
|
static int fgPartialFreeSSGtree( osg::Group *b, int n ) {
|
2007-05-08 06:12:26 +00:00
|
|
|
int num_deletes = b->getNumChildren();
|
2002-07-29 05:07:38 +00:00
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
b->removeChildren(0, b->getNumChildren());
|
|
|
|
|
2002-07-29 05:07:38 +00:00
|
|
|
return num_deletes;
|
2002-07-25 23:59:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-06-23 00:30:04 +00:00
|
|
|
// Clean up the memory used by this tile and delete the arrays used by
|
1999-06-29 14:57:00 +00:00
|
|
|
// ssg as well as the whole ssg branch
|
2002-07-25 23:59:04 +00:00
|
|
|
bool FGTileEntry::free_tile() {
|
|
|
|
int delete_size = 100;
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG,
|
2002-09-23 15:27:46 +00:00
|
|
|
"FREEING TILE = (" << tile_bucket << ")" );
|
1999-10-27 00:52:25 +00:00
|
|
|
|
2002-08-06 18:50:12 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "(start) free_tracker = " << free_tracker );
|
|
|
|
|
2002-07-25 23:59:04 +00:00
|
|
|
if ( !(free_tracker & NODES) ) {
|
|
|
|
free_tracker |= NODES;
|
|
|
|
} else if ( !(free_tracker & VEC_PTRS) ) {
|
|
|
|
free_tracker |= VEC_PTRS;
|
|
|
|
} else if ( !(free_tracker & TERRA_NODE) ) {
|
|
|
|
// delete the terrain branch (this should already have been
|
|
|
|
// disconnected from the scene graph)
|
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "FREEING terra_transform" );
|
2007-12-14 22:51:56 +00:00
|
|
|
if ( fgPartialFreeSSGtree( _node.get(), delete_size ) == 0 ) {
|
|
|
|
_node = 0;
|
2002-07-25 23:59:04 +00:00
|
|
|
free_tracker |= TERRA_NODE;
|
|
|
|
}
|
2003-05-08 03:29:49 +00:00
|
|
|
} else if ( !(free_tracker & LIGHTMAPS) ) {
|
|
|
|
free_tracker |= LIGHTMAPS;
|
2002-07-25 23:59:04 +00:00
|
|
|
} else {
|
|
|
|
return true;
|
2001-09-19 22:30:41 +00:00
|
|
|
}
|
2002-07-25 23:59:04 +00:00
|
|
|
|
2002-08-06 18:50:12 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "(end) free_tracker = " << free_tracker );
|
|
|
|
|
2002-07-25 23:59:04 +00:00
|
|
|
// if we fall down to here, we still have work todo, return false
|
|
|
|
return false;
|
1999-06-12 21:15:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2000-12-04 05:24:38 +00:00
|
|
|
// Update the ssg transform node for this tile so it can be
|
|
|
|
// properly drawn relative to our (0,0,0) point
|
2007-05-08 06:12:26 +00:00
|
|
|
void FGTileEntry::prep_ssg_node(float vis) {
|
2007-12-14 22:51:56 +00:00
|
|
|
if (!is_loaded())
|
|
|
|
return;
|
2002-10-30 21:59:05 +00:00
|
|
|
// visibility can change from frame to frame so we update the
|
|
|
|
// range selector cutoff's each time.
|
2007-12-14 22:51:56 +00:00
|
|
|
float bounding_radius = _node->getChild(0)->getBound().radius();
|
|
|
|
_node->setRange( 0, 0, vis + bounding_radius );
|
1999-07-04 07:37:30 +00:00
|
|
|
}
|
2001-03-29 01:42:31 +00:00
|
|
|
|
2003-05-14 18:33:56 +00:00
|
|
|
bool FGTileEntry::obj_load( const string& path,
|
2007-05-08 06:12:26 +00:00
|
|
|
osg::Group *geometry, bool is_base )
|
2001-03-29 01:42:31 +00:00
|
|
|
{
|
2003-05-14 20:48:31 +00:00
|
|
|
bool use_random_objects =
|
|
|
|
fgGetBool("/sim/rendering/random-objects", true);
|
2008-02-07 23:02:51 +00:00
|
|
|
|
|
|
|
bool use_random_vegetation =
|
|
|
|
fgGetBool("/sim/rendering/random-vegetation", true);
|
2003-05-14 20:48:31 +00:00
|
|
|
|
2001-03-29 01:42:31 +00:00
|
|
|
// try loading binary format
|
2007-07-29 22:34:15 +00:00
|
|
|
osg::ref_ptr<SGReaderWriterBTGOptions> options
|
|
|
|
= new SGReaderWriterBTGOptions();
|
|
|
|
options->setMatlib(globals->get_matlib());
|
|
|
|
options->setCalcLights(is_base);
|
|
|
|
options->setUseRandomObjects(use_random_objects);
|
2008-02-07 23:02:51 +00:00
|
|
|
options->setUseRandomVegetation(use_random_vegetation);
|
2007-07-29 22:34:15 +00:00
|
|
|
osg::Node* node = osgDB::readNodeFile(path, options.get());
|
2007-05-08 06:12:26 +00:00
|
|
|
if (node)
|
|
|
|
geometry->addChild(node);
|
2001-03-29 01:42:31 +00:00
|
|
|
|
2007-05-08 06:12:26 +00:00
|
|
|
return node;
|
2001-03-29 01:42:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
typedef enum {
|
|
|
|
OBJECT,
|
|
|
|
OBJECT_SHARED,
|
|
|
|
OBJECT_STATIC,
|
2006-04-14 14:50:11 +00:00
|
|
|
OBJECT_SIGN,
|
2005-12-02 21:46:33 +00:00
|
|
|
OBJECT_RUNWAY_SIGN
|
|
|
|
} object_type;
|
|
|
|
|
|
|
|
|
|
|
|
// storage class for deferred object processing in FGTileEntry::load()
|
|
|
|
struct Object {
|
2005-12-03 10:20:35 +00:00
|
|
|
Object(object_type t, const string& token, const SGPath& p, istream& in)
|
2005-12-02 21:46:33 +00:00
|
|
|
: type(t), path(p)
|
|
|
|
{
|
|
|
|
in >> name;
|
|
|
|
if (type != OBJECT)
|
|
|
|
in >> lon >> lat >> elev >> hdg;
|
|
|
|
in >> ::skipeol;
|
|
|
|
|
|
|
|
if (type == OBJECT)
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG(SG_TERRAIN, SG_INFO, " " << token << " " << name);
|
2005-12-02 21:46:33 +00:00
|
|
|
else
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG(SG_TERRAIN, SG_INFO, " " << token << " " << name << " lon=" <<
|
|
|
|
lon << " lat=" << lat << " elev=" << elev << " hdg=" << hdg);
|
2005-12-02 21:46:33 +00:00
|
|
|
}
|
|
|
|
object_type type;
|
|
|
|
string name;
|
|
|
|
SGPath path;
|
|
|
|
double lon, lat, elev, hdg;
|
|
|
|
};
|
|
|
|
|
2007-11-29 23:59:14 +00:00
|
|
|
// Work in progress... load the tile based entirely by name cuz that's
|
|
|
|
// what we'll want to do with the database pager.
|
|
|
|
|
|
|
|
osg::Node*
|
|
|
|
FGTileEntry::loadTileByName(const string& index_str,
|
|
|
|
const string_list &path_list)
|
|
|
|
{
|
|
|
|
long tileIndex;
|
|
|
|
{
|
|
|
|
istringstream idxStream(index_str);
|
|
|
|
idxStream >> tileIndex;
|
|
|
|
}
|
|
|
|
SGBucket tile_bucket(tileIndex);
|
|
|
|
const string basePath = tile_bucket.gen_base_path();
|
|
|
|
|
2003-08-08 20:11:22 +00:00
|
|
|
bool found_tile_base = false;
|
2001-03-29 01:42:31 +00:00
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
SGPath object_base;
|
|
|
|
vector<const Object*> objects;
|
2001-03-29 01:42:31 +00:00
|
|
|
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_INFO, "Loading tile " << index_str );
|
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
// scan and parse all files and store information
|
2005-12-03 10:20:35 +00:00
|
|
|
for (unsigned int i = 0; i < path_list.size(); i++) {
|
|
|
|
// If we found a terrain tile in Terrain/, we have to process the
|
|
|
|
// Objects/ dir in the same group, too, before we can stop scanning.
|
|
|
|
// FGGlobals::set_fg_scenery() inserts an empty string to path_list
|
|
|
|
// as marker.
|
|
|
|
if (path_list[i].empty()) {
|
|
|
|
if (found_tile_base)
|
|
|
|
break;
|
|
|
|
else
|
|
|
|
continue;
|
|
|
|
}
|
2003-10-01 22:49:06 +00:00
|
|
|
|
|
|
|
bool has_base = false;
|
2001-03-29 01:42:31 +00:00
|
|
|
|
Melchior FRANZ:
Wouldn't it be better to prepare the whole list of paths (or two
separate ones for Terrain/Objects if necessary) in FGGlobals::set_fg_scenery,
and to pass the vector<string>s to FGTileEntry::load? It doesn't seem to make
a lot of sense to split the path up, modify it, mount it together to one string
again, and then let FGTileEntry::load split it up again.
Here we go:
Main/globals.cxx
================
As fg_scenery is now a string_list, we don't need initialization. Furthermore,
this list is cleared with every set_fg_scenery() call.
ctor: create default dir from fg_root if necessary. Otherwise check all paths
of --fg-scenery/FG_SCENERY: If the path doesn't exist, ignore it. If it contains
a dir Terrain and/or Objects, then only add that to the list. If it contains
neither, then use the path as is.
Scenery/tileentry.cxx
=====================
Trivial: don't split a "base path", but use the given path_list as is.
(I considered a variable name "path_list" better suited than "search".)
Scenery/FGTileLoader.cxx
========================
No more fiddling with sub-paths. This has to be delivered by get_fg_scenery
already.
2004-06-08 15:32:09 +00:00
|
|
|
SGPath tile_path = path_list[i];
|
2007-11-29 23:59:14 +00:00
|
|
|
tile_path.append(basePath);
|
2002-09-23 15:27:46 +00:00
|
|
|
|
2003-08-08 20:11:22 +00:00
|
|
|
SGPath basename = tile_path;
|
|
|
|
basename.append( index_str );
|
2002-09-23 15:27:46 +00:00
|
|
|
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_INFO, " Trying " << basename.str() );
|
2002-09-23 15:27:46 +00:00
|
|
|
|
|
|
|
|
2003-08-08 20:11:22 +00:00
|
|
|
// Check for master .stg (scene terra gear) file
|
|
|
|
SGPath stg_name = basename;
|
|
|
|
stg_name.concat( ".stg" );
|
|
|
|
|
|
|
|
sg_gzifstream in( stg_name.str() );
|
2005-12-02 21:46:33 +00:00
|
|
|
if ( !in.is_open() )
|
|
|
|
continue;
|
2003-08-08 20:11:22 +00:00
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
while ( ! in.eof() ) {
|
|
|
|
string token;
|
|
|
|
in >> token;
|
2003-08-08 20:11:22 +00:00
|
|
|
|
2007-07-11 15:18:24 +00:00
|
|
|
if ( token.empty() || token[0] == '#' ) {
|
2005-12-02 21:46:33 +00:00
|
|
|
in >> ::skipeol;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Load only once (first found)
|
|
|
|
if ( token == "OBJECT_BASE" ) {
|
|
|
|
string name;
|
|
|
|
in >> name >> ::skipws;
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_INFO, " " << token << " " << name );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
if (!found_tile_base) {
|
|
|
|
found_tile_base = true;
|
|
|
|
has_base = true;
|
|
|
|
|
|
|
|
object_base = tile_path;
|
|
|
|
object_base.append(name);
|
|
|
|
|
|
|
|
} else
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG(SG_TERRAIN, SG_INFO, " (skipped)");
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
// Load only if base is not in another file
|
|
|
|
} else if ( token == "OBJECT" ) {
|
|
|
|
if (!found_tile_base || has_base)
|
|
|
|
objects.push_back(new Object(OBJECT, token, tile_path, in));
|
|
|
|
else {
|
|
|
|
string name;
|
|
|
|
in >> name >> ::skipeol;
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG(SG_TERRAIN, SG_INFO, " " << token << " "
|
|
|
|
<< name << " (skipped)");
|
2004-12-08 14:45:47 +00:00
|
|
|
}
|
2003-08-08 20:11:22 +00:00
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
// Always OK to load
|
|
|
|
} else if ( token == "OBJECT_STATIC" ) {
|
|
|
|
objects.push_back(new Object(OBJECT_STATIC, token, tile_path, in));
|
2002-09-23 15:27:46 +00:00
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
} else if ( token == "OBJECT_SHARED" ) {
|
|
|
|
objects.push_back(new Object(OBJECT_SHARED, token, tile_path, in));
|
|
|
|
|
2006-04-14 14:50:11 +00:00
|
|
|
} else if ( token == "OBJECT_SIGN" ) {
|
|
|
|
objects.push_back(new Object(OBJECT_SIGN, token, tile_path, in));
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
} else if ( token == "OBJECT_RUNWAY_SIGN" ) {
|
|
|
|
objects.push_back(new Object(OBJECT_RUNWAY_SIGN, token, tile_path, in));
|
|
|
|
|
|
|
|
} else {
|
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG,
|
|
|
|
"Unknown token '" << token << "' in " << stg_name.str() );
|
|
|
|
in >> ::skipws;
|
2002-09-23 15:27:46 +00:00
|
|
|
}
|
|
|
|
}
|
2003-08-08 20:11:22 +00:00
|
|
|
}
|
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
// obj_load() will generate ground lighting for us ...
|
2006-10-29 19:30:21 +00:00
|
|
|
osg::Group* new_tile = new osg::Group;
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
if (found_tile_base) {
|
|
|
|
// load tile if found ...
|
2007-05-08 06:12:26 +00:00
|
|
|
obj_load( object_base.str(), new_tile, true );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
// ... or generate an ocean tile on the fly
|
2005-12-03 10:20:35 +00:00
|
|
|
SG_LOG(SG_TERRAIN, SG_INFO, " Generating ocean tile");
|
2007-05-08 06:12:26 +00:00
|
|
|
if ( !SGGenTile( path_list[0], tile_bucket,
|
|
|
|
globals->get_matlib(), new_tile ) ) {
|
2002-03-11 22:50:23 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_ALERT,
|
|
|
|
"Warning: failed to generate ocean tile!" );
|
|
|
|
}
|
2001-06-22 20:35:39 +00:00
|
|
|
}
|
|
|
|
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
// now that we have a valid center, process all the objects
|
2005-12-03 10:20:35 +00:00
|
|
|
for (unsigned int j = 0; j < objects.size(); j++) {
|
2005-12-02 21:46:33 +00:00
|
|
|
const Object *obj = objects[j];
|
|
|
|
|
|
|
|
if (obj->type == OBJECT) {
|
|
|
|
SGPath custom_path = obj->path;
|
|
|
|
custom_path.append( obj->name );
|
2007-05-08 06:12:26 +00:00
|
|
|
obj_load( custom_path.str(), new_tile, false );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
} else if (obj->type == OBJECT_SHARED || obj->type == OBJECT_STATIC) {
|
|
|
|
// object loading is deferred to main render thread,
|
|
|
|
// but lets figure out the paths right now.
|
|
|
|
SGPath custom_path;
|
|
|
|
if ( obj->type == OBJECT_STATIC ) {
|
|
|
|
custom_path = obj->path;
|
|
|
|
} else {
|
|
|
|
custom_path = globals->get_fg_root();
|
|
|
|
}
|
|
|
|
custom_path.append( obj->name );
|
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
osg::Matrix obj_pos;
|
2007-05-08 06:12:26 +00:00
|
|
|
WorldCoordinate( obj_pos, obj->lat, obj->lon, obj->elev, obj->hdg );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
osg::MatrixTransform *obj_trans = new osg::MatrixTransform;
|
|
|
|
obj_trans->setMatrix( obj_pos );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
// wire as much of the scene graph together as we can
|
2006-10-29 19:30:21 +00:00
|
|
|
new_tile->addChild( obj_trans );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
2007-11-29 23:59:14 +00:00
|
|
|
osg::Node* model
|
|
|
|
= FGTileMgr::loadTileModel(custom_path.str(),
|
|
|
|
obj->type == OBJECT_SHARED);
|
|
|
|
if (model)
|
|
|
|
obj_trans->addChild(model);
|
2006-04-14 14:50:11 +00:00
|
|
|
} else if (obj->type == OBJECT_SIGN || obj->type == OBJECT_RUNWAY_SIGN) {
|
2005-12-02 21:46:33 +00:00
|
|
|
// load the object itself
|
|
|
|
SGPath custom_path = obj->path;
|
|
|
|
custom_path.append( obj->name );
|
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
osg::Matrix obj_pos;
|
2007-05-08 06:12:26 +00:00
|
|
|
WorldCoordinate( obj_pos, obj->lat, obj->lon, obj->elev, obj->hdg );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
osg::MatrixTransform *obj_trans = new osg::MatrixTransform;
|
|
|
|
obj_trans->setMatrix( obj_pos );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
2006-10-29 19:30:21 +00:00
|
|
|
osg::Node *custom_obj = 0;
|
|
|
|
if (obj->type == OBJECT_SIGN)
|
|
|
|
custom_obj = SGMakeSign(globals->get_matlib(), custom_path.str(), obj->name);
|
|
|
|
else
|
|
|
|
custom_obj = SGMakeRunwaySign(globals->get_matlib(), custom_path.str(), obj->name);
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
// wire the pieces together
|
|
|
|
if ( custom_obj != NULL ) {
|
2006-10-29 19:30:21 +00:00
|
|
|
obj_trans -> addChild( custom_obj );
|
2005-12-02 21:46:33 +00:00
|
|
|
}
|
2006-10-29 19:30:21 +00:00
|
|
|
new_tile->addChild( obj_trans );
|
2005-12-02 21:46:33 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
delete obj;
|
|
|
|
}
|
2007-11-29 23:59:14 +00:00
|
|
|
return new_tile;
|
2004-10-10 17:41:11 +00:00
|
|
|
}
|
2001-05-16 06:00:02 +00:00
|
|
|
|
2001-04-16 20:03:52 +00:00
|
|
|
void
|
2007-05-08 06:12:26 +00:00
|
|
|
FGTileEntry::add_ssg_nodes( osg::Group *terrain_branch )
|
2001-04-16 20:03:52 +00:00
|
|
|
{
|
2001-05-19 16:59:43 +00:00
|
|
|
// bump up the ref count so we can remove this later without
|
|
|
|
// having ssg try to free the memory.
|
2007-12-14 22:51:56 +00:00
|
|
|
terrain_branch->addChild( _node.get() );
|
2001-05-19 16:59:43 +00:00
|
|
|
|
2001-05-30 18:21:03 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG,
|
2007-12-14 22:51:56 +00:00
|
|
|
"connected a tile into scene graph. _node = "
|
|
|
|
<< _node.get() );
|
2001-05-30 18:21:03 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "num parents now = "
|
2007-12-14 22:51:56 +00:00
|
|
|
<< _node->getNumParents() );
|
2001-03-29 01:42:31 +00:00
|
|
|
}
|
2001-05-19 16:59:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
FGTileEntry::disconnect_ssg_nodes()
|
|
|
|
{
|
2002-08-07 02:53:01 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "disconnecting ssg nodes" );
|
2001-05-19 16:59:43 +00:00
|
|
|
|
2007-12-14 22:51:56 +00:00
|
|
|
if (! is_loaded()) {
|
2002-08-07 02:53:01 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "removing a not-fully loaded tile!" );
|
2001-05-30 18:21:03 +00:00
|
|
|
} else {
|
2007-12-14 22:51:56 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_DEBUG, "removing a fully loaded tile! _node = " << _node.get() );
|
2001-05-30 18:21:03 +00:00
|
|
|
}
|
|
|
|
|
2001-05-19 16:59:43 +00:00
|
|
|
// find the terrain branch parent
|
2007-12-14 22:51:56 +00:00
|
|
|
int pcount = _node->getNumParents();
|
2001-05-19 16:59:43 +00:00
|
|
|
if ( pcount > 0 ) {
|
2002-09-23 15:27:46 +00:00
|
|
|
// find the first parent (should only be one)
|
2007-12-14 22:51:56 +00:00
|
|
|
osg::Group *parent = _node->getParent( 0 ) ;
|
2002-09-23 15:27:46 +00:00
|
|
|
if( parent ) {
|
|
|
|
// disconnect the tile (we previously ref()'d it so it
|
|
|
|
// won't get freed now)
|
2007-12-14 22:51:56 +00:00
|
|
|
parent->removeChild( _node.get() );
|
2002-09-23 15:27:46 +00:00
|
|
|
} else {
|
2007-12-14 22:51:56 +00:00
|
|
|
// This should be impossible.
|
2002-09-23 15:27:46 +00:00
|
|
|
SG_LOG( SG_TERRAIN, SG_ALERT,
|
|
|
|
"parent pointer is NULL! Dying" );
|
|
|
|
exit(-1);
|
|
|
|
}
|
2001-05-19 16:59:43 +00:00
|
|
|
}
|
|
|
|
}
|
2007-11-29 23:59:47 +00:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
class ReaderWriterSTG : public osgDB::ReaderWriter {
|
|
|
|
public:
|
|
|
|
virtual const char* className() const;
|
|
|
|
|
|
|
|
virtual bool acceptsExtension(const string& extension) const;
|
|
|
|
|
|
|
|
virtual ReadResult readNode(const string& fileName,
|
|
|
|
const osgDB::ReaderWriter::Options* options)
|
|
|
|
const;
|
|
|
|
};
|
|
|
|
|
|
|
|
const char* ReaderWriterSTG::className() const
|
|
|
|
{
|
|
|
|
return "STG Database reader";
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ReaderWriterSTG::acceptsExtension(const string& extension) const
|
|
|
|
{
|
|
|
|
return (osgDB::equalCaseInsensitive(extension, "gz")
|
|
|
|
|| osgDB::equalCaseInsensitive(extension, "stg"));
|
|
|
|
}
|
|
|
|
|
2007-12-20 23:20:51 +00:00
|
|
|
//#define SLOW_PAGER 1
|
|
|
|
#ifdef SLOW_PAGER
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2007-11-29 23:59:47 +00:00
|
|
|
osgDB::ReaderWriter::ReadResult
|
|
|
|
ReaderWriterSTG::readNode(const string& fileName,
|
|
|
|
const osgDB::ReaderWriter::Options* options) const
|
|
|
|
{
|
|
|
|
string ext = osgDB::getLowerCaseFileExtension(fileName);
|
|
|
|
if(!acceptsExtension(ext))
|
|
|
|
return ReadResult::FILE_NOT_HANDLED;
|
|
|
|
string stgFileName;
|
|
|
|
if (osgDB::equalCaseInsensitive(ext, "gz")) {
|
|
|
|
stgFileName = osgDB::getNameLessExtension(fileName);
|
|
|
|
if (!acceptsExtension(
|
|
|
|
osgDB::getLowerCaseFileExtension(stgFileName))) {
|
|
|
|
return ReadResult::FILE_NOT_HANDLED;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
stgFileName = fileName;
|
|
|
|
}
|
|
|
|
osg::Node* result
|
|
|
|
= FGTileEntry::loadTileByName(osgDB::getNameLessExtension(stgFileName),
|
|
|
|
globals->get_fg_scenery());
|
2007-12-20 23:20:51 +00:00
|
|
|
// For debugging race conditions
|
|
|
|
#ifdef SLOW_PAGER
|
|
|
|
sleep(5);
|
|
|
|
#endif
|
2007-11-29 23:59:47 +00:00
|
|
|
if (result)
|
|
|
|
return result; // Constructor converts to ReadResult
|
|
|
|
else
|
|
|
|
return ReadResult::FILE_NOT_HANDLED;
|
|
|
|
}
|
|
|
|
|
|
|
|
osgDB::RegisterReaderWriterProxy<ReaderWriterSTG> g_readerWriterSTGProxy;
|
|
|
|
ModelRegistryCallbackProxy<LoadOnlyCallback> g_stgCallbackProxy("stg");
|
|
|
|
} // namespace
|