26193c5b19
Other minor tweaks.
1861 lines
55 KiB
C++
1861 lines
55 KiB
C++
// autopilot.cxx -- autopilot subsystem
|
|
//
|
|
// Written by Jeff Goeke-Smith, started April 1998.
|
|
//
|
|
// Copyright (C) 1998 Jeff Goeke-Smith, jgoeke@voyager.net
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU General Public License as
|
|
// published by the Free Software Foundation; either version 2 of the
|
|
// License, or (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful, but
|
|
// WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
// General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, write to the Free Software
|
|
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
//
|
|
// $Id$
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <Scenery/scenery.hxx>
|
|
|
|
#include "autopilot.hxx"
|
|
|
|
#include <Include/fg_constants.h>
|
|
#include <Debug/logstream.hxx>
|
|
#include <Airports/simple.hxx>
|
|
#include <GUI/gui.h>
|
|
#include <Main/fg_init.hxx>
|
|
#include <Main/options.hxx>
|
|
#include <Time/fg_time.hxx>
|
|
|
|
|
|
#define mySlider puSlider
|
|
|
|
/// These statics will eventually go into the class
|
|
/// they are just here while I am experimenting -- NHV :-)
|
|
// AutoPilot Gain Adjuster members
|
|
static double MaxRollAdjust; // MaxRollAdjust = 2 * APData->MaxRoll;
|
|
static double RollOutAdjust; // RollOutAdjust = 2 * APData->RollOut;
|
|
static double MaxAileronAdjust; // MaxAileronAdjust = 2 * APData->MaxAileron;
|
|
static double RollOutSmoothAdjust; // RollOutSmoothAdjust = 2 * APData->RollOutSmooth;
|
|
|
|
static float MaxRollValue; // 0.1 -> 1.0
|
|
static float RollOutValue;
|
|
static float MaxAileronValue;
|
|
static float RollOutSmoothValue;
|
|
|
|
static float TmpMaxRollValue; // for cancel operation
|
|
static float TmpRollOutValue;
|
|
static float TmpMaxAileronValue;
|
|
static float TmpRollOutSmoothValue;
|
|
|
|
static puDialogBox *APAdjustDialog;
|
|
static puFrame *APAdjustFrame;
|
|
static puText *APAdjustDialogMessage;
|
|
static puFont APAdjustLegendFont;
|
|
static puFont APAdjustLabelFont;
|
|
|
|
static puOneShot *APAdjustOkButton;
|
|
static puOneShot *APAdjustResetButton;
|
|
static puOneShot *APAdjustCancelButton;
|
|
|
|
//static puButton *APAdjustDragButton;
|
|
|
|
static puText *APAdjustMaxRollTitle;
|
|
static puText *APAdjustRollOutTitle;
|
|
static puText *APAdjustMaxAileronTitle;
|
|
static puText *APAdjustRollOutSmoothTitle;
|
|
|
|
static puText *APAdjustMaxAileronText;
|
|
static puText *APAdjustMaxRollText;
|
|
static puText *APAdjustRollOutText;
|
|
static puText *APAdjustRollOutSmoothText;
|
|
|
|
static mySlider *APAdjustHS0;
|
|
static mySlider *APAdjustHS1;
|
|
static mySlider *APAdjustHS2;
|
|
static mySlider *APAdjustHS3;
|
|
|
|
static char SliderText[ 4 ][ 8 ];
|
|
|
|
///////// AutoPilot New Heading Dialog
|
|
|
|
static puDialogBox *ApHeadingDialog;
|
|
static puFrame *ApHeadingDialogFrame;
|
|
static puText *ApHeadingDialogMessage;
|
|
static puInput *ApHeadingDialogInput;
|
|
static puOneShot *ApHeadingDialogOkButton;
|
|
static puOneShot *ApHeadingDialogCancelButton;
|
|
|
|
|
|
///////// AutoPilot New Altitude Dialog
|
|
|
|
static puDialogBox *ApAltitudeDialog = 0;
|
|
static puFrame *ApAltitudeDialogFrame = 0;
|
|
static puText *ApAltitudeDialogMessage = 0;
|
|
static puInput *ApAltitudeDialogInput = 0;
|
|
|
|
static puOneShot *ApAltitudeDialogOkButton = 0;
|
|
static puOneShot *ApAltitudeDialogCancelButton = 0;
|
|
|
|
|
|
/// The beginnings of Lock AutoPilot to target location :-)
|
|
// Needs cleaning up but works
|
|
// These statics should disapear when this is a class
|
|
static puDialogBox *TgtAptDialog = 0;
|
|
static puFrame *TgtAptDialogFrame = 0;
|
|
static puText *TgtAptDialogMessage = 0;
|
|
static puInput *TgtAptDialogInput = 0;
|
|
|
|
static char NewTgtAirportId[16];
|
|
static char NewTgtAirportLabel[] = "Enter New TgtAirport ID";
|
|
|
|
static puOneShot *TgtAptDialogOkButton = 0;
|
|
static puOneShot *TgtAptDialogCancelButton = 0;
|
|
static puOneShot *TgtAptDialogResetButton = 0;
|
|
|
|
|
|
// global variable holding the AP info
|
|
// I want this gone. Data should be in aircraft structure
|
|
fgAPDataPtr APDataGlobal;
|
|
|
|
// Local Prototype section
|
|
static double LinearExtrapolate( double x, double x1, double y1, double x2, double y2 );
|
|
static double NormalizeDegrees( double Input );
|
|
// End Local ProtoTypes
|
|
|
|
extern char *coord_format_lat(float);
|
|
extern char *coord_format_lon(float);
|
|
|
|
|
|
// The below routines were copied right from hud.c ( I hate reinventing
|
|
// the wheel more than necessary)
|
|
|
|
// The following routines obtain information concerntin the aircraft's
|
|
// current state and return it to calling instrument display routines.
|
|
// They should eventually be member functions of the aircraft.
|
|
//
|
|
static void get_control_values( void ) {
|
|
fgAPDataPtr APData;
|
|
APData = APDataGlobal;
|
|
APData->old_aileron = controls.get_aileron();
|
|
APData->old_elevator = controls.get_elevator();
|
|
APData->old_elevator_trim = controls.get_elevator_trim();
|
|
APData->old_rudder = controls.get_rudder();
|
|
}
|
|
|
|
static void MakeTargetHeadingStr( fgAPDataPtr APData, double bearing ) {
|
|
if( bearing < 0. )
|
|
bearing += 360.;
|
|
else if(bearing > 360. )
|
|
bearing -= 360.;
|
|
sprintf(APData->TargetHeadingStr, "APHeading %6.1f", bearing);
|
|
}
|
|
|
|
static inline void MakeTargetDistanceStr( fgAPDataPtr APData, double distance ) {
|
|
sprintf(APData->TargetDistanceStr, "APDistance %.2f NM", distance*METER_TO_NM);
|
|
}
|
|
|
|
static inline void MakeTargetAltitudeStr( fgAPDataPtr APData, double altitude ) {
|
|
sprintf(APData->TargetAltitudeStr, "APAltitude %6.0f", altitude);
|
|
}
|
|
|
|
static inline void MakeTargetLatLonStr( fgAPDataPtr APData, double lat, double lon ) {
|
|
float tmp;
|
|
tmp = APData->TargetLatitude;
|
|
sprintf( APData->TargetLatitudeStr , "%s", coord_format_lat(tmp) );
|
|
tmp = APData->TargetLongitude;
|
|
sprintf( APData->TargetLongitudeStr, "%s", coord_format_lon(tmp) );
|
|
|
|
sprintf(APData->TargetLatLonStr, "%s %s",
|
|
APData->TargetLatitudeStr,
|
|
APData->TargetLongitudeStr );
|
|
}
|
|
|
|
static inline double get_speed( void ) {
|
|
return( current_aircraft.fdm_state->get_V_equiv_kts() );
|
|
}
|
|
|
|
static inline double get_aoa( void ) {
|
|
return( current_aircraft.fdm_state->get_Gamma_vert_rad() * RAD_TO_DEG );
|
|
}
|
|
|
|
static inline double fgAPget_latitude( void ) {
|
|
return( current_aircraft.fdm_state->get_Latitude() * RAD_TO_DEG );
|
|
}
|
|
|
|
static inline double fgAPget_longitude( void ) {
|
|
return( current_aircraft.fdm_state->get_Longitude() * RAD_TO_DEG );
|
|
}
|
|
|
|
static inline double fgAPget_roll( void ) {
|
|
return( current_aircraft.fdm_state->get_Phi() * RAD_TO_DEG );
|
|
}
|
|
|
|
static inline double get_pitch( void ) {
|
|
return( current_aircraft.fdm_state->get_Theta() );
|
|
}
|
|
|
|
double fgAPget_heading( void ) {
|
|
return( current_aircraft.fdm_state->get_Psi() * RAD_TO_DEG );
|
|
}
|
|
|
|
static inline double fgAPget_altitude( void ) {
|
|
return( current_aircraft.fdm_state->get_Altitude() * FEET_TO_METER );
|
|
}
|
|
|
|
static inline double fgAPget_climb( void ) {
|
|
// return in meters per minute
|
|
return( current_aircraft.fdm_state->get_Climb_Rate() * FEET_TO_METER * 60 );
|
|
}
|
|
|
|
static inline double get_sideslip( void ) {
|
|
return( current_aircraft.fdm_state->get_Beta() );
|
|
}
|
|
|
|
static inline double fgAPget_agl( void ) {
|
|
double agl;
|
|
|
|
agl = current_aircraft.fdm_state->get_Altitude() * FEET_TO_METER
|
|
- scenery.cur_elev;
|
|
|
|
return( agl );
|
|
}
|
|
|
|
// End of copied section. ( thanks for the wheel :-)
|
|
double fgAPget_TargetLatitude( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetLatitude;
|
|
}
|
|
|
|
double fgAPget_TargetLongitude( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetLongitude;
|
|
}
|
|
|
|
double fgAPget_TargetHeading( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetHeading;
|
|
}
|
|
|
|
double fgAPget_TargetDistance( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetDistance;
|
|
}
|
|
|
|
double fgAPget_TargetAltitude( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetAltitude;
|
|
}
|
|
|
|
char *fgAPget_TargetLatitudeStr( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetLatitudeStr;
|
|
}
|
|
|
|
char *fgAPget_TargetLongitudeStr( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetLongitudeStr;
|
|
}
|
|
|
|
char *fgAPget_TargetDistanceStr( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetDistanceStr;
|
|
}
|
|
|
|
char *fgAPget_TargetHeadingStr( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetHeadingStr;
|
|
}
|
|
|
|
char *fgAPget_TargetAltitudeStr( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetAltitudeStr;
|
|
}
|
|
|
|
char *fgAPget_TargetLatLonStr( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->TargetLatLonStr;
|
|
}
|
|
|
|
bool fgAPWayPointEnabled( void )
|
|
{
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
|
|
// heading hold enabled?
|
|
return APData->waypoint_hold;
|
|
}
|
|
|
|
|
|
bool fgAPHeadingEnabled( void )
|
|
{
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
|
|
// heading hold enabled?
|
|
return APData->heading_hold;
|
|
}
|
|
|
|
bool fgAPAltitudeEnabled( void )
|
|
{
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
|
|
// altitude hold or terrain follow enabled?
|
|
return APData->altitude_hold;
|
|
}
|
|
|
|
bool fgAPTerrainFollowEnabled( void )
|
|
{
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
|
|
// altitude hold or terrain follow enabled?
|
|
return APData->terrain_follow ;
|
|
}
|
|
|
|
bool fgAPAutoThrottleEnabled( void )
|
|
{
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
|
|
// autothrottle enabled?
|
|
return APData->auto_throttle;
|
|
}
|
|
|
|
void fgAPAltitudeAdjust( double inc )
|
|
{
|
|
// Remove at a later date
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
// end section
|
|
|
|
double target_alt, target_agl;
|
|
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_FEET ) {
|
|
target_alt = APData->TargetAltitude * METER_TO_FEET;
|
|
target_agl = APData->TargetAGL * METER_TO_FEET;
|
|
} else {
|
|
target_alt = APData->TargetAltitude;
|
|
target_agl = APData->TargetAGL;
|
|
}
|
|
|
|
target_alt = ( int ) ( target_alt / inc ) * inc + inc;
|
|
target_agl = ( int ) ( target_agl / inc ) * inc + inc;
|
|
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_FEET ) {
|
|
target_alt *= FEET_TO_METER;
|
|
target_agl *= FEET_TO_METER;
|
|
}
|
|
|
|
APData->TargetAltitude = target_alt;
|
|
APData->TargetAGL = target_agl;
|
|
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_FEET )
|
|
target_alt *= METER_TO_FEET;
|
|
ApAltitudeDialogInput->setValue((float)target_alt);
|
|
MakeTargetAltitudeStr( APData, target_alt);
|
|
|
|
get_control_values();
|
|
}
|
|
|
|
void fgAPAltitudeSet( double new_altitude ) {
|
|
// Remove at a later date
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
// end section
|
|
double target_alt = new_altitude;
|
|
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_FEET )
|
|
target_alt = new_altitude * FEET_TO_METER;
|
|
|
|
if( target_alt < scenery.cur_elev )
|
|
target_alt = scenery.cur_elev;
|
|
|
|
APData->TargetAltitude = target_alt;
|
|
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_FEET )
|
|
target_alt *= METER_TO_FEET;
|
|
ApAltitudeDialogInput->setValue((float)target_alt);
|
|
MakeTargetAltitudeStr( APData, target_alt);
|
|
|
|
get_control_values();
|
|
}
|
|
|
|
void fgAPHeadingAdjust( double inc ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
if ( APData->waypoint_hold )
|
|
APData->waypoint_hold = false;
|
|
|
|
double target = ( int ) ( APData->TargetHeading / inc ) * inc + inc;
|
|
|
|
APData->TargetHeading = NormalizeDegrees( target );
|
|
// following cast needed ambiguous plib
|
|
ApHeadingDialogInput -> setValue ((float)APData->TargetHeading );
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
get_control_values();
|
|
}
|
|
|
|
void fgAPHeadingSet( double new_heading ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
if ( APData->waypoint_hold )
|
|
APData->waypoint_hold = false;
|
|
|
|
new_heading = NormalizeDegrees( new_heading );
|
|
APData->TargetHeading = new_heading;
|
|
// following cast needed ambiguous plib
|
|
ApHeadingDialogInput -> setValue ((float)APData->TargetHeading );
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
get_control_values();
|
|
}
|
|
|
|
void fgAPAutoThrottleAdjust( double inc ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
double target = ( int ) ( APData->TargetSpeed / inc ) * inc + inc;
|
|
|
|
APData->TargetSpeed = target;
|
|
}
|
|
|
|
// THIS NEEDS IMPROVEMENT !!!!!!!!!!!!!
|
|
static int scan_number(char *s, double *new_value)
|
|
{
|
|
int ret = 0;
|
|
char WordBuf[64];
|
|
char *cptr = s;
|
|
char *WordBufPtr = WordBuf;
|
|
|
|
if (*cptr == '+')
|
|
cptr++;
|
|
if (*cptr == '-') {
|
|
*WordBufPtr++ = *cptr++;
|
|
}
|
|
while (isdigit(*cptr) ) {
|
|
*WordBufPtr++ = *cptr++;
|
|
ret = 1;
|
|
}
|
|
if (*cptr == '.')
|
|
*WordBufPtr++ = *cptr++; // put the '.' into the string
|
|
while (isdigit(*cptr)) {
|
|
*WordBufPtr++ = *cptr++;
|
|
ret = 1;
|
|
}
|
|
if( ret == 1 ) {
|
|
*WordBufPtr = '\0';
|
|
sscanf(WordBuf, "%lf", new_value);
|
|
}
|
|
|
|
return(ret);
|
|
} // scan_number
|
|
|
|
|
|
void ApHeadingDialog_Cancel(puObject *)
|
|
{
|
|
ApHeadingDialogInput->rejectInput();
|
|
FG_POP_PUI_DIALOG( ApHeadingDialog );
|
|
}
|
|
|
|
void ApHeadingDialog_OK (puObject *me)
|
|
{
|
|
int error = 0;
|
|
char *c;
|
|
string s;
|
|
ApHeadingDialogInput -> getValue( &c );
|
|
|
|
if( strlen(c) ) {
|
|
double NewHeading;
|
|
if( scan_number( c, &NewHeading ) )
|
|
{
|
|
if(!fgAPHeadingEnabled())
|
|
fgAPToggleHeading();
|
|
fgAPHeadingSet( NewHeading );
|
|
} else {
|
|
error = 1;
|
|
s = c;
|
|
s += " is not a valid number.";
|
|
}
|
|
}
|
|
ApHeadingDialog_Cancel(me);
|
|
if( error ) mkDialog(s.c_str());
|
|
}
|
|
|
|
void NewHeading(puObject *cb)
|
|
{
|
|
// string ApHeadingLabel( "Enter New Heading" );
|
|
// ApHeadingDialogMessage -> setLabel(ApHeadingLabel.c_str());
|
|
ApHeadingDialogInput -> acceptInput();
|
|
FG_PUSH_PUI_DIALOG( ApHeadingDialog );
|
|
}
|
|
|
|
void NewHeadingInit(void)
|
|
{
|
|
// printf("NewHeadingInit\n");
|
|
char NewHeadingLabel[] = "Enter New Heading";
|
|
char *s;
|
|
|
|
float heading = fgAPget_heading();
|
|
int len = 260/2 -
|
|
(puGetStringWidth( puGetDefaultLabelFont(), NewHeadingLabel ) /2 );
|
|
|
|
ApHeadingDialog = new puDialogBox (150, 50);
|
|
{
|
|
ApHeadingDialogFrame = new puFrame (0, 0, 260, 150);
|
|
|
|
ApHeadingDialogMessage = new puText (len, 110);
|
|
ApHeadingDialogMessage -> setDefaultValue (NewHeadingLabel);
|
|
ApHeadingDialogMessage -> getDefaultValue (&s);
|
|
ApHeadingDialogMessage -> setLabel (s);
|
|
|
|
ApHeadingDialogInput = new puInput ( 50, 70, 210, 100 );
|
|
ApHeadingDialogInput -> setValue ( heading );
|
|
|
|
ApHeadingDialogOkButton = new puOneShot (50, 10, 110, 50);
|
|
ApHeadingDialogOkButton -> setLegend (gui_msg_OK);
|
|
ApHeadingDialogOkButton -> makeReturnDefault (TRUE);
|
|
ApHeadingDialogOkButton -> setCallback (ApHeadingDialog_OK);
|
|
|
|
ApHeadingDialogCancelButton = new puOneShot (140, 10, 210, 50);
|
|
ApHeadingDialogCancelButton -> setLegend (gui_msg_CANCEL);
|
|
ApHeadingDialogCancelButton -> setCallback (ApHeadingDialog_Cancel);
|
|
|
|
}
|
|
FG_FINALIZE_PUI_DIALOG( ApHeadingDialog );
|
|
}
|
|
|
|
void ApAltitudeDialog_Cancel(puObject *)
|
|
{
|
|
ApAltitudeDialogInput -> rejectInput();
|
|
FG_POP_PUI_DIALOG( ApAltitudeDialog );
|
|
}
|
|
|
|
void ApAltitudeDialog_OK (puObject *me)
|
|
{
|
|
int error = 0;
|
|
string s;
|
|
char *c;
|
|
ApAltitudeDialogInput->getValue( &c );
|
|
|
|
if( strlen( c ) ) {
|
|
double NewAltitude;
|
|
if( scan_number( c, &NewAltitude) )
|
|
{
|
|
if(!(fgAPAltitudeEnabled()))
|
|
fgAPToggleAltitude();
|
|
fgAPAltitudeSet( NewAltitude );
|
|
} else {
|
|
error = 1;
|
|
s = c;
|
|
s += " is not a valid number.";
|
|
}
|
|
}
|
|
ApAltitudeDialog_Cancel(me);
|
|
if( error ) mkDialog(s.c_str());
|
|
get_control_values();
|
|
}
|
|
|
|
void NewAltitude(puObject *cb)
|
|
{
|
|
ApAltitudeDialogInput -> acceptInput();
|
|
FG_PUSH_PUI_DIALOG( ApAltitudeDialog );
|
|
}
|
|
|
|
void NewAltitudeInit(void)
|
|
{
|
|
// printf("NewAltitudeInit\n");
|
|
char NewAltitudeLabel[] = "Enter New Altitude";
|
|
char *s;
|
|
|
|
float alt = current_aircraft.fdm_state->get_Altitude();
|
|
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_METERS) {
|
|
alt *= FEET_TO_METER;
|
|
}
|
|
|
|
int len = 260/2 -
|
|
(puGetStringWidth( puGetDefaultLabelFont(), NewAltitudeLabel )/2);
|
|
|
|
// ApAltitudeDialog = new puDialogBox (150, 50);
|
|
ApAltitudeDialog = new puDialogBox (150, 200);
|
|
{
|
|
ApAltitudeDialogFrame = new puFrame (0, 0, 260, 150);
|
|
ApAltitudeDialogMessage = new puText (len, 110);
|
|
ApAltitudeDialogMessage -> setDefaultValue (NewAltitudeLabel);
|
|
ApAltitudeDialogMessage -> getDefaultValue (&s);
|
|
ApAltitudeDialogMessage -> setLabel (s);
|
|
|
|
ApAltitudeDialogInput = new puInput ( 50, 70, 210, 100 );
|
|
ApAltitudeDialogInput -> setValue ( alt );
|
|
// Uncomment the next line to have input active on startup
|
|
// ApAltitudeDialogInput -> acceptInput ( );
|
|
// cursor at begining or end of line ?
|
|
//len = strlen(s);
|
|
// len = 0;
|
|
// ApAltitudeDialogInput -> setCursor ( len );
|
|
// ApAltitudeDialogInput -> setSelectRegion ( 5, 9 );
|
|
|
|
ApAltitudeDialogOkButton = new puOneShot (50, 10, 110, 50);
|
|
ApAltitudeDialogOkButton -> setLegend (gui_msg_OK);
|
|
ApAltitudeDialogOkButton -> makeReturnDefault (TRUE);
|
|
ApAltitudeDialogOkButton -> setCallback (ApAltitudeDialog_OK);
|
|
|
|
ApAltitudeDialogCancelButton = new puOneShot (140, 10, 210, 50);
|
|
ApAltitudeDialogCancelButton -> setLegend (gui_msg_CANCEL);
|
|
ApAltitudeDialogCancelButton -> setCallback (ApAltitudeDialog_Cancel);
|
|
|
|
}
|
|
FG_FINALIZE_PUI_DIALOG( ApAltitudeDialog );
|
|
}
|
|
|
|
/////// simple AutoPilot GAIN / LIMITS ADJUSTER
|
|
|
|
#define fgAP_CLAMP(val,min,max) \
|
|
( (val) = (val) > (max) ? (max) : (val) < (min) ? (min) : (val) )
|
|
|
|
static void maxroll_adj( puObject *hs ) {
|
|
float val ;
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
hs-> getValue ( &val ) ;
|
|
fgAP_CLAMP ( val, 0.1, 1.0 ) ;
|
|
// printf ( "maxroll_adj( %p ) %f %f\n", hs, val, MaxRollAdjust * val ) ;
|
|
APData-> MaxRoll = MaxRollAdjust * val;
|
|
sprintf( SliderText[ 0 ], "%05.2f", APData->MaxRoll );
|
|
APAdjustMaxRollText -> setLabel ( SliderText[ 0 ] ) ;
|
|
}
|
|
|
|
static void rollout_adj( puObject *hs ) {
|
|
float val ;
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
hs-> getValue ( &val ) ;
|
|
fgAP_CLAMP ( val, 0.1, 1.0 ) ;
|
|
// printf ( "rollout_adj( %p ) %f %f\n", hs, val, RollOutAdjust * val ) ;
|
|
APData-> RollOut = RollOutAdjust * val;
|
|
sprintf( SliderText[ 1 ], "%05.2f", APData->RollOut );
|
|
APAdjustRollOutText -> setLabel ( SliderText[ 1 ] );
|
|
}
|
|
|
|
static void maxaileron_adj( puObject *hs ) {
|
|
float val ;
|
|
fgAPDataPtr APData;
|
|
APData = APDataGlobal;
|
|
|
|
hs-> getValue ( &val ) ;
|
|
fgAP_CLAMP ( val, 0.1, 1.0 ) ;
|
|
// printf ( "maxaileron_adj( %p ) %f %f\n", hs, val, MaxAileronAdjust * val ) ;
|
|
APData-> MaxAileron = MaxAileronAdjust * val;
|
|
sprintf( SliderText[ 3 ], "%05.2f", APData->MaxAileron );
|
|
APAdjustMaxAileronText -> setLabel ( SliderText[ 3 ] );
|
|
}
|
|
|
|
static void rolloutsmooth_adj( puObject *hs ) {
|
|
float val ;
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
hs -> getValue ( &val ) ;
|
|
fgAP_CLAMP ( val, 0.1, 1.0 ) ;
|
|
// printf ( "rolloutsmooth_adj( %p ) %f %f\n", hs, val, RollOutSmoothAdjust * val ) ;
|
|
APData->RollOutSmooth = RollOutSmoothAdjust * val;
|
|
sprintf( SliderText[ 2 ], "%5.2f", APData-> RollOutSmooth );
|
|
APAdjustRollOutSmoothText-> setLabel ( SliderText[ 2 ] );
|
|
|
|
}
|
|
|
|
static void goAwayAPAdjust (puObject *)
|
|
{
|
|
FG_POP_PUI_DIALOG( APAdjustDialog );
|
|
}
|
|
|
|
void cancelAPAdjust( puObject *self ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
APData-> MaxRoll = TmpMaxRollValue;
|
|
APData-> RollOut = TmpRollOutValue;
|
|
APData-> MaxAileron = TmpMaxAileronValue;
|
|
APData-> RollOutSmooth = TmpRollOutSmoothValue;
|
|
|
|
goAwayAPAdjust(self);
|
|
}
|
|
|
|
void resetAPAdjust( puObject *self ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
APData-> MaxRoll = MaxRollAdjust / 2;
|
|
APData-> RollOut = RollOutAdjust / 2;
|
|
APData-> MaxAileron = MaxAileronAdjust / 2;
|
|
APData-> RollOutSmooth = RollOutSmoothAdjust / 2;
|
|
|
|
FG_POP_PUI_DIALOG( APAdjustDialog );
|
|
|
|
fgAPAdjust( self );
|
|
}
|
|
|
|
void fgAPAdjust( puObject * ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
TmpMaxRollValue = APData-> MaxRoll;
|
|
TmpRollOutValue = APData-> RollOut;
|
|
TmpMaxAileronValue = APData-> MaxAileron;
|
|
TmpRollOutSmoothValue = APData-> RollOutSmooth;
|
|
|
|
MaxRollValue = APData-> MaxRoll / MaxRollAdjust;
|
|
RollOutValue = APData-> RollOut / RollOutAdjust;
|
|
MaxAileronValue = APData-> MaxAileron / MaxAileronAdjust;
|
|
RollOutSmoothValue = APData-> RollOutSmooth / RollOutSmoothAdjust;
|
|
|
|
APAdjustHS0-> setValue ( MaxRollValue ) ;
|
|
APAdjustHS1-> setValue ( RollOutValue ) ;
|
|
APAdjustHS2-> setValue ( RollOutSmoothValue ) ;
|
|
APAdjustHS3-> setValue ( MaxAileronValue ) ;
|
|
|
|
FG_PUSH_PUI_DIALOG( APAdjustDialog );
|
|
}
|
|
|
|
// Done once at system initialization
|
|
void fgAPAdjustInit( void ) {
|
|
|
|
// printf("fgAPAdjustInit\n");
|
|
#define HORIZONTAL FALSE
|
|
|
|
int DialogX = 40;
|
|
int DialogY = 100;
|
|
int DialogWidth = 230;
|
|
|
|
char Label[] = "AutoPilot Adjust";
|
|
char *s;
|
|
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
int labelX = (DialogWidth / 2) -
|
|
(puGetStringWidth( puGetDefaultLabelFont(), Label ) / 2);
|
|
labelX -= 30; // KLUDGEY
|
|
|
|
int nSliders = 4;
|
|
int slider_x = 10;
|
|
int slider_y = 55;
|
|
int slider_width = 210;
|
|
int slider_title_x = 15;
|
|
int slider_value_x = 160;
|
|
float slider_delta = 0.1f;
|
|
|
|
TmpMaxRollValue = APData-> MaxRoll;
|
|
TmpRollOutValue = APData-> RollOut;
|
|
TmpMaxAileronValue = APData-> MaxAileron;
|
|
TmpRollOutSmoothValue = APData-> RollOutSmooth;
|
|
|
|
MaxRollValue = APData-> MaxRoll / MaxRollAdjust;
|
|
RollOutValue = APData-> RollOut / RollOutAdjust;
|
|
MaxAileronValue = APData-> MaxAileron / MaxAileronAdjust;
|
|
RollOutSmoothValue = APData-> RollOutSmooth / RollOutSmoothAdjust;
|
|
|
|
puGetDefaultFonts ( &APAdjustLegendFont, &APAdjustLabelFont );
|
|
APAdjustDialog = new puDialogBox ( DialogX, DialogY ); {
|
|
int horiz_slider_height = puGetStringHeight (APAdjustLabelFont) +
|
|
puGetStringDescender (APAdjustLabelFont) +
|
|
PUSTR_TGAP + PUSTR_BGAP + 5;
|
|
|
|
APAdjustFrame = new puFrame ( 0, 0,
|
|
DialogWidth, 85 + nSliders * horiz_slider_height );
|
|
|
|
APAdjustDialogMessage = new puText ( labelX, 52 + nSliders * horiz_slider_height );
|
|
APAdjustDialogMessage -> setDefaultValue ( Label );
|
|
APAdjustDialogMessage -> getDefaultValue ( &s );
|
|
APAdjustDialogMessage -> setLabel ( s );
|
|
|
|
APAdjustHS0 = new mySlider ( slider_x, slider_y, slider_width, HORIZONTAL ) ;
|
|
APAdjustHS0-> setDelta ( slider_delta ) ;
|
|
APAdjustHS0-> setValue ( MaxRollValue ) ;
|
|
APAdjustHS0-> setCBMode ( PUSLIDER_DELTA ) ;
|
|
APAdjustHS0-> setCallback ( maxroll_adj ) ;
|
|
|
|
sprintf( SliderText[ 0 ], "%05.2f", APData->MaxRoll );
|
|
APAdjustMaxRollTitle = new puText ( slider_title_x, slider_y ) ;
|
|
APAdjustMaxRollTitle-> setDefaultValue ( "MaxRoll" ) ;
|
|
APAdjustMaxRollTitle-> getDefaultValue ( &s ) ;
|
|
APAdjustMaxRollTitle-> setLabel ( s ) ;
|
|
APAdjustMaxRollText = new puText ( slider_value_x, slider_y ) ;
|
|
APAdjustMaxRollText-> setLabel ( SliderText[ 0 ] ) ;
|
|
|
|
slider_y += horiz_slider_height;
|
|
|
|
APAdjustHS1 = new mySlider ( slider_x, slider_y, slider_width, HORIZONTAL ) ;
|
|
APAdjustHS1-> setDelta ( slider_delta ) ;
|
|
APAdjustHS1-> setValue ( RollOutValue ) ;
|
|
APAdjustHS1-> setCBMode ( PUSLIDER_DELTA ) ;
|
|
APAdjustHS1-> setCallback ( rollout_adj ) ;
|
|
|
|
sprintf( SliderText[ 1 ], "%05.2f", APData->RollOut );
|
|
APAdjustRollOutTitle = new puText ( slider_title_x, slider_y ) ;
|
|
APAdjustRollOutTitle-> setDefaultValue ( "AdjustRollOut" ) ;
|
|
APAdjustRollOutTitle-> getDefaultValue ( &s ) ;
|
|
APAdjustRollOutTitle-> setLabel ( s ) ;
|
|
APAdjustRollOutText = new puText ( slider_value_x, slider_y ) ;
|
|
APAdjustRollOutText-> setLabel ( SliderText[ 1 ] );
|
|
|
|
slider_y += horiz_slider_height;
|
|
|
|
APAdjustHS2 = new mySlider ( slider_x, slider_y, slider_width, HORIZONTAL ) ;
|
|
APAdjustHS2-> setDelta ( slider_delta ) ;
|
|
APAdjustHS2-> setValue ( RollOutSmoothValue ) ;
|
|
APAdjustHS2-> setCBMode ( PUSLIDER_DELTA ) ;
|
|
APAdjustHS2-> setCallback ( rolloutsmooth_adj ) ;
|
|
|
|
sprintf( SliderText[ 2 ], "%5.2f", APData->RollOutSmooth );
|
|
APAdjustRollOutSmoothTitle = new puText ( slider_title_x, slider_y ) ;
|
|
APAdjustRollOutSmoothTitle-> setDefaultValue ( "RollOutSmooth" ) ;
|
|
APAdjustRollOutSmoothTitle-> getDefaultValue ( &s ) ;
|
|
APAdjustRollOutSmoothTitle-> setLabel ( s ) ;
|
|
APAdjustRollOutSmoothText = new puText ( slider_value_x, slider_y ) ;
|
|
APAdjustRollOutSmoothText-> setLabel ( SliderText[ 2 ] );
|
|
|
|
slider_y += horiz_slider_height;
|
|
|
|
APAdjustHS3 = new mySlider ( slider_x, slider_y, slider_width, HORIZONTAL ) ;
|
|
APAdjustHS3-> setDelta ( slider_delta ) ;
|
|
APAdjustHS3-> setValue ( MaxAileronValue ) ;
|
|
APAdjustHS3-> setCBMode ( PUSLIDER_DELTA ) ;
|
|
APAdjustHS3-> setCallback ( maxaileron_adj ) ;
|
|
|
|
sprintf( SliderText[ 3 ], "%05.2f", APData->MaxAileron );
|
|
APAdjustMaxAileronTitle = new puText ( slider_title_x, slider_y ) ;
|
|
APAdjustMaxAileronTitle-> setDefaultValue ( "MaxAileron" ) ;
|
|
APAdjustMaxAileronTitle-> getDefaultValue ( &s ) ;
|
|
APAdjustMaxAileronTitle-> setLabel ( s ) ;
|
|
APAdjustMaxAileronText = new puText ( slider_value_x, slider_y ) ;
|
|
APAdjustMaxAileronText-> setLabel ( SliderText[ 3 ] );
|
|
|
|
APAdjustOkButton = new puOneShot ( 10, 10, 60, 50 );
|
|
APAdjustOkButton-> setLegend ( gui_msg_OK );
|
|
APAdjustOkButton-> makeReturnDefault ( TRUE );
|
|
APAdjustOkButton-> setCallback ( goAwayAPAdjust );
|
|
|
|
APAdjustCancelButton = new puOneShot ( 70, 10, 150, 50 );
|
|
APAdjustCancelButton-> setLegend ( gui_msg_CANCEL );
|
|
APAdjustCancelButton-> setCallback ( cancelAPAdjust );
|
|
|
|
APAdjustResetButton = new puOneShot ( 160, 10, 220, 50 );
|
|
APAdjustResetButton-> setLegend ( gui_msg_RESET );
|
|
APAdjustResetButton-> setCallback ( resetAPAdjust );
|
|
}
|
|
FG_FINALIZE_PUI_DIALOG( APAdjustDialog );
|
|
|
|
#undef HORIZONTAL
|
|
}
|
|
|
|
// Simple Dialog to input Target Airport
|
|
void TgtAptDialog_Cancel(puObject *)
|
|
{
|
|
FG_POP_PUI_DIALOG( TgtAptDialog );
|
|
}
|
|
|
|
void TgtAptDialog_OK (puObject *)
|
|
{
|
|
fgAPDataPtr APData;
|
|
APData = APDataGlobal;
|
|
string TgtAptId;
|
|
|
|
// FGTime *t = FGTime::cur_time_params;
|
|
// int PauseMode = t->getPause();
|
|
// if(!PauseMode)
|
|
// t->togglePauseMode();
|
|
|
|
char *s;
|
|
TgtAptDialogInput->getValue(&s);
|
|
TgtAptId = s;
|
|
|
|
TgtAptDialog_Cancel( NULL );
|
|
|
|
if ( TgtAptId.length() ) {
|
|
// set initial position from TgtAirport id
|
|
|
|
fgAIRPORTS airports;
|
|
fgAIRPORT a;
|
|
|
|
FG_LOG( FG_GENERAL, FG_INFO,
|
|
"Attempting to set starting position from airport code "
|
|
<< s );
|
|
|
|
airports.load("apt_simple");
|
|
if ( airports.search( TgtAptId, &a ) )
|
|
{
|
|
double course, reverse, distance;
|
|
// fgAPset_tgt_airport_id( TgtAptId.c_str() );
|
|
current_options.set_airport_id( TgtAptId.c_str() );
|
|
sprintf( NewTgtAirportId, "%s", TgtAptId.c_str() );
|
|
|
|
APData->TargetLatitude = a.latitude; // * DEG_TO_RAD;
|
|
APData->TargetLongitude = a.longitude; // * DEG_TO_RAD;
|
|
MakeTargetLatLonStr( APData,
|
|
APData->TargetLatitude,
|
|
APData->TargetLongitude);
|
|
|
|
APData->old_lat = fgAPget_latitude();
|
|
APData->old_lon = fgAPget_longitude();
|
|
|
|
// need to test for iter
|
|
if( ! geo_inverse_wgs_84( fgAPget_altitude(),
|
|
fgAPget_latitude(),
|
|
fgAPget_longitude(),
|
|
APData->TargetLatitude,
|
|
APData->TargetLongitude,
|
|
&course,
|
|
&reverse,
|
|
&distance ) ) {
|
|
APData->TargetHeading = course;
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
APData->TargetDistance = distance;
|
|
MakeTargetDistanceStr( APData, distance );
|
|
// This changes the AutoPilot Heading
|
|
// following cast needed
|
|
ApHeadingDialogInput->setValue ((float)APData->TargetHeading );
|
|
// Force this !
|
|
APData->waypoint_hold = true ;
|
|
APData->heading_hold = true;
|
|
}
|
|
} else {
|
|
TgtAptId += " not in database.";
|
|
mkDialog(TgtAptId.c_str());
|
|
}
|
|
}
|
|
get_control_values();
|
|
// if( PauseMode != t->getPause() )
|
|
// t->togglePauseMode();
|
|
}
|
|
|
|
void TgtAptDialog_Reset(puObject *)
|
|
{
|
|
// strncpy( NewAirportId, current_options.get_airport_id().c_str(), 16 );
|
|
sprintf( NewTgtAirportId, "%s", current_options.get_airport_id().c_str() );
|
|
TgtAptDialogInput->setValue ( NewTgtAirportId );
|
|
TgtAptDialogInput->setCursor( 0 ) ;
|
|
}
|
|
|
|
void NewTgtAirport(puObject *cb)
|
|
{
|
|
// strncpy( NewAirportId, current_options.get_airport_id().c_str(), 16 );
|
|
sprintf( NewTgtAirportId, "%s", current_options.get_airport_id().c_str() );
|
|
TgtAptDialogInput->setValue( NewTgtAirportId );
|
|
|
|
FG_PUSH_PUI_DIALOG( TgtAptDialog );
|
|
}
|
|
|
|
void NewTgtAirportInit(void)
|
|
{
|
|
FG_LOG( FG_AUTOPILOT, FG_INFO, " enter NewTgtAirportInit()" );
|
|
// fgAPset_tgt_airport_id( current_options.get_airport_id() );
|
|
sprintf( NewTgtAirportId, "%s", current_options.get_airport_id().c_str() );
|
|
FG_LOG( FG_AUTOPILOT, FG_INFO, " NewTgtAirportId " << NewTgtAirportId );
|
|
// printf(" NewTgtAirportId %s\n", NewTgtAirportId);
|
|
int len = 150 - puGetStringWidth( puGetDefaultLabelFont(),
|
|
NewTgtAirportLabel ) / 2;
|
|
|
|
TgtAptDialog = new puDialogBox (150, 50);
|
|
{
|
|
TgtAptDialogFrame = new puFrame (0,0,350, 150);
|
|
TgtAptDialogMessage = new puText (len, 110);
|
|
TgtAptDialogMessage -> setLabel (NewTgtAirportLabel);
|
|
|
|
TgtAptDialogInput = new puInput (50, 70, 300, 100);
|
|
TgtAptDialogInput -> setValue (NewTgtAirportId);
|
|
TgtAptDialogInput -> acceptInput();
|
|
|
|
TgtAptDialogOkButton = new puOneShot (50, 10, 110, 50);
|
|
TgtAptDialogOkButton -> setLegend (gui_msg_OK);
|
|
TgtAptDialogOkButton -> setCallback (TgtAptDialog_OK);
|
|
TgtAptDialogOkButton -> makeReturnDefault(TRUE);
|
|
|
|
TgtAptDialogCancelButton = new puOneShot (140, 10, 210, 50);
|
|
TgtAptDialogCancelButton -> setLegend (gui_msg_CANCEL);
|
|
TgtAptDialogCancelButton -> setCallback (TgtAptDialog_Cancel);
|
|
|
|
TgtAptDialogResetButton = new puOneShot (240, 10, 300, 50);
|
|
TgtAptDialogResetButton -> setLegend (gui_msg_RESET);
|
|
TgtAptDialogResetButton -> setCallback (TgtAptDialog_Reset);
|
|
}
|
|
FG_FINALIZE_PUI_DIALOG( TgtAptDialog );
|
|
printf("leave NewTgtAirportInit()");
|
|
}
|
|
|
|
|
|
// Finally actual guts of AutoPilot
|
|
void fgAPInit( fgAIRCRAFT *current_aircraft ) {
|
|
fgAPDataPtr APData;
|
|
|
|
FG_LOG( FG_AUTOPILOT, FG_INFO, "Init AutoPilot Subsystem" );
|
|
|
|
APData = ( fgAPDataPtr ) calloc( sizeof( fgAPData ), 1 );
|
|
|
|
if ( APData == NULL ) {
|
|
// I couldn't get the mem. Dying
|
|
FG_LOG( FG_AUTOPILOT, FG_ALERT, "No ram for Autopilot. Dying." );
|
|
exit( -1 );
|
|
}
|
|
|
|
FG_LOG( FG_AUTOPILOT, FG_INFO, " Autopilot allocated " );
|
|
|
|
APData->waypoint_hold = false ; // turn the target hold off
|
|
APData->heading_hold = false ; // turn the heading hold off
|
|
APData->altitude_hold = false ; // turn the altitude hold off
|
|
|
|
// Initialize target location to startup location
|
|
// FG_LOG( FG_AUTOPILOT, FG_INFO, " Autopilot setting startup location" );
|
|
APData->old_lat =
|
|
APData->TargetLatitude = fgAPget_latitude();
|
|
APData->old_lon =
|
|
APData->TargetLongitude = fgAPget_longitude();
|
|
|
|
// FG_LOG( FG_AUTOPILOT, FG_INFO, " Autopilot setting TargetLatitudeStr" );
|
|
MakeTargetLatLonStr( APData, APData->TargetLatitude, APData->TargetLongitude);
|
|
|
|
APData->TargetHeading = 0.0; // default direction, due north
|
|
APData->TargetAltitude = 3000; // default altitude in meters
|
|
APData->alt_error_accum = 0.0;
|
|
|
|
MakeTargetAltitudeStr( APData, 3000.0);
|
|
MakeTargetHeadingStr( APData, 0.0 );
|
|
|
|
// These eventually need to be read from current_aircaft somehow.
|
|
|
|
#if 0
|
|
// Original values
|
|
// the maximum roll, in Deg
|
|
APData->MaxRoll = 7;
|
|
// the deg from heading to start rolling out at, in Deg
|
|
APData->RollOut = 30;
|
|
// how far can I move the aleron from center.
|
|
APData->MaxAileron = .1;
|
|
// Smoothing distance for alerion control
|
|
APData->RollOutSmooth = 10;
|
|
#endif
|
|
|
|
// the maximum roll, in Deg
|
|
APData->MaxRoll = 20;
|
|
|
|
// the deg from heading to start rolling out at, in Deg
|
|
APData->RollOut = 20;
|
|
|
|
// how far can I move the aleron from center.
|
|
APData->MaxAileron = .2;
|
|
|
|
// Smoothing distance for alerion control
|
|
APData->RollOutSmooth = 10;
|
|
|
|
//Remove at a later date
|
|
APDataGlobal = APData;
|
|
|
|
// Hardwired for now should be in options
|
|
// 25% max control variablilty 0.5 / 2.0
|
|
APData->disengage_threshold = 1.0;
|
|
|
|
#if !defined( USING_SLIDER_CLASS )
|
|
MaxRollAdjust = 2 * APData->MaxRoll;
|
|
RollOutAdjust = 2 * APData->RollOut;
|
|
MaxAileronAdjust = 2 * APData->MaxAileron;
|
|
RollOutSmoothAdjust = 2 * APData->RollOutSmooth;
|
|
#endif // !defined( USING_SLIDER_CLASS )
|
|
|
|
get_control_values();
|
|
|
|
// FG_LOG( FG_AUTOPILOT, FG_INFO, " calling NewTgtAirportInit" );
|
|
NewTgtAirportInit();
|
|
fgAPAdjustInit() ;
|
|
NewHeadingInit();
|
|
NewAltitudeInit();
|
|
};
|
|
|
|
|
|
void fgAPReset( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
|
|
if ( fgAPTerrainFollowEnabled() )
|
|
fgAPToggleTerrainFollow( );
|
|
|
|
if ( fgAPAltitudeEnabled() )
|
|
fgAPToggleAltitude();
|
|
|
|
if ( fgAPHeadingEnabled() )
|
|
fgAPToggleHeading();
|
|
|
|
if ( fgAPAutoThrottleEnabled() )
|
|
fgAPToggleAutoThrottle();
|
|
|
|
APData->TargetHeading = 0.0; // default direction, due north
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
|
|
APData->TargetAltitude = 3000; // default altitude in meters
|
|
MakeTargetAltitudeStr( APData, 3000);
|
|
|
|
APData->alt_error_accum = 0.0;
|
|
|
|
|
|
get_control_values();
|
|
|
|
sprintf( NewTgtAirportId, "%s", current_options.get_airport_id().c_str() );
|
|
|
|
APData->TargetLatitude = fgAPget_latitude();
|
|
APData->TargetLongitude = fgAPget_longitude();
|
|
MakeTargetLatLonStr( APData,
|
|
APData->TargetLatitude,
|
|
APData->TargetLongitude);
|
|
}
|
|
|
|
|
|
int fgAPRun( void ) {
|
|
// Remove the following lines when the calling funcitons start
|
|
// passing in the data pointer
|
|
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
// get control settings
|
|
double aileron = controls.get_aileron();
|
|
double elevator = controls.get_elevator();
|
|
double elevator_trim = controls.get_elevator_trim();
|
|
double rudder = controls.get_rudder();
|
|
|
|
double lat = fgAPget_latitude();
|
|
double lon = fgAPget_longitude();
|
|
|
|
// see if somebody else has changed them
|
|
if( fabs(aileron - APData->old_aileron) > APData->disengage_threshold ||
|
|
fabs(elevator - APData->old_elevator) > APData->disengage_threshold ||
|
|
fabs(elevator_trim - APData->old_elevator_trim) > APData->disengage_threshold ||
|
|
fabs(rudder - APData->old_rudder) > APData->disengage_threshold )
|
|
{
|
|
// if controls changed externally turn autopilot off
|
|
APData->waypoint_hold = false ; // turn the target hold off
|
|
APData->heading_hold = false ; // turn the heading hold off
|
|
APData->altitude_hold = false ; // turn the altitude hold off
|
|
APData->terrain_follow = false; // turn the terrain_follow hold off
|
|
// APData->auto_throttle = false; // turn the auto_throttle off
|
|
|
|
// stash this runs control settings
|
|
APData->old_aileron = aileron;
|
|
APData->old_elevator = elevator;
|
|
APData->old_elevator_trim = elevator_trim;
|
|
APData->old_rudder = rudder;
|
|
|
|
return(0);
|
|
}
|
|
|
|
// waypoint hold enabled?
|
|
if ( APData->waypoint_hold == true )
|
|
{
|
|
double wp_course, wp_reverse, wp_distance;
|
|
|
|
#ifdef DO_fgAP_CORRECTED_COURSE
|
|
// compute course made good
|
|
// this needs lots of special casing before use
|
|
double course, reverse, distance, corrected_course;
|
|
// need to test for iter
|
|
geo_inverse_wgs_84( 0, //fgAPget_altitude(),
|
|
APData->old_lat,
|
|
APData->old_lon,
|
|
lat,
|
|
lon,
|
|
&course,
|
|
&reverse,
|
|
&distance );
|
|
#endif // DO_fgAP_CORRECTED_COURSE
|
|
|
|
// compute course to way_point
|
|
// need to test for iter
|
|
if( ! geo_inverse_wgs_84( 0, //fgAPget_altitude(),
|
|
lat,
|
|
lon,
|
|
APData->TargetLatitude,
|
|
APData->TargetLongitude,
|
|
&wp_course,
|
|
&wp_reverse,
|
|
&wp_distance ) ) {
|
|
|
|
#ifdef DO_fgAP_CORRECTED_COURSE
|
|
corrected_course = course - wp_course;
|
|
if( fabs(corrected_course) > 0.1 )
|
|
printf("fgAP: course %f wp_course %f %f %f\n",
|
|
course, wp_course, fabs(corrected_course), distance );
|
|
#endif // DO_fgAP_CORRECTED_COURSE
|
|
|
|
if ( wp_distance > 100 ) {
|
|
// corrected_course = course - wp_course;
|
|
APData->TargetHeading = NormalizeDegrees(wp_course);
|
|
} else {
|
|
printf("APData->distance(%f) to close\n", wp_distance);
|
|
// Real Close -- set heading hold to current heading
|
|
// and Ring the arival bell !!
|
|
NewTgtAirport(NULL);
|
|
APData->waypoint_hold = false;
|
|
// use previous
|
|
APData->TargetHeading = fgAPget_heading();
|
|
}
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
// Force this just in case
|
|
APData->TargetDistance = wp_distance;
|
|
MakeTargetDistanceStr( APData, wp_distance );
|
|
// This changes the AutoPilot Heading Read Out
|
|
// following cast needed
|
|
ApHeadingDialogInput -> setValue ((float)APData->TargetHeading );
|
|
}
|
|
APData->heading_hold = true;
|
|
}
|
|
|
|
// heading hold enabled?
|
|
if ( APData->heading_hold == true ) {
|
|
double RelHeading;
|
|
double TargetRoll;
|
|
double RelRoll;
|
|
double AileronSet;
|
|
|
|
RelHeading =
|
|
NormalizeDegrees( APData->TargetHeading - fgAPget_heading() );
|
|
// figure out how far off we are from desired heading
|
|
|
|
// Now it is time to deterime how far we should be rolled.
|
|
FG_LOG( FG_AUTOPILOT, FG_DEBUG, "RelHeading: " << RelHeading );
|
|
|
|
|
|
// Check if we are further from heading than the roll out point
|
|
if ( fabs( RelHeading ) > APData->RollOut ) {
|
|
// set Target Roll to Max in desired direction
|
|
if ( RelHeading < 0 ) {
|
|
TargetRoll = 0 - APData->MaxRoll;
|
|
} else {
|
|
TargetRoll = APData->MaxRoll;
|
|
}
|
|
} else {
|
|
// We have to calculate the Target roll
|
|
|
|
// This calculation engine thinks that the Target roll
|
|
// should be a line from (RollOut,MaxRoll) to (-RollOut,
|
|
// -MaxRoll) I hope this works well. If I get ambitious
|
|
// some day this might become a fancier curve or
|
|
// something.
|
|
|
|
TargetRoll = LinearExtrapolate( RelHeading, -APData->RollOut,
|
|
-APData->MaxRoll, APData->RollOut,
|
|
APData->MaxRoll );
|
|
}
|
|
|
|
// Target Roll has now been Found.
|
|
|
|
// Compare Target roll to Current Roll, Generate Rel Roll
|
|
|
|
FG_LOG( FG_COCKPIT, FG_BULK, "TargetRoll: " << TargetRoll );
|
|
|
|
RelRoll = NormalizeDegrees( TargetRoll - fgAPget_roll() );
|
|
|
|
// Check if we are further from heading than the roll out smooth point
|
|
if ( fabs( RelRoll ) > APData->RollOutSmooth ) {
|
|
// set Target Roll to Max in desired direction
|
|
if ( RelRoll < 0 ) {
|
|
AileronSet = 0 - APData->MaxAileron;
|
|
} else {
|
|
AileronSet = APData->MaxAileron;
|
|
}
|
|
} else {
|
|
AileronSet = LinearExtrapolate( RelRoll, -APData->RollOutSmooth,
|
|
-APData->MaxAileron,
|
|
APData->RollOutSmooth,
|
|
APData->MaxAileron );
|
|
}
|
|
|
|
controls.set_aileron( AileronSet );
|
|
controls.set_rudder( 0.0 );
|
|
}
|
|
|
|
// altitude hold or terrain follow enabled?
|
|
if ( APData->altitude_hold || APData->terrain_follow ) {
|
|
double speed, max_climb, error;
|
|
double prop_error, int_error;
|
|
double prop_adj, int_adj, total_adj;
|
|
|
|
if ( APData->altitude_hold ) {
|
|
// normal altitude hold
|
|
APData->TargetClimbRate =
|
|
( APData->TargetAltitude - fgAPget_altitude() ) * 8.0;
|
|
} else if ( APData->terrain_follow ) {
|
|
// brain dead ground hugging with no look ahead
|
|
APData->TargetClimbRate =
|
|
( APData->TargetAGL - fgAPget_agl() ) * 16.0;
|
|
} else {
|
|
// just try to zero out rate of climb ...
|
|
APData->TargetClimbRate = 0.0;
|
|
}
|
|
|
|
speed = get_speed();
|
|
|
|
if ( speed < 90.0 ) {
|
|
max_climb = 0.0;
|
|
} else if ( speed < 100.0 ) {
|
|
max_climb = ( speed - 90.0 ) * 20;
|
|
// } else if ( speed < 150.0 ) {
|
|
} else {
|
|
max_climb = ( speed - 100.0 ) * 4.0 + 200.0;
|
|
} //else { // this is NHV hack
|
|
// max_climb = ( speed - 150.0 ) * 6.0 + 300.0;
|
|
// }
|
|
|
|
if ( APData->TargetClimbRate > max_climb ) {
|
|
APData->TargetClimbRate = max_climb;
|
|
}
|
|
|
|
else if ( APData->TargetClimbRate < -400.0 ) {
|
|
APData->TargetClimbRate = -400.0;
|
|
}
|
|
|
|
error = fgAPget_climb() - APData->TargetClimbRate;
|
|
|
|
// accumulate the error under the curve ... this really should
|
|
// be *= delta t
|
|
APData->alt_error_accum += error;
|
|
|
|
// calculate integral error, and adjustment amount
|
|
int_error = APData->alt_error_accum;
|
|
// printf("error = %.2f int_error = %.2f\n", error, int_error);
|
|
int_adj = int_error / 8000.0;
|
|
|
|
// caclulate proportional error
|
|
prop_error = error;
|
|
prop_adj = prop_error / 2000.0;
|
|
|
|
total_adj = 0.9 * prop_adj + 0.1 * int_adj;
|
|
if ( total_adj > 0.6 ) {
|
|
total_adj = 0.6;
|
|
}
|
|
else if ( total_adj < -0.2 ) {
|
|
total_adj = -0.2;
|
|
}
|
|
|
|
controls.set_elevator( total_adj );
|
|
}
|
|
|
|
// auto throttle enabled?
|
|
if ( APData->auto_throttle ) {
|
|
double error;
|
|
double prop_error, int_error;
|
|
double prop_adj, int_adj, total_adj;
|
|
|
|
error = APData->TargetSpeed - get_speed();
|
|
|
|
// accumulate the error under the curve ... this really should
|
|
// be *= delta t
|
|
APData->speed_error_accum += error;
|
|
if ( APData->speed_error_accum > 2000.0 ) {
|
|
APData->speed_error_accum = 2000.0;
|
|
}
|
|
else if ( APData->speed_error_accum < -2000.0 ) {
|
|
APData->speed_error_accum = -2000.0;
|
|
}
|
|
|
|
// calculate integral error, and adjustment amount
|
|
int_error = APData->speed_error_accum;
|
|
|
|
// printf("error = %.2f int_error = %.2f\n", error, int_error);
|
|
int_adj = int_error / 200.0;
|
|
|
|
// caclulate proportional error
|
|
prop_error = error;
|
|
prop_adj = 0.5 + prop_error / 50.0;
|
|
|
|
total_adj = 0.9 * prop_adj + 0.1 * int_adj;
|
|
if ( total_adj > 1.0 ) {
|
|
total_adj = 1.0;
|
|
}
|
|
else if ( total_adj < 0.0 ) {
|
|
total_adj = 0.0;
|
|
}
|
|
|
|
controls.set_throttle( FGControls::ALL_ENGINES, total_adj );
|
|
}
|
|
|
|
#ifdef THIS_CODE_IS_NOT_USED
|
|
if (APData->Mode == 2) // Glide slope hold
|
|
{
|
|
double RelSlope;
|
|
double RelElevator;
|
|
|
|
// First, calculate Relative slope and normalize it
|
|
RelSlope = NormalizeDegrees( APData->TargetSlope - get_pitch());
|
|
|
|
// Now calculate the elevator offset from current angle
|
|
if ( abs(RelSlope) > APData->SlopeSmooth )
|
|
{
|
|
if ( RelSlope < 0 ) // set RelElevator to max in the correct direction
|
|
RelElevator = -APData->MaxElevator;
|
|
else
|
|
RelElevator = APData->MaxElevator;
|
|
}
|
|
|
|
else
|
|
RelElevator = LinearExtrapolate(RelSlope,-APData->SlopeSmooth,-APData->MaxElevator,APData->SlopeSmooth,APData->MaxElevator);
|
|
|
|
// set the elevator
|
|
fgElevMove(RelElevator);
|
|
|
|
}
|
|
#endif // THIS_CODE_IS_NOT_USED
|
|
|
|
// stash this runs control settings
|
|
// get_control_values();
|
|
APData->old_aileron = controls.get_aileron();
|
|
APData->old_elevator = controls.get_elevator();
|
|
APData->old_elevator_trim = controls.get_elevator_trim();
|
|
APData->old_rudder = controls.get_rudder();
|
|
|
|
// for cross track error
|
|
APData->old_lat = lat;
|
|
APData->old_lon = lon;
|
|
|
|
// Ok, we are done
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
void fgAPSetMode( int mode)
|
|
{
|
|
//Remove the following line when the calling funcitons start passing in the data pointer
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
fgPrintf( FG_COCKPIT, FG_INFO, "APSetMode : %d\n", mode );
|
|
|
|
APData->Mode = mode; // set the new mode
|
|
}
|
|
*/
|
|
#if 0
|
|
void fgAPset_tgt_airport_id( const string id ) {
|
|
FG_LOG( FG_AUTOPILOT, FG_INFO, "entering fgAPset_tgt_airport_id " << id );
|
|
fgAPDataPtr APData;
|
|
APData = APDataGlobal;
|
|
APData->tgt_airport_id = id;
|
|
FG_LOG( FG_AUTOPILOT, FG_INFO, "leaving fgAPset_tgt_airport_id "
|
|
<< APData->tgt_airport_id );
|
|
};
|
|
|
|
string fgAPget_tgt_airport_id( void ) {
|
|
fgAPDataPtr APData = APDataGlobal;
|
|
return APData->tgt_airport_id;
|
|
};
|
|
#endif
|
|
|
|
void fgAPToggleHeading( void ) {
|
|
// Remove at a later date
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
if ( APData->heading_hold || APData->waypoint_hold ) {
|
|
// turn off heading hold
|
|
APData->heading_hold = false;
|
|
APData->waypoint_hold = false;
|
|
} else {
|
|
// turn on heading hold, lock at current heading
|
|
APData->heading_hold = true;
|
|
APData->TargetHeading = fgAPget_heading();
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
ApHeadingDialogInput -> setValue ((float)APData->TargetHeading );
|
|
}
|
|
|
|
get_control_values();
|
|
FG_LOG( FG_COCKPIT, FG_INFO, " fgAPSetHeading: ("
|
|
<< APData->heading_hold << ") " << APData->TargetHeading );
|
|
}
|
|
|
|
void fgAPToggleWayPoint( void ) {
|
|
// Remove at a later date
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
if ( APData->waypoint_hold ) {
|
|
// turn off location hold
|
|
APData->waypoint_hold = false;
|
|
// set heading hold to current heading
|
|
// APData->heading_hold = true;
|
|
APData->TargetHeading = fgAPget_heading();
|
|
} else {
|
|
double course, reverse, distance;
|
|
// turn on location hold
|
|
// turn on heading hold
|
|
APData->old_lat = fgAPget_latitude();
|
|
APData->old_lon = fgAPget_longitude();
|
|
|
|
// need to test for iter
|
|
if(!geo_inverse_wgs_84( fgAPget_altitude(),
|
|
fgAPget_latitude(),
|
|
fgAPget_longitude(),
|
|
APData->TargetLatitude,
|
|
APData->TargetLongitude,
|
|
&course,
|
|
&reverse,
|
|
&distance ) ) {
|
|
APData->TargetHeading = course;
|
|
APData->TargetDistance = distance;
|
|
MakeTargetDistanceStr( APData, distance );
|
|
}
|
|
// Force this !
|
|
APData->waypoint_hold = true;
|
|
APData->heading_hold = true;
|
|
}
|
|
|
|
// This changes the AutoPilot Heading
|
|
// following cast needed
|
|
ApHeadingDialogInput->setValue ((float)APData->TargetHeading );
|
|
MakeTargetHeadingStr( APData, APData->TargetHeading );
|
|
|
|
get_control_values();
|
|
|
|
FG_LOG( FG_COCKPIT, FG_INFO, " fgAPSetLocation: ( "
|
|
<< APData->waypoint_hold << " "
|
|
<< APData->TargetLatitude << " "
|
|
<< APData->TargetLongitude << " ) "
|
|
);
|
|
}
|
|
|
|
|
|
void fgAPToggleAltitude( void ) {
|
|
// Remove at a later date
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
if ( APData->altitude_hold ) {
|
|
// turn off altitude hold
|
|
APData->altitude_hold = false;
|
|
} else {
|
|
// turn on altitude hold, lock at current altitude
|
|
APData->altitude_hold = true;
|
|
APData->terrain_follow = false;
|
|
APData->TargetAltitude = fgAPget_altitude();
|
|
APData->alt_error_accum = 0.0;
|
|
// alt_error_queue.erase( alt_error_queue.begin(),
|
|
// alt_error_queue.end() );
|
|
float target_alt = APData->TargetAltitude;
|
|
if ( current_options.get_units() == fgOPTIONS::FG_UNITS_FEET )
|
|
target_alt *= METER_TO_FEET;
|
|
|
|
ApAltitudeDialogInput->setValue(target_alt);
|
|
MakeTargetAltitudeStr( APData, target_alt);
|
|
}
|
|
|
|
get_control_values();
|
|
FG_LOG( FG_COCKPIT, FG_INFO, " fgAPSetAltitude: ("
|
|
<< APData->altitude_hold << ") " << APData->TargetAltitude );
|
|
}
|
|
|
|
|
|
void fgAPToggleAutoThrottle ( void ) {
|
|
// Remove at a later date
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
if ( APData->auto_throttle ) {
|
|
// turn off altitude hold
|
|
APData->auto_throttle = false;
|
|
} else {
|
|
// turn on terrain follow, lock at current agl
|
|
APData->auto_throttle = true;
|
|
APData->TargetSpeed = get_speed();
|
|
APData->speed_error_accum = 0.0;
|
|
}
|
|
|
|
get_control_values();
|
|
FG_LOG( FG_COCKPIT, FG_INFO, " fgAPSetAutoThrottle: ("
|
|
<< APData->auto_throttle << ") " << APData->TargetSpeed );
|
|
}
|
|
|
|
void fgAPToggleTerrainFollow( void ) {
|
|
// Remove at a later date
|
|
fgAPDataPtr APData;
|
|
|
|
APData = APDataGlobal;
|
|
// end section
|
|
|
|
if ( APData->terrain_follow ) {
|
|
// turn off altitude hold
|
|
APData->terrain_follow = false;
|
|
} else {
|
|
// turn on terrain follow, lock at current agl
|
|
APData->terrain_follow = true;
|
|
APData->altitude_hold = false;
|
|
APData->TargetAGL = fgAPget_agl();
|
|
APData->alt_error_accum = 0.0;
|
|
}
|
|
get_control_values();
|
|
|
|
FG_LOG( FG_COCKPIT, FG_INFO, " fgAPSetTerrainFollow: ("
|
|
<< APData->terrain_follow << ") " << APData->TargetAGL );
|
|
}
|
|
|
|
static double NormalizeDegrees( double Input ) {
|
|
// normalize the input to the range (-180,180]
|
|
// Input should not be greater than -360 to 360.
|
|
// Current rules send the output to an undefined state.
|
|
if ( Input > 180 )
|
|
while(Input > 180 )
|
|
Input -= 360;
|
|
else if ( Input <= -180 )
|
|
while ( Input <= -180 )
|
|
Input += 360;
|
|
return ( Input );
|
|
};
|
|
|
|
static double LinearExtrapolate( double x, double x1, double y1, double x2, double y2 ) {
|
|
// This procedure extrapolates the y value for the x posistion on a line defined by x1,y1; x2,y2
|
|
//assert(x1 != x2); // Divide by zero error. Cold abort for now
|
|
|
|
// Could be
|
|
// static double y = 0.0;
|
|
// double dx = x2 -x1;
|
|
// if( (dx < -FG_EPSILON ) || ( dx > FG_EPSILON ) )
|
|
// {
|
|
|
|
double m, b, y; // the constants to find in y=mx+b
|
|
// double m, b;
|
|
|
|
m = ( y2 - y1 ) / ( x2 - x1 ); // calculate the m
|
|
|
|
b = y1 - m * x1; // calculate the b
|
|
|
|
y = m * x + b; // the final calculation
|
|
|
|
// }
|
|
|
|
return ( y );
|
|
|
|
};
|
|
|
|
|
|
/* Direct and inverse distance functions */
|
|
/** Proceedings of the 7th International Symposium on Geodetic
|
|
Computations, 1985
|
|
"The Nested Coefficient Method for Accurate Solutions of Direct
|
|
and
|
|
Inverse Geodetic Problems With Any Length"
|
|
Zhang Xue-Lian
|
|
pp 747-763
|
|
*/
|
|
/* modified for FlightGear to use WGS84 only Norman Vine */
|
|
|
|
//#include "dstazfns.h"
|
|
#include <math.h>
|
|
#define GEOD_INV_PI (3.14159265358979323846)
|
|
|
|
/* s == distance */
|
|
/* az = azimuth */
|
|
|
|
/* for WGS_84 a = 6378137.000, rf = 298.257223563; */
|
|
|
|
static double M0( double e2 )
|
|
{ //double e4 = e2*e2;
|
|
return GEOD_INV_PI*(1.0 - e2*( 1.0/4.0 + e2*( 3.0/64.0 + e2*(5.0/256.0) )))/2.0;
|
|
}
|
|
/* s == distance */
|
|
int geo_direct_wgs_84 ( double alt, double lat1, double lon1, double az1, double s,
|
|
double *lat2, double *lon2, double *az2 )
|
|
{
|
|
double a = 6378137.000, rf = 298.257223563;
|
|
double RADDEG = (GEOD_INV_PI)/180.0, testv = 1.0E-10;
|
|
double f = ( rf > 0.0 ? 1.0/rf : 0.0 );
|
|
double b = a*(1.0-f), e2 = f*(2.0-f);
|
|
double phi1 = lat1*RADDEG, lam1 = lon1*RADDEG;
|
|
double sinphi1 = sin(phi1), cosphi1 = cos(phi1);
|
|
double azm1 = az1*RADDEG;
|
|
double sinaz1 = sin(azm1), cosaz1 = cos(azm1);
|
|
|
|
|
|
if( fabs(s) < 0.01 ) /* distance < centimeter => congruency */
|
|
{ *lat2 = lat1;
|
|
*lon2 = lon1;
|
|
*az2 = 180.0 + az1;
|
|
if( *az2 > 360.0 ) *az2 -= 360.0;
|
|
return 0;
|
|
}
|
|
else
|
|
if( cosphi1 ) /* non-polar origin */
|
|
{ /* u1 is reduced latitude */
|
|
double tanu1 = sqrt(1.0-e2)*sinphi1/cosphi1;
|
|
double sig1 = atan2(tanu1,cosaz1);
|
|
double cosu1 = 1.0/sqrt( 1.0 + tanu1*tanu1 ), sinu1 = tanu1*cosu1;
|
|
double sinaz = cosu1*sinaz1, cos2saz = 1.0-sinaz*sinaz;
|
|
double us = cos2saz*e2/(1.0-e2);
|
|
/* Terms */
|
|
double ta = 1.0+us*(4096.0+us*(-768.0+us*(320.0-175.0*us)))/16384.0,
|
|
tb = us*(256.0+us*(-128.0+us*(74.0-47.0*us)))/1024.0,
|
|
tc = 0;
|
|
/* FIRST ESTIMATE OF SIGMA (SIG) */
|
|
double first = s/(b*ta); /* !!*/
|
|
double sig = first;
|
|
double c2sigm, sinsig,cossig, temp,denom,rnumer, dlams, dlam;
|
|
do
|
|
{ c2sigm = cos(2.0*sig1+sig);
|
|
sinsig = sin(sig); cossig = cos(sig);
|
|
temp = sig;
|
|
sig = first +
|
|
tb*sinsig*(c2sigm+tb*(cossig*(-1.0+2.0*pow(c2sigm,2.0)) -
|
|
tb*c2sigm*(-3.0+4.0*pow(sinsig,2.0))*(-3.0+4.0*pow(c2sigm,2.0))/6.0)/4.0);
|
|
}
|
|
while( fabs(sig-temp) > testv);
|
|
/* LATITUDE OF POINT 2 */
|
|
/* DENOMINATOR IN 2 PARTS (TEMP ALSO USED LATER) */
|
|
temp = sinu1*sinsig-cosu1*cossig*cosaz1;
|
|
denom = (1.0-f)*sqrt(sinaz*sinaz+temp*temp);
|
|
/* NUMERATOR */
|
|
rnumer = sinu1*cossig+cosu1*sinsig*cosaz1;
|
|
*lat2 = atan2(rnumer,denom)/RADDEG;
|
|
/* DIFFERENCE IN LONGITUDE ON AUXILARY SPHERE (DLAMS ) */
|
|
rnumer = sinsig*sinaz1;
|
|
denom = cosu1*cossig-sinu1*sinsig*cosaz1;
|
|
dlams = atan2(rnumer,denom);
|
|
/* TERM C */
|
|
tc = f*cos2saz*(4.0+f*(4.0-3.0*cos2saz))/16.0;
|
|
/* DIFFERENCE IN LONGITUDE */
|
|
dlam = dlams-(1.0-tc)*f*sinaz*(sig+tc*sinsig*(c2sigm+tc*cossig*(-1.0+2.0*
|
|
pow(c2sigm,2.0))));
|
|
*lon2 = (lam1+dlam)/RADDEG;
|
|
if(*lon2 > 180.0 ) *lon2 -= 360.0;
|
|
if(*lon2 < -180.0 ) *lon2 += 360.0;
|
|
/* AZIMUTH - FROM NORTH */
|
|
*az2 = atan2(-sinaz,temp)/RADDEG;
|
|
if( fabs(*az2) < testv ) *az2 = 0.0;
|
|
if( *az2 < 0.0) *az2 += 360.0;
|
|
return 0;
|
|
}
|
|
else /* phi1 == 90 degrees, polar origin */
|
|
{ double dM = a*M0(e2) - s;
|
|
double paz = ( phi1 < 0.0 ? 180.0 : 0.0 );
|
|
return geo_direct_wgs_84( alt, 0.0, lon1, paz, dM,lat2,lon2,az2 );
|
|
}
|
|
}
|
|
|
|
int geo_inverse_wgs_84( double alt, double lat1, double lon1, double lat2,
|
|
double lon2, double *az1, double *az2, double *s )
|
|
{
|
|
double a = 6378137.000, rf = 298.257223563;
|
|
int iter=0;
|
|
double RADDEG = (GEOD_INV_PI)/180.0, testv = 1.0E-10;
|
|
double f = ( rf > 0.0 ? 1.0/rf : 0.0 );
|
|
double b = a*(1.0-f), e2 = f*(2.0-f);
|
|
double phi1 = lat1*RADDEG, lam1 = lon1*RADDEG;
|
|
double sinphi1 = sin(phi1), cosphi1 = cos(phi1);
|
|
double phi2 = lat2*RADDEG, lam2 = lon2*RADDEG;
|
|
double sinphi2 = sin(phi2), cosphi2 = cos(phi2);
|
|
|
|
if( (fabs(lat1-lat2) < testv &&
|
|
( fabs(lon1-lon2) < testv) || fabs(lat1-90.0) < testv ) )
|
|
{ /* TWO STATIONS ARE IDENTICAL : SET DISTANCE & AZIMUTHS TO ZERO */
|
|
*az1 = 0.0; *az2 = 0.0; *s = 0.0;
|
|
return 0;
|
|
} else
|
|
if( fabs(cosphi1) < testv ) /* initial point is polar */
|
|
{
|
|
int k = geo_inverse_wgs_84( alt, lat2,lon2,lat1,lon1, az1,az2,s );
|
|
b = *az1; *az1 = *az2; *az2 = b;
|
|
return 0;
|
|
} else
|
|
if( fabs(cosphi2) < testv ) /* terminal point is polar */
|
|
{
|
|
int k = geo_inverse_wgs_84( alt, lat1,lon1,lat1,lon1+180.0,
|
|
az1,az2,s );
|
|
*s /= 2.0;
|
|
*az2 = *az1 + 180.0;
|
|
if( *az2 > 360.0 ) *az2 -= 360.0;
|
|
return 0;
|
|
} else /* Geodesic passes through the pole (antipodal) */
|
|
if( (fabs( fabs(lon1-lon2) - 180 ) < testv) &&
|
|
(fabs(lat1+lat2) < testv) )
|
|
{
|
|
double s1,s2;
|
|
geo_inverse_wgs_84( alt, lat1,lon1, lat1,lon2, az1,az2, &s1 );
|
|
geo_inverse_wgs_84( alt, lat2,lon2, lat1,lon2, az1,az2, &s2 );
|
|
*az2 = *az1;
|
|
*s = s1 + s2;
|
|
return 0;
|
|
} else /* antipodal and polar points don't get here */
|
|
{
|
|
double dlam = lam2 - lam1, dlams = dlam;
|
|
double sdlams,cdlams, sig,sinsig,cossig, sinaz,
|
|
cos2saz, c2sigm;
|
|
double tc,temp, us,rnumer,denom, ta,tb;
|
|
double cosu1,sinu1, sinu2,cosu2;
|
|
/* Reduced latitudes */
|
|
temp = (1.0-f)*sinphi1/cosphi1;
|
|
cosu1 = 1.0/sqrt(1.0+temp*temp);
|
|
sinu1 = temp*cosu1;
|
|
temp = (1.0-f)*sinphi2/cosphi2;
|
|
cosu2 = 1.0/sqrt(1.0+temp*temp);
|
|
sinu2 = temp*cosu2;
|
|
|
|
do {
|
|
sdlams = sin(dlams), cdlams = cos(dlams);
|
|
sinsig = sqrt(pow(cosu2*sdlams,2.0)+
|
|
pow(cosu1*sinu2-sinu1*cosu2*cdlams,2.0));
|
|
cossig = sinu1*sinu2+cosu1*cosu2*cdlams;
|
|
|
|
sig = atan2(sinsig,cossig);
|
|
sinaz = cosu1*cosu2*sdlams/sinsig;
|
|
cos2saz = 1.0-sinaz*sinaz;
|
|
c2sigm = (sinu1 == 0.0 || sinu2 == 0.0 ? cossig :
|
|
cossig-2.0*sinu1*sinu2/cos2saz);
|
|
tc = f*cos2saz*(4.0+f*(4.0-3.0*cos2saz))/16.0;
|
|
temp = dlams;
|
|
dlams = dlam+(1.0-tc)*f*sinaz*
|
|
(sig+tc*sinsig*
|
|
(c2sigm+tc*cossig*(-1.0+2.0*pow(c2sigm,2.0))));
|
|
if (fabs(dlams) > GEOD_INV_PI && iter++ > 50)
|
|
return iter;
|
|
} while ( fabs(temp-dlams) > testv);
|
|
|
|
us = cos2saz*(pow(a,2.0)-pow(b,2.0))/pow(b,2.0); /* !! */
|
|
/* BACK AZIMUTH FROM NORTH */
|
|
rnumer = -(cosu1*sdlams);
|
|
denom = sinu1*cosu2-cosu1*sinu2*cdlams;
|
|
*az2 = atan2(rnumer,denom)/RADDEG;
|
|
if( fabs(*az2) < testv ) *az2 = 0.0;
|
|
if(*az2 < 0.0) *az2 += 360.0;
|
|
/* FORWARD AZIMUTH FROM NORTH */
|
|
rnumer = cosu2*sdlams;
|
|
denom = cosu1*sinu2-sinu1*cosu2*cdlams;
|
|
*az1 = atan2(rnumer,denom)/RADDEG;
|
|
if( fabs(*az1) < testv ) *az1 = 0.0;
|
|
if(*az1 < 0.0) *az1 += 360.0;
|
|
/* Terms a & b */
|
|
ta = 1.0+us*(4096.0+us*(-768.0+us*(320.0-175.0*us)))/
|
|
16384.0;
|
|
tb = us*(256.0+us*(-128.0+us*(74.0-47.0*us)))/1024.0;
|
|
/* GEODETIC DISTANCE */
|
|
*s = b*ta*(sig-tb*sinsig*
|
|
(c2sigm+tb*(cossig*(-1.0+2.0*pow(c2sigm,2.0))-tb*
|
|
c2sigm*(-3.0+4.0*pow(sinsig,2.0))*
|
|
(-3.0+4.0*pow(c2sigm,2.0))/6.0)/
|
|
4.0));
|
|
return 0;
|
|
}
|
|
}
|