2001-07-02 22:14:09 +00:00
|
|
|
// fg_init.cxx -- Flight Gear top level initialization routines
|
1998-04-25 15:11:10 +00:00
|
|
|
//
|
|
|
|
// Written by Curtis Olson, started August 1997.
|
|
|
|
//
|
2004-11-19 22:10:41 +00:00
|
|
|
// Copyright (C) 1997 Curtis L. Olson - http://www.flightgear.org/~curt
|
1998-04-25 15:11:10 +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.
|
1998-04-25 15:11:10 +00:00
|
|
|
//
|
|
|
|
// $Id$
|
1997-08-23 01:46:20 +00:00
|
|
|
|
|
|
|
|
1998-04-24 00:49:17 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
1998-04-22 13:25:39 +00:00
|
|
|
|
1999-05-08 02:33:13 +00:00
|
|
|
// For BC 5.01 this must be included before OpenGL includes.
|
2001-03-23 22:42:49 +00:00
|
|
|
#ifdef SG_MATH_EXCEPTION_CLASH
|
1999-05-08 02:33:13 +00:00
|
|
|
# include <math.h>
|
|
|
|
#endif
|
|
|
|
|
1997-08-23 01:46:20 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2002-10-04 13:20:53 +00:00
|
|
|
#include <string.h> // strcmp()
|
1998-07-22 21:40:43 +00:00
|
|
|
|
2001-03-22 04:02:11 +00:00
|
|
|
#if defined( unix ) || defined( __CYGWIN__ )
|
2002-10-04 13:20:53 +00:00
|
|
|
# include <unistd.h> // for gethostname()
|
2001-03-22 04:02:11 +00:00
|
|
|
#endif
|
|
|
|
|
1998-07-22 21:40:43 +00:00
|
|
|
// work around a stdc++ lib bug in some versions of linux, but doesn't
|
|
|
|
// seem to hurt to have this here for all versions of Linux.
|
|
|
|
#ifdef linux
|
|
|
|
# define _G_NO_EXTERN_TEMPLATES
|
|
|
|
#endif
|
|
|
|
|
2000-02-15 03:30:01 +00:00
|
|
|
#include <simgear/compiler.h>
|
1999-02-26 22:08:34 +00:00
|
|
|
|
|
|
|
#include STL_STRING
|
1997-08-23 01:46:20 +00:00
|
|
|
|
2000-02-15 03:30:01 +00:00
|
|
|
#include <simgear/constants.h>
|
2000-02-16 23:01:03 +00:00
|
|
|
#include <simgear/debug/logstream.hxx>
|
2003-09-24 17:20:55 +00:00
|
|
|
#include <simgear/structure/exception.hxx>
|
|
|
|
#include <simgear/structure/event_mgr.hxx>
|
2000-02-16 23:01:03 +00:00
|
|
|
#include <simgear/math/point3d.hxx>
|
|
|
|
#include <simgear/math/polar3d.hxx>
|
2000-09-27 20:16:22 +00:00
|
|
|
#include <simgear/math/sg_geodesy.hxx>
|
2001-03-25 14:20:12 +00:00
|
|
|
#include <simgear/misc/sg_path.hxx>
|
2003-12-05 01:52:34 +00:00
|
|
|
#include <simgear/misc/interpolator.hxx>
|
2003-05-12 21:34:29 +00:00
|
|
|
#include <simgear/scene/material/matlib.hxx>
|
2000-07-06 22:13:24 +00:00
|
|
|
#include <simgear/timing/sg_time.hxx>
|
2002-02-06 23:31:33 +00:00
|
|
|
#include <simgear/timing/lowleveltime.h>
|
2000-02-15 03:30:01 +00:00
|
|
|
|
1998-10-17 01:33:52 +00:00
|
|
|
#include <Aircraft/aircraft.hxx>
|
2005-11-01 13:41:49 +00:00
|
|
|
#include <Aircraft/replay.hxx>
|
2004-12-22 23:57:07 +00:00
|
|
|
#include <Airports/apt_loader.hxx>
|
2000-08-16 00:09:03 +00:00
|
|
|
#include <Airports/runways.hxx>
|
1998-08-25 16:59:08 +00:00
|
|
|
#include <Airports/simple.hxx>
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
#include <AIModel/AIManager.hxx>
|
2001-11-07 17:55:28 +00:00
|
|
|
#include <ATC/ATCdisplay.hxx>
|
2002-03-01 17:39:52 +00:00
|
|
|
#include <ATC/ATCmgr.hxx>
|
2002-04-03 23:54:44 +00:00
|
|
|
#include <ATC/AIMgr.hxx>
|
2000-04-30 06:51:49 +00:00
|
|
|
#include <Autopilot/auto_gui.hxx>
|
2004-01-31 19:47:45 +00:00
|
|
|
#include <Autopilot/route_mgr.hxx>
|
|
|
|
#include <Autopilot/xmlauto.hxx>
|
1998-04-24 00:49:17 +00:00
|
|
|
#include <Cockpit/cockpit.hxx>
|
User-visible
- knobs now continue to rotate when you hold down the mouse
- the middle mouse button makes knobs rotate much faster
- there are NAV1, NAV2, and ADF radios that can be tuned using the mouse
- there are standby frequencies for NAV1 and NAV2, and buttons to swap
- there is a crude, rather silly-looking DME, hard-wired to NAV1
- there is a crude, rather silly-looking autopilot that can lock
the heading (to the bug on the gyro), can lock to NAV1, and can lock
the current altitude
- the knobs for changing the radials on NAV1 and NAV2 look much better
and are in the right place
- tuning into an ILS frequency doesn't change the displayed radial for
NAV1
Code
- I've created a new module, sp_panel.[ch]xx, that constructs the
default single-prop panel; this works entirely outside of FGPanel,
so it is possible to construct similar modules for other sorts of
panels; all code specific to the default panel has been removed from
panel.cxx
- current_panel is now a pointer
- radiostack.[ch]xx keeps track both of the actual radial and of the
selected radial (they will differ with ILS); the NAV gauges should
not spin around automatically to show the actual radial (we need to
do something similar with the autopilot)
- the panel is initialized fairly early
- make sure that standby frequencies also get initialized
- I've started combining and clipping small textures to save texture
memory; there's a lot more to do, but at least I've made a start
2000-05-02 18:26:00 +00:00
|
|
|
#include <Cockpit/panel.hxx>
|
2000-09-22 17:20:56 +00:00
|
|
|
#include <Cockpit/panel_io.hxx>
|
2004-10-19 11:10:20 +00:00
|
|
|
#ifdef ENABLE_SP_FMDS
|
|
|
|
#include <FDM/SP/ADA.hxx>
|
|
|
|
#include <FDM/SP/ACMS.hxx>
|
|
|
|
#endif
|
1999-10-15 22:21:09 +00:00
|
|
|
#include <FDM/Balloon.h>
|
2002-09-28 21:42:03 +00:00
|
|
|
#include <FDM/ExternalNet/ExternalNet.hxx>
|
2003-03-03 04:34:27 +00:00
|
|
|
#include <FDM/ExternalPipe/ExternalPipe.hxx>
|
2002-02-25 03:22:03 +00:00
|
|
|
#include <FDM/JSBSim/JSBSim.hxx>
|
2003-05-20 02:08:31 +00:00
|
|
|
#include <FDM/LaRCsim/LaRCsim.hxx>
|
1999-10-14 01:53:43 +00:00
|
|
|
#include <FDM/MagicCarpet.hxx>
|
2002-04-25 16:23:16 +00:00
|
|
|
#include <FDM/UFO.hxx>
|
2001-11-12 04:49:06 +00:00
|
|
|
#include <FDM/NullFDM.hxx>
|
2001-12-01 06:22:49 +00:00
|
|
|
#include <FDM/YASim/YASim.hxx>
|
2002-11-07 16:27:47 +00:00
|
|
|
#include <GUI/new_gui.hxx>
|
1999-03-11 23:09:26 +00:00
|
|
|
#include <Include/general.hxx>
|
I'm attaching diffs to add a new FGInput module to FlightGear
(src/Input). So far, FGInput replaces most of src/Main/keyboard.cxx
(I've left a tiny stub); in the very near future, it will also take
over control of the joystick, mouse (Norm permitting), and panel
instrument interactions, so that there is a single mechanism for
configuring all input devices.
The new format should be (close to) self-explanatory by looking at the
new base-package file keyboard.xml, which is now included by
preferences.xml (I'll do the same thing for the joystick when I have a
chance). I have not managed to move all keybindings into this file
yet, but I've made a good start. I'm including Tony in the recipient
list so that he can see how bindings can use an external XML file.
This patch also adds support for multiple bindings for a single key,
special keys (i.e. keypad and function keys), and key modifiers
(shift/alt/ctrl); special keys use the PUI convention of adding 256 to
the Glut key code.
Unfortunately, everything comes with a price; in this case, I have not
yet found a general mechanism for the old (hard-coded) modal bindings,
which behaved differently depending on the autopilot state (i.e. left
rudder or move AP heading left); with my patches, this functionality
disappears, but you can still adjust the autopilot using the panel or
the GUI input dialogs.
2001-05-23 23:01:15 +00:00
|
|
|
#include <Input/input.hxx>
|
2002-09-24 14:51:37 +00:00
|
|
|
#include <Instrumentation/instrument_mgr.hxx>
|
2002-04-09 18:58:24 +00:00
|
|
|
#include <Model/acmodel.hxx>
|
2004-10-22 09:58:24 +00:00
|
|
|
#include <AIModel/submodel.hxx>
|
2003-11-28 15:48:05 +00:00
|
|
|
#include <AIModel/AIManager.hxx>
|
2004-05-28 05:24:54 +00:00
|
|
|
#include <Navaids/navdb.hxx>
|
2000-04-24 23:51:26 +00:00
|
|
|
#include <Navaids/navlist.hxx>
|
1998-04-30 12:34:17 +00:00
|
|
|
#include <Scenery/scenery.hxx>
|
1998-04-22 13:25:39 +00:00
|
|
|
#include <Scenery/tilemgr.hxx>
|
2003-11-25 21:08:36 +00:00
|
|
|
#include <Scripting/NasalSys.hxx>
|
2001-10-29 04:39:05 +00:00
|
|
|
#include <Sound/fg_fx.hxx>
|
2004-12-03 21:21:16 +00:00
|
|
|
#include <Sound/beacon.hxx>
|
|
|
|
#include <Sound/morse.hxx>
|
2006-02-11 12:02:40 +00:00
|
|
|
#include <Sound/voice.hxx>
|
2002-09-23 19:55:10 +00:00
|
|
|
#include <Systems/system_mgr.hxx>
|
1998-04-22 13:25:39 +00:00
|
|
|
#include <Time/light.hxx>
|
2003-09-15 22:55:39 +00:00
|
|
|
#include <Time/sunsolver.hxx>
|
2000-07-05 02:39:30 +00:00
|
|
|
#include <Time/tmp.hxx>
|
2004-06-03 17:59:14 +00:00
|
|
|
#include <Traffic/TrafficMgr.hxx>
|
2005-11-01 13:41:49 +00:00
|
|
|
#include <MultiPlayer/multiplaymgr.hxx>
|
2003-03-19 20:45:09 +00:00
|
|
|
|
2004-03-16 20:19:07 +00:00
|
|
|
#include <Environment/environment_mgr.hxx>
|
1997-08-23 01:46:20 +00:00
|
|
|
|
1998-04-22 13:25:39 +00:00
|
|
|
#include "fg_init.hxx"
|
1999-11-19 02:10:24 +00:00
|
|
|
#include "fg_io.hxx"
|
2001-06-01 17:52:28 +00:00
|
|
|
#include "fg_commands.hxx"
|
2001-06-05 22:12:08 +00:00
|
|
|
#include "fg_props.hxx"
|
2001-01-13 22:06:39 +00:00
|
|
|
#include "options.hxx"
|
2000-07-08 06:29:19 +00:00
|
|
|
#include "globals.hxx"
|
2002-03-12 16:29:00 +00:00
|
|
|
#include "logger.hxx"
|
2001-07-22 19:51:16 +00:00
|
|
|
#include "viewmgr.hxx"
|
2005-12-21 13:36:04 +00:00
|
|
|
#include "main.hxx"
|
1998-04-22 13:25:39 +00:00
|
|
|
|
1999-03-11 23:09:26 +00:00
|
|
|
#if defined(FX) && defined(XMESA)
|
|
|
|
#include <GL/xmesa.h>
|
|
|
|
#endif
|
|
|
|
|
2001-03-23 22:59:18 +00:00
|
|
|
SG_USING_STD(string);
|
1998-06-27 16:54:32 +00:00
|
|
|
|
2003-05-09 18:44:33 +00:00
|
|
|
|
|
|
|
class Sound;
|
1998-02-12 21:58:27 +00:00
|
|
|
extern const char *default_root;
|
2004-12-16 13:19:01 +00:00
|
|
|
float init_volume;
|
2004-04-18 18:01:10 +00:00
|
|
|
|
1997-08-25 20:27:21 +00:00
|
|
|
|
2002-11-16 20:41:00 +00:00
|
|
|
// Scan the command line options for the specified option and return
|
|
|
|
// the value.
|
|
|
|
static string fgScanForOption( const string& option, int argc, char **argv ) {
|
2002-11-16 20:17:11 +00:00
|
|
|
int i = 1;
|
|
|
|
|
2005-12-27 14:02:38 +00:00
|
|
|
if (hostname == NULL)
|
|
|
|
{
|
|
|
|
char _hostname[256];
|
|
|
|
gethostname(_hostname, 256);
|
|
|
|
hostname = strdup(_hostname);
|
|
|
|
free_hostname = true;
|
|
|
|
}
|
|
|
|
|
2002-11-16 20:41:00 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Scanning command line for: " << option );
|
|
|
|
|
|
|
|
int len = option.length();
|
2002-11-16 20:17:11 +00:00
|
|
|
|
|
|
|
while ( i < argc ) {
|
|
|
|
SG_LOG( SG_GENERAL, SG_DEBUG, "argv[" << i << "] = " << argv[i] );
|
|
|
|
|
|
|
|
string arg = argv[i];
|
2002-11-16 20:41:00 +00:00
|
|
|
if ( arg.find( option ) == 0 ) {
|
|
|
|
return arg.substr( len );
|
2002-11-16 20:17:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-16 20:41:00 +00:00
|
|
|
// Scan the user config files for the specified option and return
|
|
|
|
// the value.
|
|
|
|
static string fgScanForOption( const string& option, const string& path ) {
|
2002-11-16 20:17:11 +00:00
|
|
|
sg_gzifstream in( path );
|
2002-11-16 20:41:00 +00:00
|
|
|
if ( !in.is_open() ) {
|
|
|
|
return "";
|
|
|
|
}
|
2002-11-16 20:17:11 +00:00
|
|
|
|
2002-11-16 20:41:00 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Scanning " << path << " for: " << option );
|
|
|
|
|
|
|
|
int len = option.length();
|
2002-11-16 20:17:11 +00:00
|
|
|
|
|
|
|
in >> skipcomment;
|
|
|
|
#ifndef __MWERKS__
|
|
|
|
while ( ! in.eof() ) {
|
|
|
|
#else
|
|
|
|
char c = '\0';
|
|
|
|
while ( in.get(c) && c != '\0' ) {
|
|
|
|
in.putback(c);
|
|
|
|
#endif
|
|
|
|
string line;
|
|
|
|
|
|
|
|
#if defined( macintosh )
|
|
|
|
getline( in, line, '\r' );
|
|
|
|
#else
|
|
|
|
getline( in, line, '\n' );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// catch extraneous (DOS) line ending character
|
|
|
|
if ( line[line.length() - 1] < 32 ) {
|
|
|
|
line = line.substr( 0, line.length()-1 );
|
|
|
|
}
|
|
|
|
|
2002-11-16 20:41:00 +00:00
|
|
|
if ( line.find( option ) == 0 ) {
|
|
|
|
return line.substr( len );
|
2002-11-16 20:17:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
in >> skipcomment;
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2005-12-22 10:25:07 +00:00
|
|
|
// Scan the user config files for the specified option and return
|
|
|
|
// the value.
|
|
|
|
static string fgScanForOption( const string& option ) {
|
|
|
|
string arg("");
|
2001-01-13 22:06:39 +00:00
|
|
|
|
2001-03-22 04:02:11 +00:00
|
|
|
#if defined( unix ) || defined( __CYGWIN__ )
|
|
|
|
// Next check home directory for .fgfsrc.hostname file
|
2005-12-22 10:25:07 +00:00
|
|
|
if ( arg.empty() ) {
|
2005-12-21 13:36:04 +00:00
|
|
|
if ( homedir != NULL ) {
|
|
|
|
SGPath config( homedir );
|
2002-10-04 13:20:53 +00:00
|
|
|
config.append( ".fgfsrc" );
|
|
|
|
config.concat( "." );
|
2005-12-21 13:36:04 +00:00
|
|
|
config.concat( hostname );
|
2005-12-22 10:25:07 +00:00
|
|
|
arg = fgScanForOption( option, config.str() );
|
2002-10-04 13:20:53 +00:00
|
|
|
}
|
2001-03-22 04:02:11 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2001-01-13 22:06:39 +00:00
|
|
|
// Next check home directory for .fgfsrc file
|
2005-12-22 10:25:07 +00:00
|
|
|
if ( arg.empty() ) {
|
2005-12-21 13:36:04 +00:00
|
|
|
if ( homedir != NULL ) {
|
|
|
|
SGPath config( homedir );
|
2002-10-04 13:20:53 +00:00
|
|
|
config.append( ".fgfsrc" );
|
2005-12-22 10:25:07 +00:00
|
|
|
arg = fgScanForOption( option, config.str() );
|
2002-10-04 13:20:53 +00:00
|
|
|
}
|
2001-01-13 22:06:39 +00:00
|
|
|
}
|
2005-12-22 10:25:07 +00:00
|
|
|
|
|
|
|
if ( arg.empty() ) {
|
|
|
|
// Check for $fg_root/system.fgfsrc
|
|
|
|
SGPath config( globals->get_fg_root() );
|
|
|
|
config.append( "system.fgfsrc" );
|
|
|
|
arg = fgScanForOption( option, config.str() );
|
|
|
|
}
|
|
|
|
|
|
|
|
return arg;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Read in configuration (files and command line options) but only set
|
|
|
|
// fg_root
|
|
|
|
bool fgInitFGRoot ( int argc, char **argv ) {
|
|
|
|
string root;
|
|
|
|
|
|
|
|
// First parse command line options looking for --fg-root=, this
|
|
|
|
// will override anything specified in a config file
|
|
|
|
root = fgScanForOption( "--fg-root=", argc, argv);
|
|
|
|
|
|
|
|
// Check in one of the user configuration files.
|
2005-12-22 14:14:08 +00:00
|
|
|
if (root.empty() )
|
|
|
|
root = fgScanForOption( "--fg-root=" );
|
2003-03-20 09:38:06 +00:00
|
|
|
|
2001-01-13 22:06:39 +00:00
|
|
|
// Next check if fg-root is set as an env variable
|
2002-03-20 19:16:13 +00:00
|
|
|
if ( root.empty() ) {
|
2005-12-21 13:36:04 +00:00
|
|
|
char *envp = ::getenv( "FG_ROOT" );
|
2002-10-04 13:20:53 +00:00
|
|
|
if ( envp != NULL ) {
|
|
|
|
root = envp;
|
|
|
|
}
|
2001-01-13 22:06:39 +00:00
|
|
|
}
|
2000-10-04 22:52:34 +00:00
|
|
|
|
2001-01-13 22:06:39 +00:00
|
|
|
// Otherwise, default to a random compiled-in location if we can't
|
|
|
|
// find fg-root any other way.
|
2002-03-20 19:16:13 +00:00
|
|
|
if ( root.empty() ) {
|
2001-04-05 20:25:40 +00:00
|
|
|
#if defined( __CYGWIN__ )
|
|
|
|
root = "/FlightGear";
|
|
|
|
#elif defined( WIN32 )
|
2002-10-04 13:20:53 +00:00
|
|
|
root = "\\FlightGear";
|
2003-10-19 19:15:41 +00:00
|
|
|
#elif defined(OSX_BUNDLE)
|
|
|
|
/* the following code looks for the base package directly inside
|
|
|
|
the application bundle. This can be changed fairly easily by
|
|
|
|
fiddling with the code below. And yes, I know it's ugly and verbose.
|
|
|
|
*/
|
|
|
|
CFBundleRef appBundle = CFBundleGetMainBundle();
|
|
|
|
CFURLRef appUrl = CFBundleCopyBundleURL(appBundle);
|
|
|
|
CFRelease(appBundle);
|
|
|
|
|
|
|
|
// look for a 'data' subdir directly inside the bundle : is there
|
|
|
|
// a better place? maybe in Resources? I don't know ...
|
|
|
|
CFURLRef dataDir = CFURLCreateCopyAppendingPathComponent(NULL, appUrl, CFSTR("data"), true);
|
|
|
|
|
|
|
|
// now convert down to a path, and the a c-string
|
|
|
|
CFStringRef path = CFURLCopyFileSystemPath(dataDir, kCFURLPOSIXPathStyle);
|
|
|
|
root = CFStringGetCStringPtr(path, CFStringGetSystemEncoding());
|
|
|
|
|
|
|
|
// tidy up.
|
|
|
|
CFRelease(appBundle);
|
|
|
|
CFRelease(dataDir);
|
|
|
|
CFRelease(path);
|
2001-01-13 22:06:39 +00:00
|
|
|
#else
|
2002-10-04 13:20:53 +00:00
|
|
|
root = PKGLIBDIR;
|
2001-01-13 22:06:39 +00:00
|
|
|
#endif
|
2000-10-04 22:52:34 +00:00
|
|
|
}
|
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "fg_root = " << root );
|
2001-01-13 22:06:39 +00:00
|
|
|
globals->set_fg_root(root);
|
2000-10-04 22:52:34 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-16 20:17:11 +00:00
|
|
|
// Read in configuration (files and command line options) but only set
|
|
|
|
// aircraft
|
|
|
|
bool fgInitFGAircraft ( int argc, char **argv ) {
|
|
|
|
string aircraft;
|
|
|
|
|
|
|
|
// First parse command line options looking for --aircraft=, this
|
|
|
|
// will override anything specified in a config file
|
2002-11-16 20:41:00 +00:00
|
|
|
aircraft = fgScanForOption( "--aircraft=", argc, argv );
|
2002-11-16 20:17:11 +00:00
|
|
|
|
2005-12-22 10:25:07 +00:00
|
|
|
// Check in one of the user configuration files.
|
2005-12-22 14:14:08 +00:00
|
|
|
if ( aircraft.empty() )
|
|
|
|
aircraft = fgScanForOption( "--aircraft=" );
|
2002-11-16 20:17:11 +00:00
|
|
|
|
|
|
|
// if an aircraft was specified, set the property name
|
|
|
|
if ( !aircraft.empty() ) {
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "aircraft = " << aircraft );
|
|
|
|
fgSetString("/sim/aircraft", aircraft.c_str() );
|
|
|
|
} else {
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "No user specified aircraft, using default" );
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-07-16 19:17:58 +00:00
|
|
|
// Return the current base package version
|
|
|
|
string fgBasePackageVersion() {
|
|
|
|
SGPath base_path( globals->get_fg_root() );
|
|
|
|
base_path.append("version");
|
|
|
|
|
|
|
|
sg_gzifstream in( base_path.str() );
|
|
|
|
if ( !in.is_open() ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
SGPath old_path( globals->get_fg_root() );
|
|
|
|
old_path.append( "Thanks" );
|
|
|
|
sg_gzifstream old( old_path.str() );
|
|
|
|
if ( !old.is_open() ) {
|
|
|
|
return "[none]";
|
|
|
|
} else {
|
|
|
|
return "[old version]";
|
|
|
|
}
|
2001-07-16 19:17:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
string version;
|
|
|
|
in >> version;
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-07 15:45:00 +00:00
|
|
|
// Initialize the localization
|
|
|
|
SGPropertyNode *fgInitLocale(const char *language) {
|
|
|
|
SGPropertyNode *c_node = NULL, *d_node = NULL;
|
|
|
|
SGPropertyNode *intl = fgGetNode("/sim/intl");
|
|
|
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Selecting language: " << language );
|
|
|
|
|
|
|
|
// localization not defined
|
|
|
|
if (!intl)
|
|
|
|
return NULL;
|
2002-10-10 15:02:50 +00:00
|
|
|
|
2002-10-07 15:45:00 +00:00
|
|
|
//
|
|
|
|
// Select the proper language from the list
|
|
|
|
//
|
|
|
|
vector<SGPropertyNode_ptr> locale = intl->getChildren("locale");
|
|
|
|
for (unsigned int i = 0; i < locale.size(); i++) {
|
|
|
|
|
|
|
|
vector<SGPropertyNode_ptr> lang = locale[i]->getChildren("lang");
|
|
|
|
for (unsigned int j = 0; j < lang.size(); j++) {
|
|
|
|
|
|
|
|
if (!strcmp(lang[j]->getStringValue(), language)) {
|
|
|
|
c_node = locale[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Get the defaults
|
|
|
|
d_node = intl->getChild("locale");
|
|
|
|
if (!c_node)
|
|
|
|
c_node = d_node;
|
|
|
|
|
|
|
|
// Check for localized font
|
|
|
|
SGPropertyNode *font_n = c_node->getNode("font", true);
|
|
|
|
if ( !strcmp(font_n->getStringValue(), "") )
|
|
|
|
font_n->setStringValue(d_node->getStringValue("font", "typewriter.txf"));
|
|
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
// Load the default strings
|
|
|
|
//
|
|
|
|
SGPath d_path( globals->get_fg_root() );
|
|
|
|
|
|
|
|
const char *d_path_str = d_node->getStringValue("strings");
|
|
|
|
if (!d_path_str) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Incorrect path in configuration file.");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
d_path.append(d_path_str);
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Reading localized strings from "
|
|
|
|
<< d_path.str());
|
|
|
|
|
|
|
|
SGPropertyNode *strings = c_node->getNode("strings");
|
|
|
|
try {
|
|
|
|
readProperties(d_path.str(), strings);
|
|
|
|
} catch (const sg_exception &e) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Unable to read the localized strings");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Load the language specific strings
|
|
|
|
//
|
|
|
|
if (c_node != d_node) {
|
|
|
|
SGPath c_path( globals->get_fg_root() );
|
|
|
|
|
|
|
|
const char *c_path_str = c_node->getStringValue("strings");
|
|
|
|
if (!c_path_str) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Incorrect path in configuration file.");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
c_path.append(c_path_str);
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Reading localized strings from "
|
|
|
|
<< c_path.str());
|
|
|
|
|
|
|
|
try {
|
|
|
|
readProperties(c_path.str(), strings);
|
|
|
|
} catch (const sg_exception &e) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "Unable to read the localized strings");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return c_node;
|
|
|
|
}
|
|
|
|
|
1999-06-18 03:42:54 +00:00
|
|
|
|
2002-10-10 15:02:50 +00:00
|
|
|
|
|
|
|
// Initialize the localization routines
|
|
|
|
bool fgDetectLanguage() {
|
|
|
|
char *language = ::getenv("LANG");
|
|
|
|
|
|
|
|
if (language == NULL) {
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Unable to detect the language" );
|
|
|
|
language = "C";
|
|
|
|
}
|
|
|
|
|
|
|
|
SGPropertyNode *locale = fgInitLocale(language);
|
|
|
|
if (!locale) {
|
2003-03-20 10:43:02 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT,
|
|
|
|
"No internationalization settings specified in preferences.xml" );
|
2002-10-10 15:02:50 +00:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
globals->set_locale( locale );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2002-10-10 18:39:52 +00:00
|
|
|
// Attempt to locate and parse the various non-XML config files in order
|
|
|
|
// from least precidence to greatest precidence
|
|
|
|
static void
|
|
|
|
do_options (int argc, char ** argv)
|
|
|
|
{
|
|
|
|
// Check for $fg_root/system.fgfsrc
|
|
|
|
SGPath config( globals->get_fg_root() );
|
|
|
|
config.append( "system.fgfsrc" );
|
|
|
|
fgParseOptions(config.str());
|
|
|
|
|
|
|
|
#if defined( unix ) || defined( __CYGWIN__ )
|
|
|
|
config.concat( "." );
|
2005-12-21 13:36:04 +00:00
|
|
|
config.concat( hostname );
|
2002-10-10 18:39:52 +00:00
|
|
|
fgParseOptions(config.str());
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Check for ~/.fgfsrc
|
2005-12-21 13:36:04 +00:00
|
|
|
if ( homedir != NULL ) {
|
|
|
|
config.set( homedir );
|
2002-10-10 18:39:52 +00:00
|
|
|
config.append( ".fgfsrc" );
|
|
|
|
fgParseOptions(config.str());
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined( unix ) || defined( __CYGWIN__ )
|
|
|
|
// Check for ~/.fgfsrc.hostname
|
|
|
|
config.concat( "." );
|
2005-12-21 13:36:04 +00:00
|
|
|
config.concat( hostname );
|
2002-10-10 18:39:52 +00:00
|
|
|
fgParseOptions(config.str());
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Parse remaining command line options
|
|
|
|
// These will override anything specified in a config file
|
|
|
|
fgParseArgs(argc, argv);
|
|
|
|
}
|
|
|
|
|
2002-10-10 15:02:50 +00:00
|
|
|
|
2003-09-19 20:06:27 +00:00
|
|
|
static string fgFindAircraftPath( const SGPath &path, const string &aircraft ) {
|
|
|
|
ulDirEnt* dire;
|
|
|
|
ulDir *dirp = ulOpenDir(path.str().c_str());
|
|
|
|
if (dirp == NULL) {
|
2005-07-03 10:27:35 +00:00
|
|
|
cerr << "Unable to open aircraft directory '" << path.str() << '\'' << endl;
|
2003-09-19 20:06:27 +00:00
|
|
|
exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
while ((dire = ulReadDir(dirp)) != NULL) {
|
|
|
|
if (dire->d_isdir) {
|
|
|
|
if ( strcmp("CVS", dire->d_name) && strcmp(".", dire->d_name)
|
2006-01-30 13:29:49 +00:00
|
|
|
&& strcmp("..", dire->d_name) && strcmp("AI", dire->d_name))
|
2003-09-19 20:06:27 +00:00
|
|
|
{
|
|
|
|
SGPath next = path;
|
|
|
|
next.append(dire->d_name);
|
|
|
|
|
|
|
|
string result = fgFindAircraftPath( next, aircraft );
|
|
|
|
if ( ! result.empty() ) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if ( !strcmp(dire->d_name, aircraft.c_str()) ) {
|
|
|
|
return path.str();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ulCloseDir(dirp);
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-10-10 15:02:50 +00:00
|
|
|
// Read in configuration (file and command line)
|
|
|
|
bool fgInitConfig ( int argc, char **argv ) {
|
|
|
|
|
2002-11-16 20:17:11 +00:00
|
|
|
// First, set some sane default values
|
2002-10-10 15:02:50 +00:00
|
|
|
fgSetDefaults();
|
|
|
|
|
|
|
|
// Read global preferences from $FG_ROOT/preferences.xml
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Reading global preferences");
|
2002-11-06 18:57:31 +00:00
|
|
|
fgLoadProps("preferences.xml", globals->get_props());
|
2002-10-10 15:02:50 +00:00
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Finished Reading global preferences");
|
|
|
|
|
|
|
|
// Detect the required language as early as possible
|
2002-11-16 20:17:11 +00:00
|
|
|
if ( !fgDetectLanguage() ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan user config files and command line for a specified aircraft.
|
|
|
|
fgInitFGAircraft(argc, argv);
|
2002-10-10 15:02:50 +00:00
|
|
|
|
2003-09-19 20:06:27 +00:00
|
|
|
string aircraft = fgGetString( "/sim/aircraft", "" );
|
2002-11-16 20:17:11 +00:00
|
|
|
if ( aircraft.size() > 0 ) {
|
2003-09-19 20:06:27 +00:00
|
|
|
SGPath aircraft_search( globals->get_fg_root() );
|
|
|
|
aircraft_search.append( "Aircraft" );
|
|
|
|
|
|
|
|
string aircraft_set = aircraft + "-set.xml";
|
|
|
|
|
|
|
|
string result = fgFindAircraftPath( aircraft_search, aircraft_set );
|
|
|
|
if ( !result.empty() ) {
|
2003-09-20 03:33:06 +00:00
|
|
|
fgSetString( "/sim/aircraft-dir", result.c_str() );
|
2003-09-19 20:06:27 +00:00
|
|
|
SGPath full_name( result );
|
|
|
|
full_name.append( aircraft_set );
|
|
|
|
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Reading default aircraft: " << aircraft
|
|
|
|
<< " from " << full_name.str());
|
|
|
|
try {
|
|
|
|
readProperties( full_name.str(), globals->get_props() );
|
|
|
|
} catch ( const sg_exception &e ) {
|
|
|
|
string message = "Error reading default aircraft: ";
|
|
|
|
message += e.getFormattedMessage();
|
|
|
|
SG_LOG(SG_INPUT, SG_ALERT, message);
|
|
|
|
exit(2);
|
2003-09-15 16:00:18 +00:00
|
|
|
}
|
2003-09-19 20:06:27 +00:00
|
|
|
} else {
|
|
|
|
SG_LOG( SG_INPUT, SG_ALERT, "Cannot find specified aircraft: "
|
|
|
|
<< aircraft );
|
2004-01-07 13:03:44 +00:00
|
|
|
return false;
|
2002-11-16 20:17:11 +00:00
|
|
|
}
|
2003-09-19 20:06:27 +00:00
|
|
|
|
2002-10-10 15:02:50 +00:00
|
|
|
} else {
|
2003-09-19 20:06:27 +00:00
|
|
|
SG_LOG( SG_INPUT, SG_ALERT, "No default aircraft specified" );
|
2002-10-10 15:02:50 +00:00
|
|
|
}
|
|
|
|
|
2005-12-18 09:35:01 +00:00
|
|
|
#ifdef _MSC_VER
|
2005-12-21 13:36:04 +00:00
|
|
|
char *envp = ::getenv( "APPDATA" );
|
|
|
|
if (envp != NULL ) {
|
2005-12-22 06:42:00 +00:00
|
|
|
SGPath config( envp );
|
2005-12-18 09:35:01 +00:00
|
|
|
config.append( "flightgear.org" );
|
|
|
|
#else
|
2005-12-21 13:36:04 +00:00
|
|
|
if ( homedir != NULL ) {
|
|
|
|
SGPath config( homedir );
|
2005-12-17 15:34:37 +00:00
|
|
|
config.append( ".fgfs" );
|
2005-12-18 09:35:01 +00:00
|
|
|
#endif
|
2006-02-13 16:49:18 +00:00
|
|
|
fgSetString("/sim/fg-home", config.c_str());
|
2006-02-11 11:51:20 +00:00
|
|
|
config.append( "autosave.xml" );
|
|
|
|
SG_LOG(SG_INPUT, SG_INFO, "Reading user settings from autosave.xml");
|
2005-12-21 13:36:04 +00:00
|
|
|
try {
|
|
|
|
fgLoadProps(config.str().c_str(), globals->get_props(), false,
|
|
|
|
SGPropertyNode::USERARCHIVE);
|
|
|
|
} catch (...) {
|
2006-02-11 11:51:20 +00:00
|
|
|
SG_LOG(SG_INPUT, SG_BULK, "First time reading user settings");
|
2005-12-21 13:36:04 +00:00
|
|
|
}
|
2006-02-11 11:51:20 +00:00
|
|
|
SG_LOG(SG_INPUT, SG_BULK, "Finished Reading user settings");
|
2005-12-17 15:34:37 +00:00
|
|
|
}
|
|
|
|
|
2002-11-16 20:17:11 +00:00
|
|
|
// parse options after loading aircraft to ensure any user
|
|
|
|
// overrides of defaults are honored.
|
2002-10-10 18:39:52 +00:00
|
|
|
do_options(argc, argv);
|
2002-10-10 15:02:50 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1998-07-30 23:48:24 +00:00
|
|
|
|
2002-11-30 20:10:16 +00:00
|
|
|
|
|
|
|
|
2004-04-01 15:27:53 +00:00
|
|
|
#if 0
|
|
|
|
//
|
|
|
|
// This function is never used, but maybe useful in the future ???
|
|
|
|
//
|
|
|
|
|
2002-11-15 21:13:29 +00:00
|
|
|
// Preset lon/lat given an airport id
|
2002-11-16 21:34:51 +00:00
|
|
|
static bool fgSetPosFromAirportID( const string& id ) {
|
2005-12-29 13:58:21 +00:00
|
|
|
FGAirport *a;
|
2000-10-25 15:27:55 +00:00
|
|
|
// double lon, lat;
|
2000-10-12 01:08:09 +00:00
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2002-10-04 13:20:53 +00:00
|
|
|
"Attempting to set starting position from airport code " << id );
|
2000-10-12 01:08:09 +00:00
|
|
|
|
|
|
|
if ( fgFindAirportID( id, &a ) ) {
|
2002-11-15 21:13:29 +00:00
|
|
|
// presets
|
2005-12-29 13:58:21 +00:00
|
|
|
fgSetDouble("/sim/presets/longitude-deg", a->longitude );
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", a->latitude );
|
2002-11-15 21:13:29 +00:00
|
|
|
|
|
|
|
// other code depends on the actual postition being set so set
|
|
|
|
// that as well
|
2005-12-29 13:58:21 +00:00
|
|
|
fgSetDouble("/position/longitude-deg", a->longitude );
|
|
|
|
fgSetDouble("/position/latitude-deg", a->latitude );
|
2002-11-15 21:13:29 +00:00
|
|
|
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2005-12-29 13:58:21 +00:00
|
|
|
"Position for " << id << " is (" << a->longitude
|
|
|
|
<< ", " << a->latitude << ")" );
|
2002-10-04 13:20:53 +00:00
|
|
|
|
|
|
|
return true;
|
2000-10-12 01:08:09 +00:00
|
|
|
} else {
|
2002-10-04 13:20:53 +00:00
|
|
|
return false;
|
2000-10-12 01:08:09 +00:00
|
|
|
}
|
|
|
|
}
|
2004-04-01 15:27:53 +00:00
|
|
|
#endif
|
2000-10-12 01:08:09 +00:00
|
|
|
|
|
|
|
|
2002-04-06 17:23:18 +00:00
|
|
|
// Set current tower position lon/lat given an airport id
|
2002-11-16 21:34:51 +00:00
|
|
|
static bool fgSetTowerPosFromAirportID( const string& id, double hdg ) {
|
2005-12-29 13:58:21 +00:00
|
|
|
|
2002-04-06 17:23:18 +00:00
|
|
|
// tower height hard coded for now...
|
|
|
|
float towerheight=50.0f;
|
|
|
|
|
|
|
|
// make a little off the heading for 1 runway airports...
|
|
|
|
float fudge_lon = fabs(sin(hdg)) * .003f;
|
|
|
|
float fudge_lat = .003f - fudge_lon;
|
|
|
|
|
2005-12-29 13:58:21 +00:00
|
|
|
const FGAirport *a = fgFindAirportID( id);
|
|
|
|
if ( a) {
|
|
|
|
fgSetDouble("/sim/tower/longitude-deg", a->getLongitude() + fudge_lon);
|
|
|
|
fgSetDouble("/sim/tower/latitude-deg", a->getLatitude() + fudge_lat);
|
|
|
|
fgSetDouble("/sim/tower/altitude-ft", a->getElevation() + towerheight);
|
2002-10-04 13:20:53 +00:00
|
|
|
return true;
|
2002-04-06 17:23:18 +00:00
|
|
|
} else {
|
2002-10-04 13:20:53 +00:00
|
|
|
return false;
|
2002-04-06 17:23:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2005-10-19 19:21:45 +00:00
|
|
|
struct FGTowerLocationListener : SGPropertyChangeListener {
|
|
|
|
void valueChanged(SGPropertyNode* node) {
|
|
|
|
const double hdg = fgGetDouble( "/orientation/heading-deg", 0);
|
|
|
|
const string id(node->getStringValue());
|
|
|
|
fgSetTowerPosFromAirportID(id, hdg);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static void fgInitTowerLocationListener() {
|
|
|
|
fgGetNode("/sim/tower/airport-id", true)
|
|
|
|
->addChangeListener( new FGTowerLocationListener() );
|
|
|
|
}
|
2002-04-06 17:23:18 +00:00
|
|
|
|
2000-08-16 00:09:03 +00:00
|
|
|
// Set current_options lon/lat given an airport id and heading (degrees)
|
2002-11-16 21:34:51 +00:00
|
|
|
static bool fgSetPosFromAirportIDandHdg( const string& id, double tgt_hdg ) {
|
2000-08-16 00:09:03 +00:00
|
|
|
FGRunway r;
|
|
|
|
|
|
|
|
if ( id.length() ) {
|
2002-02-05 20:54:08 +00:00
|
|
|
// set initial position from runway and heading
|
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2003-03-20 09:38:06 +00:00
|
|
|
"Attempting to set starting position from airport code "
|
2002-02-05 20:54:08 +00:00
|
|
|
<< id << " heading " << tgt_hdg );
|
2003-03-20 09:38:06 +00:00
|
|
|
|
2003-08-29 04:11:23 +00:00
|
|
|
if ( ! globals->get_runways()->search( id, (int)tgt_hdg, &r ) ) {
|
2002-02-05 20:54:08 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
2003-03-20 09:38:06 +00:00
|
|
|
"Failed to find a good runway for " << id << '\n' );
|
2002-02-05 20:54:08 +00:00
|
|
|
return false;
|
2003-03-20 09:38:06 +00:00
|
|
|
}
|
2000-08-16 00:09:03 +00:00
|
|
|
} else {
|
2002-10-04 13:20:53 +00:00
|
|
|
return false;
|
2000-08-16 00:09:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
double lat2, lon2, az2;
|
2004-12-22 23:57:07 +00:00
|
|
|
double heading = r._heading;
|
2003-03-20 09:38:06 +00:00
|
|
|
double azimuth = heading + 180.0;
|
2000-08-16 00:09:03 +00:00
|
|
|
while ( azimuth >= 360.0 ) { azimuth -= 360.0; }
|
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2004-12-22 23:57:07 +00:00
|
|
|
"runway = " << r._lon << ", " << r._lat
|
|
|
|
<< " length = " << r._length * SG_FEET_TO_METER
|
2002-10-04 13:20:53 +00:00
|
|
|
<< " heading = " << azimuth );
|
2003-03-20 09:38:06 +00:00
|
|
|
|
2004-12-22 23:57:07 +00:00
|
|
|
geo_direct_wgs_84 ( 0, r._lat, r._lon, azimuth,
|
|
|
|
r._length * SG_FEET_TO_METER * 0.5 - 5.0,
|
2002-10-04 13:20:53 +00:00
|
|
|
&lat2, &lon2, &az2 );
|
2001-03-02 23:27:22 +00:00
|
|
|
|
2002-11-15 21:13:29 +00:00
|
|
|
if ( fabs( fgGetDouble("/sim/presets/offset-distance") ) > SG_EPSILON ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
double olat, olon;
|
2002-11-15 21:13:29 +00:00
|
|
|
double odist = fgGetDouble("/sim/presets/offset-distance");
|
2002-10-04 13:20:53 +00:00
|
|
|
odist *= SG_NM_TO_METER;
|
|
|
|
double oaz = azimuth;
|
2002-11-15 21:13:29 +00:00
|
|
|
if ( fabs(fgGetDouble("/sim/presets/offset-azimuth")) > SG_EPSILON ) {
|
|
|
|
oaz = fgGetDouble("/sim/presets/offset-azimuth") + 180;
|
2003-07-16 19:12:02 +00:00
|
|
|
heading = tgt_hdg;
|
2002-10-04 13:20:53 +00:00
|
|
|
}
|
|
|
|
while ( oaz >= 360.0 ) { oaz -= 360.0; }
|
|
|
|
geo_direct_wgs_84 ( 0, lat2, lon2, oaz, odist, &olat, &olon, &az2 );
|
|
|
|
lat2=olat;
|
|
|
|
lon2=olon;
|
2001-03-02 23:27:22 +00:00
|
|
|
}
|
2002-11-15 21:13:29 +00:00
|
|
|
|
|
|
|
// presets
|
|
|
|
fgSetDouble("/sim/presets/longitude-deg", lon2 );
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", lat2 );
|
|
|
|
fgSetDouble("/sim/presets/heading-deg", heading );
|
|
|
|
|
|
|
|
// other code depends on the actual values being set ...
|
2001-07-02 22:27:24 +00:00
|
|
|
fgSetDouble("/position/longitude-deg", lon2 );
|
|
|
|
fgSetDouble("/position/latitude-deg", lat2 );
|
|
|
|
fgSetDouble("/orientation/heading-deg", heading );
|
2000-08-16 00:09:03 +00:00
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2002-10-04 13:20:53 +00:00
|
|
|
"Position for " << id << " is ("
|
|
|
|
<< lon2 << ", "
|
|
|
|
<< lat2 << ") new heading is "
|
|
|
|
<< heading );
|
2003-03-20 09:38:06 +00:00
|
|
|
|
2000-08-16 00:09:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2002-05-14 05:22:52 +00:00
|
|
|
|
2003-03-20 09:38:06 +00:00
|
|
|
// Set current_options lon/lat given an airport id and runway number
|
2002-11-17 04:04:21 +00:00
|
|
|
static bool fgSetPosFromAirportIDandRwy( const string& id, const string& rwy ) {
|
|
|
|
FGRunway r;
|
|
|
|
|
|
|
|
if ( id.length() ) {
|
2003-03-20 09:38:06 +00:00
|
|
|
// set initial position from airport and runway number
|
2002-11-17 04:04:21 +00:00
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
|
|
|
"Attempting to set starting position for "
|
2003-03-20 22:31:25 +00:00
|
|
|
<< id << ":" << rwy );
|
2002-11-17 04:04:21 +00:00
|
|
|
|
2003-08-29 04:11:23 +00:00
|
|
|
if ( ! globals->get_runways()->search( id, rwy, &r ) ) {
|
2002-11-17 04:04:21 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
2003-03-20 09:38:06 +00:00
|
|
|
"Failed to find runway " << rwy <<
|
|
|
|
" at airport " << id );
|
2002-11-17 04:04:21 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2003-03-20 09:38:06 +00:00
|
|
|
double lat2, lon2, az2;
|
2004-12-22 23:57:07 +00:00
|
|
|
double heading = r._heading;
|
2003-03-20 09:38:06 +00:00
|
|
|
double azimuth = heading + 180.0;
|
|
|
|
while ( azimuth >= 360.0 ) { azimuth -= 360.0; }
|
2002-11-17 04:04:21 +00:00
|
|
|
|
2003-03-20 09:38:06 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2004-12-22 23:57:07 +00:00
|
|
|
"runway = " << r._lon << ", " << r._lat
|
|
|
|
<< " length = " << r._length * SG_FEET_TO_METER
|
2003-03-20 09:38:06 +00:00
|
|
|
<< " heading = " << azimuth );
|
|
|
|
|
2004-12-22 23:57:07 +00:00
|
|
|
geo_direct_wgs_84 ( 0, r._lat, r._lon,
|
2003-03-20 09:38:06 +00:00
|
|
|
azimuth,
|
2004-12-22 23:57:07 +00:00
|
|
|
r._length * SG_FEET_TO_METER * 0.5 - 5.0,
|
2003-03-20 09:38:06 +00:00
|
|
|
&lat2, &lon2, &az2 );
|
|
|
|
|
|
|
|
if ( fabs( fgGetDouble("/sim/presets/offset-distance") ) > SG_EPSILON )
|
|
|
|
{
|
|
|
|
double olat, olon;
|
|
|
|
double odist = fgGetDouble("/sim/presets/offset-distance");
|
|
|
|
odist *= SG_NM_TO_METER;
|
|
|
|
double oaz = azimuth;
|
|
|
|
if ( fabs(fgGetDouble("/sim/presets/offset-azimuth")) > SG_EPSILON )
|
|
|
|
{
|
|
|
|
oaz = fgGetDouble("/sim/presets/offset-azimuth") + 180;
|
2003-07-16 19:12:02 +00:00
|
|
|
heading = fgGetDouble("/sim/presets/heading-deg");
|
2003-03-20 09:38:06 +00:00
|
|
|
}
|
|
|
|
while ( oaz >= 360.0 ) { oaz -= 360.0; }
|
|
|
|
geo_direct_wgs_84 ( 0, lat2, lon2, oaz, odist, &olat, &olon, &az2 );
|
|
|
|
lat2=olat;
|
|
|
|
lon2=olon;
|
2002-11-17 04:17:27 +00:00
|
|
|
}
|
2003-03-20 09:38:06 +00:00
|
|
|
|
|
|
|
// presets
|
|
|
|
fgSetDouble("/sim/presets/longitude-deg", lon2 );
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", lat2 );
|
|
|
|
fgSetDouble("/sim/presets/heading-deg", heading );
|
|
|
|
|
|
|
|
// other code depends on the actual values being set ...
|
|
|
|
fgSetDouble("/position/longitude-deg", lon2 );
|
|
|
|
fgSetDouble("/position/latitude-deg", lat2 );
|
|
|
|
fgSetDouble("/orientation/heading-deg", heading );
|
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
|
|
|
"Position for " << id << " is ("
|
|
|
|
<< lon2 << ", "
|
|
|
|
<< lat2 << ") new heading is "
|
|
|
|
<< heading );
|
|
|
|
|
|
|
|
return true;
|
2002-11-17 04:04:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-16 21:34:51 +00:00
|
|
|
static void fgSetDistOrAltFromGlideSlope() {
|
2003-11-11 17:46:23 +00:00
|
|
|
// cout << "fgSetDistOrAltFromGlideSlope()" << endl;
|
2002-11-30 20:10:16 +00:00
|
|
|
string apt_id = fgGetString("/sim/presets/airport-id");
|
2002-11-18 21:31:33 +00:00
|
|
|
double gs = fgGetDouble("/sim/presets/glideslope-deg")
|
|
|
|
* SG_DEGREES_TO_RADIANS ;
|
2002-11-15 21:13:29 +00:00
|
|
|
double od = fgGetDouble("/sim/presets/offset-distance");
|
|
|
|
double alt = fgGetDouble("/sim/presets/altitude-ft");
|
2002-11-30 20:10:16 +00:00
|
|
|
|
|
|
|
double apt_elev = 0.0;
|
|
|
|
if ( ! apt_id.empty() ) {
|
|
|
|
apt_elev = fgGetAirportElev( apt_id );
|
|
|
|
if ( apt_elev < -9990.0 ) {
|
|
|
|
apt_elev = 0.0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
apt_elev = 0.0;
|
|
|
|
}
|
|
|
|
|
2002-02-20 16:56:44 +00:00
|
|
|
if( fabs(gs) > 0.01 && fabs(od) > 0.1 && alt < -9990 ) {
|
2002-11-16 21:34:51 +00:00
|
|
|
// set altitude from glideslope and offset-distance
|
2002-10-04 13:20:53 +00:00
|
|
|
od *= SG_NM_TO_METER * SG_METER_TO_FEET;
|
2002-11-30 20:10:16 +00:00
|
|
|
alt = fabs(od*tan(gs)) + apt_elev;
|
2002-11-16 21:34:51 +00:00
|
|
|
fgSetDouble("/sim/presets/altitude-ft", alt);
|
2002-11-15 21:13:29 +00:00
|
|
|
fgSetBool("/sim/presets/onground", false);
|
2002-12-12 01:24:48 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Calculated altitude as: "
|
|
|
|
<< alt << " ft" );
|
2002-02-20 16:56:44 +00:00
|
|
|
} else if( fabs(gs) > 0.01 && alt > 0 && fabs(od) < 0.1) {
|
2002-11-16 21:34:51 +00:00
|
|
|
// set offset-distance from glideslope and altitude
|
2002-11-30 20:10:16 +00:00
|
|
|
od = (alt - apt_elev) / tan(gs);
|
2002-10-04 13:20:53 +00:00
|
|
|
od *= -1*SG_FEET_TO_METER * SG_METER_TO_NM;
|
2002-11-16 21:34:51 +00:00
|
|
|
fgSetDouble("/sim/presets/offset-distance", od);
|
2002-12-12 01:24:48 +00:00
|
|
|
fgSetBool("/sim/presets/onground", false);
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Calculated offset distance as: "
|
|
|
|
<< od << " nm" );
|
2002-02-20 16:56:44 +00:00
|
|
|
} else if( fabs(gs) > 0.01 ) {
|
2002-11-16 20:17:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
|
|
|
"Glideslope given but not altitude or offset-distance." );
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Resetting glideslope to zero" );
|
2002-11-18 21:31:33 +00:00
|
|
|
fgSetDouble("/sim/presets/glideslope-deg", 0);
|
2002-11-30 20:10:16 +00:00
|
|
|
fgSetBool("/sim/presets/onground", true);
|
2003-03-20 09:38:06 +00:00
|
|
|
}
|
2004-04-15 22:09:46 +00:00
|
|
|
}
|
2000-08-16 00:09:03 +00:00
|
|
|
|
2002-11-16 20:17:11 +00:00
|
|
|
|
2003-01-05 00:10:36 +00:00
|
|
|
// Set current_options lon/lat given an airport id and heading (degrees)
|
|
|
|
static bool fgSetPosFromNAV( const string& id, const double& freq ) {
|
2004-05-28 05:24:54 +00:00
|
|
|
FGNavRecord *nav
|
|
|
|
= globals->get_navlist()->findByIdentAndFreq( id.c_str(), freq );
|
2003-01-05 00:10:36 +00:00
|
|
|
|
|
|
|
// set initial position from runway and heading
|
2003-01-25 20:45:39 +00:00
|
|
|
if ( nav != NULL ) {
|
2003-01-05 00:10:36 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Attempting to set starting position for "
|
|
|
|
<< id << ":" << freq );
|
|
|
|
|
2003-01-25 20:45:39 +00:00
|
|
|
double lon = nav->get_lon();
|
|
|
|
double lat = nav->get_lat();
|
2003-01-05 00:10:36 +00:00
|
|
|
|
|
|
|
if ( fabs( fgGetDouble("/sim/presets/offset-distance") ) > SG_EPSILON )
|
|
|
|
{
|
|
|
|
double odist = fgGetDouble("/sim/presets/offset-distance")
|
|
|
|
* SG_NM_TO_METER;
|
|
|
|
double oaz = fabs(fgGetDouble("/sim/presets/offset-azimuth"))
|
|
|
|
+ 180.0;
|
|
|
|
while ( oaz >= 360.0 ) { oaz -= 360.0; }
|
|
|
|
double olat, olon, az2;
|
|
|
|
geo_direct_wgs_84 ( 0, lat, lon, oaz, odist, &olat, &olon, &az2 );
|
|
|
|
|
|
|
|
lat = olat;
|
|
|
|
lon = olon;
|
|
|
|
}
|
|
|
|
|
|
|
|
// presets
|
|
|
|
fgSetDouble("/sim/presets/longitude-deg", lon );
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", lat );
|
|
|
|
|
|
|
|
// other code depends on the actual values being set ...
|
|
|
|
fgSetDouble("/position/longitude-deg", lon );
|
|
|
|
fgSetDouble("/position/latitude-deg", lat );
|
|
|
|
fgSetDouble("/orientation/heading-deg",
|
|
|
|
fgGetDouble("/sim/presets/heading-deg") );
|
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
|
|
|
"Position for " << id << ":" << freq << " is ("
|
|
|
|
<< lon << ", "<< lat << ")" );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Failed to locate NAV = "
|
|
|
|
<< id << ":" << freq );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
// Set current_options lon/lat given an aircraft carrier id
|
|
|
|
static bool fgSetPosFromCarrier( const string& carrier, const string& posid ) {
|
|
|
|
|
|
|
|
// set initial position from runway and heading
|
2006-02-19 17:28:31 +00:00
|
|
|
SGGeod geodPos;
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
double heading;
|
2006-02-19 17:28:31 +00:00
|
|
|
SGVec3d uvw;
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
if (FGAIManager::getStartPosition(carrier, posid, geodPos, heading, uvw)) {
|
2006-02-19 17:28:31 +00:00
|
|
|
double lon = geodPos.getLongitudeDeg();
|
|
|
|
double lat = geodPos.getLatitudeDeg();
|
|
|
|
double alt = geodPos.getElevationFt();
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Attempting to set starting position for "
|
|
|
|
<< carrier << " at lat = " << lat << ", lon = " << lon
|
|
|
|
<< ", alt = " << alt << ", heading = " << heading);
|
|
|
|
|
|
|
|
fgSetDouble("/sim/presets/longitude-deg", lon);
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", lat);
|
|
|
|
fgSetDouble("/sim/presets/altitude-ft", alt);
|
|
|
|
fgSetDouble("/sim/presets/heading-deg", heading);
|
|
|
|
fgSetDouble("/position/longitude-deg", lon);
|
|
|
|
fgSetDouble("/position/latitude-deg", lat);
|
|
|
|
fgSetDouble("/position/altitude-ft", alt);
|
|
|
|
fgSetDouble("/orientation/heading-deg", heading);
|
|
|
|
|
|
|
|
fgSetString("/sim/presets/speed-set", "UVW");
|
2006-02-19 17:28:31 +00:00
|
|
|
fgSetDouble("/velocities/uBody-fps", uvw(0));
|
|
|
|
fgSetDouble("/velocities/vBody-fps", uvw(1));
|
|
|
|
fgSetDouble("/velocities/wBody-fps", uvw(2));
|
|
|
|
fgSetDouble("/sim/presets/uBody-fps", uvw(0));
|
|
|
|
fgSetDouble("/sim/presets/vBody-fps", uvw(1));
|
|
|
|
fgSetDouble("/sim/presets/wBody-fps", uvw(2));
|
2003-01-05 00:10:36 +00:00
|
|
|
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
fgSetBool("/sim/presets/onground", true);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
2005-10-04 20:36:38 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Failed to locate aircraft carrier = "
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
<< carrier );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-01-05 00:10:36 +00:00
|
|
|
// Set current_options lon/lat given an airport id and heading (degrees)
|
|
|
|
static bool fgSetPosFromFix( const string& id ) {
|
|
|
|
FGFix fix;
|
|
|
|
|
|
|
|
// set initial position from runway and heading
|
2004-05-26 18:15:19 +00:00
|
|
|
if ( globals->get_fixlist()->query( id.c_str(), &fix ) ) {
|
2003-01-05 00:10:36 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Attempting to set starting position for "
|
|
|
|
<< id );
|
|
|
|
|
|
|
|
double lon = fix.get_lon();
|
|
|
|
double lat = fix.get_lat();
|
|
|
|
|
|
|
|
if ( fabs( fgGetDouble("/sim/presets/offset-distance") ) > SG_EPSILON )
|
|
|
|
{
|
|
|
|
double odist = fgGetDouble("/sim/presets/offset-distance")
|
|
|
|
* SG_NM_TO_METER;
|
|
|
|
double oaz = fabs(fgGetDouble("/sim/presets/offset-azimuth"))
|
|
|
|
+ 180.0;
|
|
|
|
while ( oaz >= 360.0 ) { oaz -= 360.0; }
|
|
|
|
double olat, olon, az2;
|
|
|
|
geo_direct_wgs_84 ( 0, lat, lon, oaz, odist, &olat, &olon, &az2 );
|
|
|
|
|
|
|
|
lat = olat;
|
|
|
|
lon = olon;
|
|
|
|
}
|
|
|
|
|
|
|
|
// presets
|
|
|
|
fgSetDouble("/sim/presets/longitude-deg", lon );
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", lat );
|
|
|
|
|
|
|
|
// other code depends on the actual values being set ...
|
|
|
|
fgSetDouble("/position/longitude-deg", lon );
|
|
|
|
fgSetDouble("/position/latitude-deg", lat );
|
|
|
|
fgSetDouble("/orientation/heading-deg",
|
|
|
|
fgGetDouble("/sim/presets/heading-deg") );
|
|
|
|
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
|
|
|
"Position for " << id << " is ("
|
|
|
|
<< lon << ", "<< lat << ")" );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Failed to locate NAV = "
|
|
|
|
<< id );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2004-04-15 22:09:46 +00:00
|
|
|
|
2004-04-19 02:24:22 +00:00
|
|
|
static void parseWaypoints() {
|
2004-04-18 18:01:10 +00:00
|
|
|
string_list *waypoints = globals->get_initial_waypoints();
|
2004-04-19 02:24:22 +00:00
|
|
|
if (waypoints) {
|
2004-04-18 18:01:10 +00:00
|
|
|
vector<string>::iterator i;
|
|
|
|
for (i = waypoints->begin();
|
|
|
|
i != waypoints->end();
|
|
|
|
i++)
|
2004-04-19 02:24:22 +00:00
|
|
|
{
|
|
|
|
NewWaypoint(*i);
|
|
|
|
}
|
2004-04-18 18:01:10 +00:00
|
|
|
// Now were done using the way points we can deallocate the
|
|
|
|
// memory they used
|
2004-04-19 02:24:22 +00:00
|
|
|
while (waypoints->begin() != waypoints->end()) {
|
|
|
|
waypoints->pop_back();
|
|
|
|
}
|
2004-04-18 18:01:10 +00:00
|
|
|
delete waypoints;
|
|
|
|
globals->set_initial_waypoints(0);
|
2004-04-19 02:24:22 +00:00
|
|
|
}
|
|
|
|
}
|
2004-04-15 22:09:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
2003-01-05 00:10:36 +00:00
|
|
|
|
|
|
|
|
2003-02-21 02:45:47 +00:00
|
|
|
/**
|
2003-08-28 20:53:08 +00:00
|
|
|
* Initialize vor/ndb/ils/fix list management and query systems (as
|
|
|
|
* well as simple airport db list)
|
2003-02-21 02:45:47 +00:00
|
|
|
*/
|
|
|
|
bool
|
|
|
|
fgInitNav ()
|
|
|
|
{
|
2004-12-22 23:57:07 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, "Loading Airport Database ...");
|
|
|
|
|
|
|
|
SGPath aptdb( globals->get_fg_root() );
|
|
|
|
aptdb.append( "Airports/apt.dat" );
|
|
|
|
|
2004-02-23 18:25:29 +00:00
|
|
|
SGPath p_metar( globals->get_fg_root() );
|
|
|
|
p_metar.append( "Airports/metar.dat" );
|
2003-08-28 20:53:08 +00:00
|
|
|
|
2004-12-22 23:57:07 +00:00
|
|
|
FGAirportList *airports = new FGAirportList();
|
|
|
|
globals->set_airports( airports );
|
|
|
|
FGRunwayList *runways = new FGRunwayList();
|
2003-08-29 04:11:23 +00:00
|
|
|
globals->set_runways( runways );
|
|
|
|
|
2004-12-22 23:57:07 +00:00
|
|
|
fgAirportDBLoad( airports, runways, aptdb.str(), p_metar.str() );
|
|
|
|
|
2004-05-26 18:15:19 +00:00
|
|
|
FGNavList *navlist = new FGNavList;
|
2004-05-28 05:24:54 +00:00
|
|
|
FGNavList *loclist = new FGNavList;
|
|
|
|
FGNavList *gslist = new FGNavList;
|
|
|
|
FGNavList *dmelist = new FGNavList;
|
|
|
|
FGNavList *mkrlist = new FGNavList;
|
2005-10-01 09:56:53 +00:00
|
|
|
FGNavList *tacanlist = new FGNavList;
|
|
|
|
FGNavList *carrierlist = new FGNavList;
|
|
|
|
FGTACANList *channellist = new FGTACANList;
|
2004-05-28 05:24:54 +00:00
|
|
|
|
2004-05-26 18:15:19 +00:00
|
|
|
globals->set_navlist( navlist );
|
2004-05-28 05:24:54 +00:00
|
|
|
globals->set_loclist( loclist );
|
|
|
|
globals->set_gslist( gslist );
|
|
|
|
globals->set_dmelist( dmelist );
|
|
|
|
globals->set_mkrlist( mkrlist );
|
2005-10-01 09:56:53 +00:00
|
|
|
globals->set_tacanlist( tacanlist );
|
|
|
|
globals->set_carrierlist( carrierlist );
|
|
|
|
globals->set_channellist( channellist );
|
2003-02-21 02:45:47 +00:00
|
|
|
|
2005-10-01 09:56:53 +00:00
|
|
|
if ( !fgNavDBInit(airports, navlist, loclist, gslist, dmelist, mkrlist, tacanlist, carrierlist, channellist) ) {
|
2004-05-28 05:24:54 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
|
|
|
"Problems loading one or more navigational database" );
|
|
|
|
}
|
2003-02-21 02:45:47 +00:00
|
|
|
|
2004-05-28 20:57:05 +00:00
|
|
|
if ( fgGetBool("/sim/navdb/localizers/auto-align", true) ) {
|
2004-05-28 16:24:43 +00:00
|
|
|
// align all the localizers with their corresponding runways
|
|
|
|
// since data sources are good for cockpit navigation
|
|
|
|
// purposes, but not always to the error tolerances needed to
|
|
|
|
// exactly place these items.
|
2004-05-28 20:57:05 +00:00
|
|
|
double threshold
|
|
|
|
= fgGetDouble( "/sim/navdb/localizers/auto-align-threshold-deg",
|
|
|
|
5.0 );
|
|
|
|
fgNavDBAlignLOCwithRunway( runways, loclist, threshold );
|
2004-05-28 16:24:43 +00:00
|
|
|
}
|
|
|
|
|
2003-02-21 02:45:47 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, " Fixes");
|
|
|
|
SGPath p_fix( globals->get_fg_root() );
|
2004-05-26 16:40:27 +00:00
|
|
|
p_fix.append( "Navaids/fix.dat" );
|
2004-05-28 05:24:54 +00:00
|
|
|
FGFixList *fixlist = new FGFixList;
|
2004-05-26 18:15:19 +00:00
|
|
|
fixlist->init( p_fix );
|
|
|
|
globals->set_fixlist( fixlist );
|
2003-03-03 04:34:27 +00:00
|
|
|
|
|
|
|
return true;
|
2003-02-21 02:45:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-16 21:34:51 +00:00
|
|
|
// Set the initial position based on presets (or defaults)
|
|
|
|
bool fgInitPosition() {
|
2003-11-11 17:46:23 +00:00
|
|
|
// cout << "fgInitPosition()" << endl;
|
2003-11-10 22:00:22 +00:00
|
|
|
double gs = fgGetDouble("/sim/presets/glideslope-deg")
|
|
|
|
* SG_DEGREES_TO_RADIANS ;
|
|
|
|
double od = fgGetDouble("/sim/presets/offset-distance");
|
|
|
|
double alt = fgGetDouble("/sim/presets/altitude-ft");
|
|
|
|
|
2002-11-17 04:04:21 +00:00
|
|
|
bool set_pos = false;
|
|
|
|
|
2003-05-13 03:18:42 +00:00
|
|
|
// If glideslope is specified, then calculate offset-distance or
|
|
|
|
// altitude relative to glide slope if either of those was not
|
|
|
|
// specified.
|
2003-11-10 22:00:22 +00:00
|
|
|
if ( fabs( gs ) > 0.01 ) {
|
2003-05-13 03:18:42 +00:00
|
|
|
fgSetDistOrAltFromGlideSlope();
|
|
|
|
}
|
|
|
|
|
2003-05-07 16:00:31 +00:00
|
|
|
|
2002-11-16 21:34:51 +00:00
|
|
|
// If we have an explicit, in-range lon/lat, don't change it, just use it.
|
|
|
|
// If not, check for an airport-id and use that.
|
|
|
|
// If not, default to the middle of the KSFO field.
|
|
|
|
// The default values for lon/lat are deliberately out of range
|
|
|
|
// so that the airport-id can take effect; valid lon/lat will
|
|
|
|
// override airport-id, however.
|
|
|
|
double lon_deg = fgGetDouble("/sim/presets/longitude-deg");
|
|
|
|
double lat_deg = fgGetDouble("/sim/presets/latitude-deg");
|
|
|
|
if ( lon_deg >= -180.0 && lon_deg <= 180.0
|
|
|
|
&& lat_deg >= -90.0 && lat_deg <= 90.0 )
|
|
|
|
{
|
2002-11-17 04:04:21 +00:00
|
|
|
set_pos = true;
|
|
|
|
}
|
2002-11-16 21:34:51 +00:00
|
|
|
|
2002-11-17 04:04:21 +00:00
|
|
|
string apt = fgGetString("/sim/presets/airport-id");
|
|
|
|
string rwy_no = fgGetString("/sim/presets/runway");
|
|
|
|
double hdg = fgGetDouble("/sim/presets/heading-deg");
|
2003-01-05 00:10:36 +00:00
|
|
|
string vor = fgGetString("/sim/presets/vor-id");
|
|
|
|
double vor_freq = fgGetDouble("/sim/presets/vor-freq");
|
|
|
|
string ndb = fgGetString("/sim/presets/ndb-id");
|
|
|
|
double ndb_freq = fgGetDouble("/sim/presets/ndb-freq");
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
string carrier = fgGetString("/sim/presets/carrier");
|
|
|
|
string parkpos = fgGetString("/sim/presets/parkpos");
|
2003-01-05 00:10:36 +00:00
|
|
|
string fix = fgGetString("/sim/presets/fix");
|
2003-07-02 16:24:50 +00:00
|
|
|
|
2005-10-19 19:21:45 +00:00
|
|
|
fgSetDouble( "/orientation/heading-deg", hdg );
|
|
|
|
fgInitTowerLocationListener();
|
|
|
|
|
2002-11-17 04:04:21 +00:00
|
|
|
if ( !set_pos && !apt.empty() && !rwy_no.empty() ) {
|
|
|
|
// An airport + runway is requested
|
|
|
|
if ( fgSetPosFromAirportIDandRwy( apt, rwy_no ) ) {
|
2003-07-02 16:24:50 +00:00
|
|
|
// set tower position (a little off the heading for single
|
2002-11-16 21:34:51 +00:00
|
|
|
// runway airports)
|
2005-10-19 19:21:45 +00:00
|
|
|
fgSetString("/sim/tower/airport-id", apt.c_str());
|
2002-11-17 04:04:21 +00:00
|
|
|
set_pos = true;
|
|
|
|
}
|
|
|
|
}
|
2003-07-02 16:24:50 +00:00
|
|
|
|
2002-11-17 04:04:21 +00:00
|
|
|
if ( !set_pos && !apt.empty() ) {
|
2003-01-05 00:10:36 +00:00
|
|
|
// An airport is requested (find runway closest to hdg)
|
2003-07-02 16:24:50 +00:00
|
|
|
if ( fgSetPosFromAirportIDandHdg( apt, hdg ) ) {
|
|
|
|
// set tower position (a little off the heading for single
|
|
|
|
// runway airports)
|
2005-10-19 19:21:45 +00:00
|
|
|
fgSetString("/sim/tower/airport-id", apt.c_str());
|
2002-11-17 04:04:21 +00:00
|
|
|
set_pos = true;
|
2002-11-16 21:34:51 +00:00
|
|
|
}
|
|
|
|
}
|
2003-07-02 16:24:50 +00:00
|
|
|
|
2003-01-05 00:10:36 +00:00
|
|
|
if ( !set_pos && !vor.empty() ) {
|
|
|
|
// a VOR is requested
|
|
|
|
if ( fgSetPosFromNAV( vor, vor_freq ) ) {
|
|
|
|
set_pos = true;
|
|
|
|
}
|
|
|
|
}
|
2003-07-02 16:24:50 +00:00
|
|
|
|
2003-01-05 00:10:36 +00:00
|
|
|
if ( !set_pos && !ndb.empty() ) {
|
|
|
|
// an NDB is requested
|
|
|
|
if ( fgSetPosFromNAV( ndb, ndb_freq ) ) {
|
|
|
|
set_pos = true;
|
|
|
|
}
|
|
|
|
}
|
2003-07-02 16:24:50 +00:00
|
|
|
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
if ( !set_pos && !carrier.empty() ) {
|
|
|
|
// an aircraft carrier is requested
|
|
|
|
if ( fgSetPosFromCarrier( carrier, parkpos ) ) {
|
|
|
|
set_pos = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-01-05 00:10:36 +00:00
|
|
|
if ( !set_pos && !fix.empty() ) {
|
|
|
|
// a Fix is requested
|
|
|
|
if ( fgSetPosFromFix( fix ) ) {
|
|
|
|
set_pos = true;
|
|
|
|
}
|
|
|
|
}
|
2002-11-16 21:34:51 +00:00
|
|
|
|
2002-11-17 04:04:21 +00:00
|
|
|
if ( !set_pos ) {
|
|
|
|
// No lon/lat specified, no airport specified, default to
|
|
|
|
// middle of KSFO field.
|
|
|
|
fgSetDouble("/sim/presets/longitude-deg", -122.374843);
|
|
|
|
fgSetDouble("/sim/presets/latitude-deg", 37.619002);
|
|
|
|
}
|
|
|
|
|
2002-11-16 21:34:51 +00:00
|
|
|
fgSetDouble( "/position/longitude-deg",
|
|
|
|
fgGetDouble("/sim/presets/longitude-deg") );
|
|
|
|
fgSetDouble( "/position/latitude-deg",
|
|
|
|
fgGetDouble("/sim/presets/latitude-deg") );
|
|
|
|
|
2003-11-10 22:00:22 +00:00
|
|
|
// determine if this should be an on-ground or in-air start
|
Mathias Fröhlich:
I have introduced the posibility to start directly on the carrier.
With that patch you will have a --carrrier=id argument where id can either be
the pennant number configured in the nimitz scenario or the carriers name
also configured in the carriers scenario.
Additionaly you can use --parkpos=id to select different positions on the
carrier. They are also configured in the scenario file.
That includes the switch of the whole FGInterface class to make use of the
groundcache.
That means that an aircraft no longer uses the current elevation value from
the scenery class. It rather has its own local cache of the aircrafts
environment which is setup in the common_init method of FGInterface and
updated either manually by calling
FGInterface::get_groundlevel_m(lat, lon, alt_m);
or implicitly by calling the above method in the
FGInterface::_updateGeo*Position(lat, lon, alt);
methods.
A call get_groundlevel_m rebuilds the groundcache if the request is outside
the range of the cache.
Note that for the real usage of the groundcache including the correct
information about the movement of objects and the velocity information, you
still need to set up the groundcache in the usual way like YASim and JSBSim
currently does.
If you use the native interface, you will get only static objects correctly.
But for FDM's only using one single ground level for a whole step this is IMO
sufficient.
The AIManager gets a way to return the location of a object which is placed
wrt an AI Object. At the moment it only honours AICarriers for that.
That method is a static one, which loads the scenario file for that reason and
throws it away afterwards. This looked like the aprioriate way, because the
AIManager is initialized much later in flightgears bootstrap, and I did not
find an easy way to reorder that for my needs. Since this additional load is
very small and does only happen if such a relative location is required, I
think that this is ok.
Note that moving on the carrier will only work correctly for JSBSim and YASim,
but you should now be able to start and move on every not itself moving
object with any FDM.
2005-07-03 09:39:14 +00:00
|
|
|
if ((fabs(gs) > 0.01 || fabs(od) > 0.1 || alt > 0.1) && carrier.empty()) {
|
2003-11-10 22:00:22 +00:00
|
|
|
fgSetBool("/sim/presets/onground", false);
|
|
|
|
} else {
|
|
|
|
fgSetBool("/sim/presets/onground", true);
|
|
|
|
}
|
|
|
|
|
2002-11-16 21:34:51 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
1998-04-25 15:11:10 +00:00
|
|
|
// General house keeping initializations
|
2002-11-16 21:34:51 +00:00
|
|
|
bool fgInitGeneral() {
|
1998-08-27 17:01:55 +00:00
|
|
|
string root;
|
1999-11-19 02:10:24 +00:00
|
|
|
|
|
|
|
#if defined(FX) && defined(XMESA)
|
1999-03-11 23:09:26 +00:00
|
|
|
char *mesa_win_state;
|
1999-11-19 02:10:24 +00:00
|
|
|
#endif
|
1998-04-25 15:11:10 +00:00
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "General Initialization" );
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "======= ==============" );
|
1998-04-25 20:24:00 +00:00
|
|
|
|
2001-01-13 22:06:39 +00:00
|
|
|
root = globals->get_fg_root();
|
1998-09-08 21:40:08 +00:00
|
|
|
if ( ! root.length() ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
// No root path set? Then bail ...
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
|
|
|
"Cannot continue without a path to the base package "
|
|
|
|
<< "being defined." );
|
|
|
|
exit(-1);
|
1998-04-25 15:11:10 +00:00
|
|
|
}
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "FG_ROOT = " << '"' << root << '"' << endl );
|
1998-04-25 15:11:10 +00:00
|
|
|
|
1999-03-11 23:09:26 +00:00
|
|
|
#if defined(FX) && defined(XMESA)
|
|
|
|
// initialize full screen flag
|
2002-01-06 16:51:31 +00:00
|
|
|
globals->set_fullscreen(false);
|
1999-03-11 23:09:26 +00:00
|
|
|
if ( strstr ( general.get_glRenderer(), "Glide" ) ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
// Test for the MESA_GLX_FX env variable
|
|
|
|
if ( (mesa_win_state = getenv( "MESA_GLX_FX" )) != NULL) {
|
|
|
|
// test if we are fullscreen mesa/glide
|
|
|
|
if ( (mesa_win_state[0] == 'f') ||
|
|
|
|
(mesa_win_state[0] == 'F') ) {
|
|
|
|
globals->set_fullscreen(true);
|
|
|
|
}
|
|
|
|
}
|
1999-03-11 23:09:26 +00:00
|
|
|
}
|
|
|
|
#endif
|
1998-05-07 23:14:14 +00:00
|
|
|
|
1999-06-18 03:42:54 +00:00
|
|
|
return true;
|
1998-04-25 15:11:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-02-05 20:54:08 +00:00
|
|
|
// Initialize the flight model subsystem. This just creates the
|
|
|
|
// object. The actual fdm initialization is delayed until we get a
|
|
|
|
// proper scenery elevation hit. This is checked for in main.cxx
|
|
|
|
|
|
|
|
void fgInitFDM() {
|
|
|
|
|
|
|
|
if ( cur_fdm_state ) {
|
|
|
|
delete cur_fdm_state;
|
|
|
|
cur_fdm_state = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
double dt = 1.0 / fgGetInt("/sim/model-hz");
|
2005-10-12 08:55:58 +00:00
|
|
|
string model = fgGetString("/sim/flight-model");
|
2002-02-05 20:54:08 +00:00
|
|
|
|
|
|
|
try {
|
2002-10-04 13:20:53 +00:00
|
|
|
if ( model == "larcsim" ) {
|
|
|
|
cur_fdm_state = new FGLaRCsim( dt );
|
|
|
|
} else if ( model == "jsb" ) {
|
|
|
|
cur_fdm_state = new FGJSBsim( dt );
|
2005-11-29 03:12:24 +00:00
|
|
|
#if ENABLE_SP_FDMS
|
2002-10-04 13:20:53 +00:00
|
|
|
} else if ( model == "ada" ) {
|
|
|
|
cur_fdm_state = new FGADA( dt );
|
2004-10-19 11:10:20 +00:00
|
|
|
} else if ( model == "acms" ) {
|
|
|
|
cur_fdm_state = new FGACMS( dt );
|
|
|
|
#endif
|
2002-10-04 13:20:53 +00:00
|
|
|
} else if ( model == "balloon" ) {
|
|
|
|
cur_fdm_state = new FGBalloonSim( dt );
|
|
|
|
} else if ( model == "magic" ) {
|
|
|
|
cur_fdm_state = new FGMagicCarpet( dt );
|
|
|
|
} else if ( model == "ufo" ) {
|
|
|
|
cur_fdm_state = new FGUFO( dt );
|
|
|
|
} else if ( model == "external" ) {
|
2002-09-28 21:42:03 +00:00
|
|
|
// external is a synonym for "--fdm=null" and is
|
|
|
|
// maintained here for backwards compatibility
|
2002-10-04 13:20:53 +00:00
|
|
|
cur_fdm_state = new FGNullFDM( dt );
|
|
|
|
} else if ( model.find("network") == 0 ) {
|
2002-04-11 05:40:34 +00:00
|
|
|
string host = "localhost";
|
|
|
|
int port1 = 5501;
|
|
|
|
int port2 = 5502;
|
|
|
|
int port3 = 5503;
|
|
|
|
string net_options = model.substr(8);
|
|
|
|
string::size_type begin, end;
|
|
|
|
begin = 0;
|
|
|
|
// host
|
|
|
|
end = net_options.find( ",", begin );
|
|
|
|
if ( end != string::npos ) {
|
|
|
|
host = net_options.substr(begin, end - begin);
|
|
|
|
begin = end + 1;
|
|
|
|
}
|
|
|
|
// port1
|
|
|
|
end = net_options.find( ",", begin );
|
|
|
|
if ( end != string::npos ) {
|
|
|
|
port1 = atoi( net_options.substr(begin, end - begin).c_str() );
|
|
|
|
begin = end + 1;
|
|
|
|
}
|
|
|
|
// port2
|
|
|
|
end = net_options.find( ",", begin );
|
|
|
|
if ( end != string::npos ) {
|
|
|
|
port2 = atoi( net_options.substr(begin, end - begin).c_str() );
|
|
|
|
begin = end + 1;
|
|
|
|
}
|
|
|
|
// port3
|
|
|
|
end = net_options.find( ",", begin );
|
|
|
|
if ( end != string::npos ) {
|
|
|
|
port3 = atoi( net_options.substr(begin, end - begin).c_str() );
|
|
|
|
begin = end + 1;
|
|
|
|
}
|
2002-10-04 13:20:53 +00:00
|
|
|
cur_fdm_state = new FGExternalNet( dt, host, port1, port2, port3 );
|
2003-03-03 04:34:27 +00:00
|
|
|
} else if ( model.find("pipe") == 0 ) {
|
2005-04-19 01:44:56 +00:00
|
|
|
// /* old */ string pipe_path = model.substr(5);
|
|
|
|
// /* old */ cur_fdm_state = new FGExternalPipe( dt, pipe_path );
|
|
|
|
string pipe_path = "";
|
|
|
|
string pipe_protocol = "";
|
|
|
|
string pipe_options = model.substr(5);
|
|
|
|
string::size_type begin, end;
|
|
|
|
begin = 0;
|
|
|
|
// pipe file path
|
|
|
|
end = pipe_options.find( ",", begin );
|
|
|
|
if ( end != string::npos ) {
|
|
|
|
pipe_path = pipe_options.substr(begin, end - begin);
|
|
|
|
begin = end + 1;
|
|
|
|
}
|
|
|
|
// protocol (last option)
|
|
|
|
pipe_protocol = pipe_options.substr(begin);
|
|
|
|
cur_fdm_state = new FGExternalPipe( dt, pipe_path, pipe_protocol );
|
2002-10-04 13:20:53 +00:00
|
|
|
} else if ( model == "null" ) {
|
|
|
|
cur_fdm_state = new FGNullFDM( dt );
|
|
|
|
} else if ( model == "yasim" ) {
|
|
|
|
cur_fdm_state = new YASim( dt );
|
|
|
|
} else {
|
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT,
|
|
|
|
"Unrecognized flight model '" << model
|
|
|
|
<< "', cannot init flight dynamics model.");
|
|
|
|
exit(-1);
|
|
|
|
}
|
2002-02-05 20:54:08 +00:00
|
|
|
} catch ( ... ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_ALERT, "FlightGear aborting\n\n");
|
|
|
|
exit(-1);
|
2002-02-05 20:54:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-09-18 14:57:20 +00:00
|
|
|
static void printMat(const sgVec4 *mat, char *name="")
|
|
|
|
{
|
|
|
|
int i;
|
2003-03-20 10:43:02 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_BULK, name );
|
2002-09-18 14:57:20 +00:00
|
|
|
for(i=0; i<4; i++) {
|
2003-03-20 10:43:02 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_BULK, " " << mat[i][0] << " " << mat[i][1]
|
|
|
|
<< " " << mat[i][2] << " " << mat[i][3] );
|
2002-09-18 14:57:20 +00:00
|
|
|
}
|
|
|
|
}
|
2002-02-05 20:54:08 +00:00
|
|
|
|
|
|
|
// Initialize view parameters
|
|
|
|
void fgInitView() {
|
2002-04-13 13:11:34 +00:00
|
|
|
// force update of model so that viewer can get some data...
|
|
|
|
globals->get_aircraft_model()->update(0);
|
2002-04-21 03:27:34 +00:00
|
|
|
// run update for current view so that data is current...
|
2002-03-14 00:29:20 +00:00
|
|
|
globals->get_viewmgr()->update(0);
|
2002-09-18 14:57:20 +00:00
|
|
|
|
|
|
|
printMat(globals->get_current_view()->get_VIEW(),"VIEW");
|
|
|
|
printMat(globals->get_current_view()->get_UP(),"UP");
|
|
|
|
// printMat(globals->get_current_view()->get_LOCAL(),"LOCAL");
|
|
|
|
|
2002-02-05 20:54:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-02-06 23:31:33 +00:00
|
|
|
SGTime *fgInitTime() {
|
|
|
|
// Initialize time
|
|
|
|
static const SGPropertyNode *longitude
|
2002-10-04 13:20:53 +00:00
|
|
|
= fgGetNode("/position/longitude-deg");
|
2002-02-06 23:31:33 +00:00
|
|
|
static const SGPropertyNode *latitude
|
2002-10-04 13:20:53 +00:00
|
|
|
= fgGetNode("/position/latitude-deg");
|
2002-02-11 23:33:20 +00:00
|
|
|
static const SGPropertyNode *cur_time_override
|
2002-10-04 13:20:53 +00:00
|
|
|
= fgGetNode("/sim/time/cur-time-override", true);
|
2002-02-11 23:33:20 +00:00
|
|
|
|
2002-02-06 23:31:33 +00:00
|
|
|
SGPath zone( globals->get_fg_root() );
|
|
|
|
zone.append( "Timezone" );
|
|
|
|
SGTime *t = new SGTime( longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
2002-02-11 23:33:20 +00:00
|
|
|
zone.str(),
|
|
|
|
cur_time_override->getLongValue() );
|
2002-02-06 23:31:33 +00:00
|
|
|
|
2003-09-15 22:55:39 +00:00
|
|
|
globals->set_warp_delta( 0 );
|
|
|
|
|
|
|
|
t->update( 0.0, 0.0,
|
|
|
|
cur_time_override->getLongValue(),
|
|
|
|
globals->get_warp() );
|
|
|
|
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// set up a time offset (aka warp) if one is specified
|
|
|
|
void fgInitTimeOffset() {
|
|
|
|
static const SGPropertyNode *longitude
|
|
|
|
= fgGetNode("/position/longitude-deg");
|
|
|
|
static const SGPropertyNode *latitude
|
|
|
|
= fgGetNode("/position/latitude-deg");
|
|
|
|
static const SGPropertyNode *cur_time_override
|
|
|
|
= fgGetNode("/sim/time/cur-time-override", true);
|
|
|
|
|
2002-02-06 23:31:33 +00:00
|
|
|
// Handle potential user specified time offsets
|
2003-09-16 22:34:22 +00:00
|
|
|
int orig_warp = globals->get_warp();
|
2003-09-15 22:55:39 +00:00
|
|
|
SGTime *t = globals->get_time_params();
|
2002-02-06 23:31:33 +00:00
|
|
|
time_t cur_time = t->get_cur_time();
|
|
|
|
time_t currGMT = sgTimeGetGMT( gmtime(&cur_time) );
|
|
|
|
time_t systemLocalTime = sgTimeGetGMT( localtime(&cur_time) );
|
|
|
|
time_t aircraftLocalTime =
|
|
|
|
sgTimeGetGMT( fgLocaltime(&cur_time, t->get_zonename() ) );
|
|
|
|
|
2003-09-15 22:55:39 +00:00
|
|
|
// Okay, we now have several possible scenarios
|
2002-02-06 23:31:33 +00:00
|
|
|
int offset = fgGetInt("/sim/startup/time-offset");
|
2005-10-12 08:55:58 +00:00
|
|
|
string offset_type = fgGetString("/sim/startup/time-offset-type");
|
2003-09-17 15:49:15 +00:00
|
|
|
|
|
|
|
int warp = 0;
|
|
|
|
if ( offset_type == "real" ) {
|
|
|
|
warp = 0;
|
|
|
|
} else if ( offset_type == "dawn" ) {
|
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
2003-09-15 22:55:39 +00:00
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
2003-09-17 15:49:15 +00:00
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
90.0, true );
|
|
|
|
} else if ( offset_type == "morning" ) {
|
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
75.0, true );
|
|
|
|
} else if ( offset_type == "noon" ) {
|
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
0.0, true );
|
|
|
|
} else if ( offset_type == "afternoon" ) {
|
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
60.0, false );
|
2003-09-15 22:55:39 +00:00
|
|
|
} else if ( offset_type == "dusk" ) {
|
2003-09-17 15:49:15 +00:00
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
90.0, false );
|
|
|
|
} else if ( offset_type == "evening" ) {
|
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
100.0, false );
|
|
|
|
} else if ( offset_type == "midnight" ) {
|
|
|
|
warp = fgTimeSecondsUntilSunAngle( cur_time,
|
|
|
|
longitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue()
|
|
|
|
* SGD_DEGREES_TO_RADIANS,
|
|
|
|
180.0, false );
|
2003-09-15 22:55:39 +00:00
|
|
|
} else if ( offset_type == "system-offset" ) {
|
|
|
|
warp = offset;
|
|
|
|
} else if ( offset_type == "gmt-offset" ) {
|
|
|
|
warp = offset - (currGMT - systemLocalTime);
|
|
|
|
} else if ( offset_type == "latitude-offset" ) {
|
|
|
|
warp = offset - (aircraftLocalTime - systemLocalTime);
|
|
|
|
} else if ( offset_type == "system" ) {
|
|
|
|
warp = offset - cur_time;
|
|
|
|
} else if ( offset_type == "gmt" ) {
|
|
|
|
warp = offset - currGMT;
|
|
|
|
} else if ( offset_type == "latitude" ) {
|
|
|
|
warp = offset - (aircraftLocalTime - systemLocalTime) - cur_time;
|
2002-02-06 23:31:33 +00:00
|
|
|
} else {
|
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT,
|
|
|
|
"FG_TIME::Unsupported offset type " << offset_type );
|
|
|
|
exit( -1 );
|
|
|
|
}
|
2003-09-16 22:34:22 +00:00
|
|
|
globals->set_warp( orig_warp + warp );
|
|
|
|
t->update( longitude->getDoubleValue() * SGD_DEGREES_TO_RADIANS,
|
|
|
|
latitude->getDoubleValue() * SGD_DEGREES_TO_RADIANS,
|
|
|
|
cur_time_override->getLongValue(),
|
2003-09-15 22:55:39 +00:00
|
|
|
globals->get_warp() );
|
2002-02-06 23:31:33 +00:00
|
|
|
|
2003-09-15 22:55:39 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "After fgInitTimeOffset(): warp = "
|
2002-02-06 23:31:33 +00:00
|
|
|
<< globals->get_warp() );
|
|
|
|
}
|
|
|
|
|
2004-11-18 05:00:44 +00:00
|
|
|
|
1998-04-25 15:11:10 +00:00
|
|
|
// This is the top level init routine which calls all the other
|
|
|
|
// initialization routines. If you are adding a subsystem to flight
|
|
|
|
// gear, its initialization call should located in this routine.
|
|
|
|
// Returns non-zero if a problem encountered.
|
2002-11-18 21:31:33 +00:00
|
|
|
bool fgInitSubsystems() {
|
2003-05-06 23:46:24 +00:00
|
|
|
// static const SGPropertyNode *longitude
|
|
|
|
// = fgGetNode("/sim/presets/longitude-deg");
|
|
|
|
// static const SGPropertyNode *latitude
|
|
|
|
// = fgGetNode("/sim/presets/latitude-deg");
|
|
|
|
// static const SGPropertyNode *altitude
|
|
|
|
// = fgGetNode("/sim/presets/altitude-ft");
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "Initialize Subsystems");
|
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, "========== ==========");
|
1998-04-25 15:11:10 +00:00
|
|
|
|
2003-09-24 17:20:55 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the event manager subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2004-11-18 05:00:44 +00:00
|
|
|
globals->get_event_mgr()->init();
|
2005-11-09 20:34:46 +00:00
|
|
|
globals->get_event_mgr()->setRealtimeProperty(fgGetNode("/sim/time/delta-realtime-sec", true));
|
2001-04-05 20:20:44 +00:00
|
|
|
|
2003-12-05 01:52:34 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the property interpolator subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2004-02-20 17:35:33 +00:00
|
|
|
globals->add_subsystem("interpolator", new SGInterpolator);
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Add the FlightGear property utilities.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("properties", new FGProperties);
|
2003-12-05 01:52:34 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the material property subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-03-25 14:20:12 +00:00
|
|
|
SGPath mpath( globals->get_fg_root() );
|
2001-12-28 23:33:32 +00:00
|
|
|
mpath.append( "materials.xml" );
|
2005-10-23 13:48:36 +00:00
|
|
|
string season = fgGetString("/sim/startup/season", "summer");
|
|
|
|
if ( ! globals->get_matlib()->load(globals->get_fg_root(), mpath.str(), season.c_str()) ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Error loading material lib!" );
|
|
|
|
exit(-1);
|
2000-06-23 00:30:04 +00:00
|
|
|
}
|
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the scenery management subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2002-12-06 22:29:46 +00:00
|
|
|
if ( globals->get_tile_mgr()->init() ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
// Load the local scenery data
|
|
|
|
double visibility_meters = fgGetDouble("/environment/visibility-m");
|
|
|
|
|
2002-12-06 22:29:46 +00:00
|
|
|
globals->get_tile_mgr()->update( visibility_meters );
|
1998-08-22 14:49:55 +00:00
|
|
|
} else {
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Error in Tile Manager initialization!" );
|
|
|
|
exit(-1);
|
1998-08-22 14:49:55 +00:00
|
|
|
}
|
|
|
|
|
From: "Jim Wilson" <jimw@kelcomaine.com>
This is a new improved patch for the previous tile manager fixes.
Rather than building dependencies between FGlocation or the viewer or fdm with
tilemgr what I ended up doing was linking the pieces together in the Mainloop
in main.cxx. You'll see what I mean...it's been commented fairly well. More
than likely we should move that chunk somewhere...just not sure where yet.
The changes seem clean now. As I get more ideas there could be some further
improvement in organizing the update in tilemgr. You'll note that I left an
override in there for the tilemgr::update() function to preserve earlier
functionality if someone needs it (e.g. usage independent of an fdm or
viewer), not to mention there are a few places in flightgear that call it
directly that have not been changed to the new interface (and may not need to be).
The code has been optimized to avoid duplicate traversals and seems to run
generally quite well. Note that there can be a short delay reloading tiles
that have been dropped from static views. We could call the tile scheduler on
a view switch, but it's not a big deal and at the moment I'd like to get this
in so people can try it and comment on it as it is.
Everything has been resycned with CVS tonight and I've included the
description submitted earlier (below).
Best,
Jim
Changes synced with CVS approx 20:30EDT 2002-05-09 (after this evenings updates).
Files:
http://www.spiderbark.com/fgfs/viewer-update-20020516.tar.gz
or
http://www.spiderbark.com/fgfs/viewer-update-20020516.diffs.gz
Description:
In a nutshell, these patches begin to take what was one value for ground
elevation and calculate ground elevation values seperately for the FDM and the
viewer (eye position). Several outstanding view related bugs have been fixed.
With the introduction of the new viewer code a lot of that Flight Gear code
broke related to use of a global variable called "scenery.cur_elev".
Therefore the ground_elevation and other associated items (like the current
tile bucket) is maintained per FDM instance and per View. Each of these has a
"point" or location that can be identified. See changes to FGLocation class
and main.cxx.
Most of the problems related to the new viewer in terms of sky, ground and
runway lights, and tower views are fixed.
There are four minor problems remaining. 1) The sun/moon spins when you pan
the "lookat" tower view only (view #3). 2) Under stress (esp. magic carpet
full speed with max visibility), there is a memory leak in the tile caching
that was not introduced with these changes. 3) I have not tested these
changes or made corrections to the ADA or External FDM interfaces. 4) The
change view function doesn't call the time/light update (not a problem unless
a tower is very far away).
Details:
FDM/flight.cxx, flight.hxx - FGInterface ties to FGAircraftModel so that it's
location data can be accessed for runway (ground elevation under aircraft)
elevation.
FDM/larsim.cxx, larcsim.hxx - gets runway elevation from FGInterface now.
Commented out function that is causing a namespace conflict, hasn't been
called with recent code anyway.
FDM/JSBSim/JSBSim.cxx, YASim/YASim.cxx - gets runway elevation from
FGInterface now.
Scenery/newcache.cxx, newcache.hxx - changed caching scheme to time based
(oldest tiles discard).
Scenery/tileentry.cxx, tileentry.hxx - added place to record time, changed
rendering to reference viewer altitude in order to fix a problem with ground
and runway lights.
Scenery/tilemgr.cxx, tilemgr.hxx - Modified update() to accept values for
multiple locations. Refresh function added in order to periodically make
the tiles current for a non-moving view (like a tower).
Main/fg_init.cxx - register event for making tiles current in a non-moving
view (like a tower).
Main/location.hxx - added support for current ground elevation data.
Main/main.cxx - added second tilemgr call for fdm, fixed places where viewer
position data was required for correct sky rendering.
Main/options.cxx - fixed segfault reported by Curtis when using --view-offset
command line parameter.
Main/viewer.cxx, viewer.hxx - removed fudging of view position. Fixed numerous
bugs that were causing eye and target values to get mixed up.
2002-05-17 17:25:28 +00:00
|
|
|
// cause refresh of viewer scenery timestamps every 15 seconds...
|
2003-11-25 21:08:36 +00:00
|
|
|
globals->get_event_mgr()->addTask( "FGTileMgr::refresh_view_timestamps()",
|
|
|
|
globals->get_tile_mgr(),
|
|
|
|
&FGTileMgr::refresh_view_timestamps,
|
|
|
|
15 );
|
From: "Jim Wilson" <jimw@kelcomaine.com>
This is a new improved patch for the previous tile manager fixes.
Rather than building dependencies between FGlocation or the viewer or fdm with
tilemgr what I ended up doing was linking the pieces together in the Mainloop
in main.cxx. You'll see what I mean...it's been commented fairly well. More
than likely we should move that chunk somewhere...just not sure where yet.
The changes seem clean now. As I get more ideas there could be some further
improvement in organizing the update in tilemgr. You'll note that I left an
override in there for the tilemgr::update() function to preserve earlier
functionality if someone needs it (e.g. usage independent of an fdm or
viewer), not to mention there are a few places in flightgear that call it
directly that have not been changed to the new interface (and may not need to be).
The code has been optimized to avoid duplicate traversals and seems to run
generally quite well. Note that there can be a short delay reloading tiles
that have been dropped from static views. We could call the tile scheduler on
a view switch, but it's not a big deal and at the moment I'd like to get this
in so people can try it and comment on it as it is.
Everything has been resycned with CVS tonight and I've included the
description submitted earlier (below).
Best,
Jim
Changes synced with CVS approx 20:30EDT 2002-05-09 (after this evenings updates).
Files:
http://www.spiderbark.com/fgfs/viewer-update-20020516.tar.gz
or
http://www.spiderbark.com/fgfs/viewer-update-20020516.diffs.gz
Description:
In a nutshell, these patches begin to take what was one value for ground
elevation and calculate ground elevation values seperately for the FDM and the
viewer (eye position). Several outstanding view related bugs have been fixed.
With the introduction of the new viewer code a lot of that Flight Gear code
broke related to use of a global variable called "scenery.cur_elev".
Therefore the ground_elevation and other associated items (like the current
tile bucket) is maintained per FDM instance and per View. Each of these has a
"point" or location that can be identified. See changes to FGLocation class
and main.cxx.
Most of the problems related to the new viewer in terms of sky, ground and
runway lights, and tower views are fixed.
There are four minor problems remaining. 1) The sun/moon spins when you pan
the "lookat" tower view only (view #3). 2) Under stress (esp. magic carpet
full speed with max visibility), there is a memory leak in the tile caching
that was not introduced with these changes. 3) I have not tested these
changes or made corrections to the ADA or External FDM interfaces. 4) The
change view function doesn't call the time/light update (not a problem unless
a tower is very far away).
Details:
FDM/flight.cxx, flight.hxx - FGInterface ties to FGAircraftModel so that it's
location data can be accessed for runway (ground elevation under aircraft)
elevation.
FDM/larsim.cxx, larcsim.hxx - gets runway elevation from FGInterface now.
Commented out function that is causing a namespace conflict, hasn't been
called with recent code anyway.
FDM/JSBSim/JSBSim.cxx, YASim/YASim.cxx - gets runway elevation from
FGInterface now.
Scenery/newcache.cxx, newcache.hxx - changed caching scheme to time based
(oldest tiles discard).
Scenery/tileentry.cxx, tileentry.hxx - added place to record time, changed
rendering to reference viewer altitude in order to fix a problem with ground
and runway lights.
Scenery/tilemgr.cxx, tilemgr.hxx - Modified update() to accept values for
multiple locations. Refresh function added in order to periodically make
the tiles current for a non-moving view (like a tower).
Main/fg_init.cxx - register event for making tiles current in a non-moving
view (like a tower).
Main/location.hxx - added support for current ground elevation data.
Main/main.cxx - added second tilemgr call for fdm, fixed places where viewer
position data was required for correct sky rendering.
Main/options.cxx - fixed segfault reported by Curtis when using --view-offset
command line parameter.
Main/viewer.cxx, viewer.hxx - removed fudging of view position. Fixed numerous
bugs that were causing eye and target values to get mixed up.
2002-05-17 17:25:28 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the flight model subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2002-02-05 20:54:08 +00:00
|
|
|
fgInitFDM();
|
2002-10-04 13:20:53 +00:00
|
|
|
|
2000-07-23 21:32:59 +00:00
|
|
|
// allocates structures so must happen before any of the flight
|
|
|
|
// model or control parameters are set
|
|
|
|
fgAircraftInit(); // In the future this might not be the case.
|
|
|
|
|
1998-08-22 14:49:55 +00:00
|
|
|
|
2004-01-31 19:47:45 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the XML Autopilot subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
globals->add_subsystem( "xml-autopilot", new FGXMLAutopilot );
|
|
|
|
globals->add_subsystem( "route-manager", new FGRouteMgr );
|
|
|
|
|
2004-04-15 22:09:46 +00:00
|
|
|
|
2002-04-13 13:11:34 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the view manager subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
fgInitView();
|
|
|
|
|
2002-03-12 16:29:00 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
2002-11-07 16:27:47 +00:00
|
|
|
// Create and register the logger.
|
2002-03-12 16:29:00 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2003-01-16 16:01:26 +00:00
|
|
|
globals->add_subsystem("logger", new FGLogger);
|
2002-03-12 16:29:00 +00:00
|
|
|
|
2002-11-07 16:27:47 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Create and register the XML GUI.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2003-09-24 17:20:55 +00:00
|
|
|
globals->add_subsystem("gui", new NewGUI, SGSubsystemMgr::INIT);
|
2002-11-07 16:27:47 +00:00
|
|
|
|
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the local time subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
1999-10-22 00:27:49 +00:00
|
|
|
// update the current timezone each 30 minutes
|
2003-11-25 21:08:36 +00:00
|
|
|
globals->get_event_mgr()->addTask( "fgUpdateLocalTime()",
|
|
|
|
&fgUpdateLocalTime, 30*60 );
|
2005-05-04 21:28:42 +00:00
|
|
|
fgInitTimeOffset(); // the environment subsystem needs this
|
1998-04-22 13:25:39 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the weather subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
1998-04-22 13:25:39 +00:00
|
|
|
// Initialize the weather modeling subsystem
|
2003-06-08 14:47:03 +00:00
|
|
|
globals->add_subsystem("environment", new FGEnvironmentMgr);
|
1997-08-23 01:46:20 +00:00
|
|
|
|
2003-09-24 17:20:55 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the lighting subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
globals->add_subsystem("lighting", new FGLight);
|
|
|
|
|
2004-05-20 13:27:40 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the 2D cloud subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
2004-10-20 08:18:29 +00:00
|
|
|
fgGetBool("/sim/rendering/bump-mapping", false);
|
2003-09-24 17:20:55 +00:00
|
|
|
|
2004-04-24 19:28:12 +00:00
|
|
|
#ifdef ENABLE_AUDIO_SUPPORT
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the sound subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2004-12-16 13:19:01 +00:00
|
|
|
init_volume = fgGetFloat("/sim/sound/volume");
|
|
|
|
fgSetFloat("/sim/sound/volume", 0.0f);
|
2004-04-24 19:28:12 +00:00
|
|
|
globals->set_soundmgr(new SGSoundMgr);
|
|
|
|
globals->get_soundmgr()->init();
|
|
|
|
globals->get_soundmgr()->bind();
|
|
|
|
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the sound-effects subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
globals->add_subsystem("fx", new FGFX);
|
2006-02-11 12:02:40 +00:00
|
|
|
globals->add_subsystem("voice", new FGVoiceMgr);
|
2004-04-24 19:28:12 +00:00
|
|
|
#endif
|
|
|
|
|
2001-11-07 17:55:28 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialise ATC display system
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, " ATC Display");
|
2002-03-01 17:39:52 +00:00
|
|
|
globals->set_ATC_display(new FGATCDisplay);
|
|
|
|
globals->get_ATC_display()->init();
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialise the ATC Manager
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, " ATC Manager");
|
|
|
|
globals->set_ATC_mgr(new FGATCMgr);
|
2002-04-03 23:54:44 +00:00
|
|
|
globals->get_ATC_mgr()->init();
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialise the AI Manager
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2003-04-01 13:42:10 +00:00
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, " AI Manager");
|
|
|
|
globals->set_AI_mgr(new FGAIMgr);
|
|
|
|
globals->get_AI_mgr()->init();
|
2001-04-05 20:20:44 +00:00
|
|
|
|
2003-11-28 15:48:05 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialise the AI Model Manager
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
SG_LOG(SG_GENERAL, SG_INFO, " AI Model Manager");
|
|
|
|
globals->add_subsystem("ai_model", new FGAIManager);
|
2004-10-22 09:58:24 +00:00
|
|
|
globals->add_subsystem("submodel_mgr", new FGSubmodelMgr);
|
2003-11-28 15:48:05 +00:00
|
|
|
|
|
|
|
|
2004-06-03 17:59:14 +00:00
|
|
|
// It's probably a good idea to initialize the top level traffic manager
|
|
|
|
// After the AI and ATC systems have been initialized properly.
|
|
|
|
// AI Traffic manager
|
|
|
|
globals->add_subsystem("Traffic Manager", new FGTrafficManager);
|
|
|
|
FGTrafficManager *dispatcher =
|
|
|
|
(FGTrafficManager*) globals->get_subsystem("Traffic Manager");
|
2005-02-10 09:01:51 +00:00
|
|
|
SGPath path = globals->get_fg_root();
|
|
|
|
path.append("/Traffic/fgtraffic.xml");
|
2004-12-27 17:35:22 +00:00
|
|
|
readXML(path.str(),
|
2005-02-10 09:01:51 +00:00
|
|
|
*dispatcher);
|
|
|
|
//globals->get_subsystem("Traffic Manager")->init();
|
2004-06-03 17:59:14 +00:00
|
|
|
|
2003-01-16 16:01:26 +00:00
|
|
|
globals->add_subsystem("instrumentation", new FGInstrumentMgr);
|
|
|
|
globals->add_subsystem("systems", new FGSystemMgr);
|
2002-09-24 14:51:37 +00:00
|
|
|
|
2004-06-03 17:59:14 +00:00
|
|
|
|
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the cockpit subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
1998-02-12 21:58:27 +00:00
|
|
|
if( fgCockpitInit( ¤t_aircraft )) {
|
2002-10-04 13:20:53 +00:00
|
|
|
// Cockpit initialized ok.
|
1998-02-12 21:58:27 +00:00
|
|
|
} else {
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_ALERT, "Error in Cockpit initialization!" );
|
|
|
|
exit(-1);
|
1997-09-04 02:17:18 +00:00
|
|
|
}
|
1997-08-25 20:27:21 +00:00
|
|
|
|
1998-02-12 21:58:27 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the autopilot subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2002-10-04 13:20:53 +00:00
|
|
|
// FIXME: these should go in the
|
|
|
|
// GUI initialization code, not here.
|
2004-01-31 19:47:45 +00:00
|
|
|
// fgAPAdjustInit();
|
2000-04-30 06:51:49 +00:00
|
|
|
NewTgtAirportInit();
|
|
|
|
NewHeadingInit();
|
|
|
|
NewAltitudeInit();
|
1998-11-16 13:59:58 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize I/O subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2002-08-25 20:56:16 +00:00
|
|
|
globals->get_io()->init();
|
|
|
|
globals->get_io()->bind();
|
1998-11-16 13:59:58 +00:00
|
|
|
|
2003-03-30 12:46:08 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Add a new 2D panel.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-01-13 22:06:39 +00:00
|
|
|
string panel_path = fgGetString("/sim/panel/path",
|
2002-10-04 13:20:53 +00:00
|
|
|
"Panels/Default/default.xml");
|
2003-03-30 12:46:08 +00:00
|
|
|
|
|
|
|
globals->set_current_panel( fgReadPanel(panel_path) );
|
|
|
|
if (globals->get_current_panel() == 0) {
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG( SG_INPUT, SG_ALERT,
|
|
|
|
"Error reading new panel from " << panel_path );
|
2000-10-25 19:27:13 +00:00
|
|
|
} else {
|
2002-10-04 13:20:53 +00:00
|
|
|
SG_LOG( SG_INPUT, SG_INFO, "Loaded new panel from " << panel_path );
|
2003-03-30 12:46:08 +00:00
|
|
|
globals->get_current_panel()->init();
|
|
|
|
globals->get_current_panel()->bind();
|
2000-09-22 17:20:56 +00:00
|
|
|
}
|
User-visible
- knobs now continue to rotate when you hold down the mouse
- the middle mouse button makes knobs rotate much faster
- there are NAV1, NAV2, and ADF radios that can be tuned using the mouse
- there are standby frequencies for NAV1 and NAV2, and buttons to swap
- there is a crude, rather silly-looking DME, hard-wired to NAV1
- there is a crude, rather silly-looking autopilot that can lock
the heading (to the bug on the gyro), can lock to NAV1, and can lock
the current altitude
- the knobs for changing the radials on NAV1 and NAV2 look much better
and are in the right place
- tuning into an ILS frequency doesn't change the displayed radial for
NAV1
Code
- I've created a new module, sp_panel.[ch]xx, that constructs the
default single-prop panel; this works entirely outside of FGPanel,
so it is possible to construct similar modules for other sorts of
panels; all code specific to the default panel has been removed from
panel.cxx
- current_panel is now a pointer
- radiostack.[ch]xx keeps track both of the actual radial and of the
selected radial (they will differ with ILS); the NAV gauges should
not spin around automatically to show the actual radial (we need to
do something similar with the autopilot)
- the panel is initialized fairly early
- make sure that standby frequencies also get initialized
- I've started combining and clipping small textures to save texture
memory; there's a lot more to do, but at least I've made a start
2000-05-02 18:26:00 +00:00
|
|
|
|
2003-03-20 09:38:06 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the controls subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-07-22 19:51:16 +00:00
|
|
|
globals->get_controls()->init();
|
|
|
|
globals->get_controls()->bind();
|
2001-01-05 17:38:58 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
|
I'm attaching diffs to add a new FGInput module to FlightGear
(src/Input). So far, FGInput replaces most of src/Main/keyboard.cxx
(I've left a tiny stub); in the very near future, it will also take
over control of the joystick, mouse (Norm permitting), and panel
instrument interactions, so that there is a single mechanism for
configuring all input devices.
The new format should be (close to) self-explanatory by looking at the
new base-package file keyboard.xml, which is now included by
preferences.xml (I'll do the same thing for the joystick when I have a
chance). I have not managed to move all keybindings into this file
yet, but I've made a good start. I'm including Tony in the recipient
list so that he can see how bindings can use an external XML file.
This patch also adds support for multiple bindings for a single key,
special keys (i.e. keypad and function keys), and key modifiers
(shift/alt/ctrl); special keys use the PUI convention of adding 256 to
the Glut key code.
Unfortunately, everything comes with a price; in this case, I have not
yet found a general mechanism for the old (hard-coded) modal bindings,
which behaved differently depending on the autopilot state (i.e. left
rudder or move AP heading left); with my patches, this functionality
disappears, but you can still adjust the autopilot using the panel or
the GUI input dialogs.
2001-05-23 23:01:15 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the input subsystem.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2003-01-16 16:01:26 +00:00
|
|
|
globals->add_subsystem("input", new FGInput);
|
I'm attaching diffs to add a new FGInput module to FlightGear
(src/Input). So far, FGInput replaces most of src/Main/keyboard.cxx
(I've left a tiny stub); in the very near future, it will also take
over control of the joystick, mouse (Norm permitting), and panel
instrument interactions, so that there is a single mechanism for
configuring all input devices.
The new format should be (close to) self-explanatory by looking at the
new base-package file keyboard.xml, which is now included by
preferences.xml (I'll do the same thing for the joystick when I have a
chance). I have not managed to move all keybindings into this file
yet, but I've made a good start. I'm including Tony in the recipient
list so that he can see how bindings can use an external XML file.
This patch also adds support for multiple bindings for a single key,
special keys (i.e. keypad and function keys), and key modifiers
(shift/alt/ctrl); special keys use the PUI convention of adding 256 to
the Glut key code.
Unfortunately, everything comes with a price; in this case, I have not
yet found a general mechanism for the old (hard-coded) modal bindings,
which behaved differently depending on the autopilot state (i.e. left
rudder or move AP heading left); with my patches, this functionality
disappears, but you can still adjust the autopilot using the panel or
the GUI input dialogs.
2001-05-23 23:01:15 +00:00
|
|
|
|
|
|
|
|
2003-07-17 18:24:17 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the replay subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
globals->add_subsystem("replay", new FGReplay);
|
|
|
|
|
2002-10-03 21:20:56 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Bind and initialize subsystems.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
globals->get_subsystem_mgr()->bind();
|
|
|
|
globals->get_subsystem_mgr()->init();
|
|
|
|
|
2003-03-19 20:45:09 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize multiplayer subsystem
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2005-10-30 18:01:51 +00:00
|
|
|
globals->set_multiplayer_mgr(new FGMultiplayMgr);
|
|
|
|
globals->get_multiplayer_mgr()->init();
|
2003-03-19 20:45:09 +00:00
|
|
|
|
2003-11-25 21:08:36 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// Initialize the Nasal interpreter.
|
|
|
|
// Do this last, so that the loaded scripts see initialized state
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
FGNasalSys* nasal = new FGNasalSys();
|
|
|
|
globals->add_subsystem("nasal", nasal);
|
|
|
|
nasal->init();
|
|
|
|
|
2004-04-15 22:09:46 +00:00
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
// At this point we could try and parse the waypoint options
|
|
|
|
///////////////////////////////////////////////////////////////////
|
|
|
|
parseWaypoints();
|
|
|
|
|
2005-06-11 09:13:44 +00:00
|
|
|
// initialize methods that depend on other subsystems.
|
|
|
|
globals->get_subsystem_mgr()->postinit();
|
2004-04-15 22:09:46 +00:00
|
|
|
|
2001-04-05 20:20:44 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
// End of subsystem initialization.
|
|
|
|
////////////////////////////////////////////////////////////////////
|
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO, endl);
|
1998-04-24 00:49:17 +00:00
|
|
|
|
2002-10-04 13:20:53 +00:00
|
|
|
// Save the initial state for future
|
|
|
|
// reference.
|
2001-01-19 22:57:24 +00:00
|
|
|
globals->saveInitialState();
|
2005-12-29 13:58:21 +00:00
|
|
|
|
1999-06-18 03:42:54 +00:00
|
|
|
return true;
|
1997-08-23 01:46:20 +00:00
|
|
|
}
|
1999-04-27 15:52:32 +00:00
|
|
|
|
|
|
|
|
2002-11-18 21:31:33 +00:00
|
|
|
void fgReInitSubsystems()
|
1999-04-27 15:52:32 +00:00
|
|
|
{
|
2003-05-06 23:46:24 +00:00
|
|
|
// static const SGPropertyNode *longitude
|
|
|
|
// = fgGetNode("/sim/presets/longitude-deg");
|
|
|
|
// static const SGPropertyNode *latitude
|
|
|
|
// = fgGetNode("/sim/presets/latitude-deg");
|
2001-10-28 16:16:30 +00:00
|
|
|
static const SGPropertyNode *altitude
|
2002-11-15 21:13:29 +00:00
|
|
|
= fgGetNode("/sim/presets/altitude-ft");
|
2002-01-20 03:52:36 +00:00
|
|
|
static const SGPropertyNode *master_freeze
|
2002-10-04 13:20:53 +00:00
|
|
|
= fgGetNode("/sim/freeze/master");
|
2001-10-28 16:16:30 +00:00
|
|
|
|
2001-03-24 06:03:11 +00:00
|
|
|
SG_LOG( SG_GENERAL, SG_INFO,
|
2002-10-04 13:20:53 +00:00
|
|
|
"fgReInitSubsystems(): /position/altitude = "
|
2002-07-02 20:45:02 +00:00
|
|
|
<< altitude->getDoubleValue() );
|
2001-01-17 23:30:35 +00:00
|
|
|
|
2002-01-20 03:52:36 +00:00
|
|
|
bool freeze = master_freeze->getBoolValue();
|
|
|
|
if ( !freeze ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
fgSetBool("/sim/freeze/master", true);
|
2001-10-28 16:16:30 +00:00
|
|
|
}
|
2003-12-29 10:10:35 +00:00
|
|
|
fgSetBool("/sim/crashed", false);
|
2000-07-23 21:32:59 +00:00
|
|
|
|
2005-07-13 12:25:16 +00:00
|
|
|
// Force reupdating the positions of the ai 3d models. They are used for
|
|
|
|
// initializing ground level for the FDM.
|
|
|
|
globals->get_subsystem("ai_model")->reinit();
|
|
|
|
|
2002-11-18 21:31:33 +00:00
|
|
|
// Initialize the FDM
|
2002-02-05 20:54:08 +00:00
|
|
|
fgInitFDM();
|
2002-02-17 21:05:27 +00:00
|
|
|
|
|
|
|
// allocates structures so must happen before any of the flight
|
|
|
|
// model or control parameters are set
|
|
|
|
fgAircraftInit(); // In the future this might not be the case.
|
2000-10-27 22:00:43 +00:00
|
|
|
|
2003-02-06 19:44:32 +00:00
|
|
|
// reload offsets from config defaults
|
|
|
|
globals->get_viewmgr()->reinit();
|
2002-04-21 03:27:34 +00:00
|
|
|
|
2002-02-05 20:54:08 +00:00
|
|
|
fgInitView();
|
1999-04-27 15:52:32 +00:00
|
|
|
|
2001-07-22 19:51:16 +00:00
|
|
|
globals->get_controls()->reset_all();
|
1999-05-12 02:07:21 +00:00
|
|
|
|
2001-04-05 21:14:37 +00:00
|
|
|
fgUpdateLocalTime();
|
|
|
|
|
2003-10-23 17:00:16 +00:00
|
|
|
// re-init to proper time of day setting
|
|
|
|
fgInitTimeOffset();
|
|
|
|
|
2002-01-20 03:52:36 +00:00
|
|
|
if ( !freeze ) {
|
2002-10-04 13:20:53 +00:00
|
|
|
fgSetBool("/sim/freeze/master", false);
|
2002-01-20 03:52:36 +00:00
|
|
|
}
|
2004-07-22 16:42:14 +00:00
|
|
|
fgSetBool("/sim/sceneryloaded",false);
|
1999-04-27 15:52:32 +00:00
|
|
|
}
|
2002-04-13 13:11:34 +00:00
|
|
|
|