45#include <GeographicLib/Math.hpp>
46#include <GeographicLib/Geodesic.hpp>
48#include "FGLocation.h"
57 : mECLoc(1.0, 0.0, 0.0), mCacheValid(false)
62 mLon = mLat = mRadius = 0.0;
63 mGeodLat = GeodeticAltitude = 0.0;
77 mLon = mLat = mRadius = 0.0;
78 mGeodLat = GeodeticAltitude = 0.0;
83 double sinLat = sin(lat);
84 double cosLat = cos(lat);
85 double sinLon = sin(lon);
86 double cosLon = cos(lon);
87 mECLoc = { radius*cosLat*cosLon,
95 : mECLoc(lv), mCacheValid(false)
100 mLon = mLat = mRadius = 0.0;
101 mGeodLat = GeodeticAltitude = 0.0;
110 : mECLoc(l.mECLoc), mCacheValid(l.mCacheValid)
117 mEllipseSet = l.mEllipseSet;
125 if (!mCacheValid)
return;
134 mGeodLat = l.mGeodLat;
135 GeodeticAltitude = l.GeodeticAltitude;
143 mCacheValid = l.mCacheValid;
144 mEllipseSet = l.mEllipseSet;
153 if (!mCacheValid)
return *
this;
162 mGeodLat = l.mGeodLat;
163 GeodeticAltitude = l.GeodeticAltitude;
184 mECLoc(eX) = rtmp*cos(longitude);
185 mECLoc(eY) = rtmp*sin(longitude);
202 double fac = r/rtmp*cos(latitude);
206 mECLoc(eX) = r*cos(latitude);
209 mECLoc(eZ) = r*sin(latitude);
222 mECLoc *= radius/rold;
231 double sinLat = sin(lat);
232 double cosLat = cos(lat);
233 double sinLon = sin(lon);
234 double cosLon = cos(lon);
236 mECLoc = { radius*cosLat*cosLon,
237 radius*cosLat*sinLon,
248 double slat = sin(lat);
249 double clat = cos(lat);
250 double RN = a / sqrt(1.0 - e2*slat*slat);
252 mECLoc(eX) = (RN + height)*clat*cos(lon);
253 mECLoc(eY) = (RN + height)*clat*sin(lon);
254 mECLoc(eZ) = ((1 - e2)*RN + height)*slat;
277 double cosLat = cos(mLat);
278 return a*ec/sqrt(1.0-e2*cosLat*cosLat);
283void FGLocation::ComputeDerivedUnconditional(
void)
const
293 double sinLon, cosLon;
299 sinLon = mECLoc(eY)/rxy;
300 cosLon = mECLoc(eX)/rxy;
301 mLon = atan2(mECLoc(eY), mECLoc(eX));
305 double sinLat, cosLat;
306 if (mRadius == 0.0) {
312 GeodeticAltitude = -a;
316 mLat = atan2( mECLoc(eZ), rxy );
327 double s0 = fabs(mECLoc(eZ));
329 double c0 = ec * rxy;
330 double c02 = c0 * c0;
331 double s02 = s0 * s0;
332 double a02 = c02 + s02;
333 double a0 = sqrt(a02);
334 double a03 = a02 * a0;
335 double s1 = zc*a03 + c*s02*s0;
336 double c1 = rxy*a03 - c*c02*c0;
337 double cs0c0 = c*c0*s0;
338 double b0 = 1.5*cs0c0*((rxy*s0-zc*c0)*a0-cs0c0);
340 double cc = ec*(c1*a03-b0*c0);
341 mGeodLat = sign(mECLoc(eZ))*atan(s1 / cc);
342 double s12 = s1 * s1;
343 double cc2 = cc * cc;
344 double norm = sqrt(s12 + cc2);
346 sinLat = sign(mECLoc(eZ)) * s1 / norm;
347 GeodeticAltitude = (rxy*cc + s0*s1 - a*sqrt(ec2*s12 + cc2)) / norm;
350 sinLat = mECLoc(eZ)/mRadius;
351 cosLat = rxy/mRadius;
361 mTec2l = { -cosLon*sinLat, -sinLon*sinLat, cosLat,
362 -sinLon , cosLon , 0.0 ,
363 -cosLon*cosLat, -sinLon*cosLat, -sinLat };
378 double target_latitude)
const
382 GeographicLib::Geodesic geod(a, 1 - ec);
383 GeographicLib::Math::real distance;
384 geod.Inverse(mGeodLat * radtodeg, mLon * radtodeg, target_latitude * radtodeg,
385 target_longitude * radtodeg, distance);
393 double target_latitude)
const
397 GeographicLib::Geodesic geod(a, 1 - ec);
398 GeographicLib::Math::real heading, azimuth2;
399 geod.Inverse(mGeodLat * radtodeg, mLon * radtodeg, target_latitude * radtodeg,
400 target_longitude * radtodeg, heading, azimuth2);
402 return heading * degtorad;
This class implements a 3 element column vector.
double Magnitude(void) const
Length of the vector.
FGLocation holds an arbitrary location in the Earth centered Earth fixed reference frame (ECEF).
void SetEllipse(double semimajor, double semiminor)
Sets the semimajor and semiminor axis lengths for this planet.
void SetLongitude(double longitude)
Set the longitude.
const FGLocation & operator=(const FGColumnVector3 &v)
Sets this location via the supplied vector.
FGLocation(void)
Default constructor.
void SetLatitude(double latitude)
Set the GEOCENTRIC latitude.
double GetHeadingTo(double target_longitude, double target_latitude) const
Get the heading that should be followed from the current location to a given location along the short...
double GetSeaLevelRadius(void) const
Get the sea level radius in feet below the current location.
void SetRadius(double radius)
Set the distance from the center of the earth.
void SetPosition(double lon, double lat, double radius)
Sets the longitude, latitude and the distance from the center of the earth.
void SetPositionGeodetic(double lon, double lat, double height)
Sets the longitude, latitude and the distance above the reference spheroid.
double GetDistanceTo(double target_longitude, double target_latitude) const
Get the geodetic distance between the current location and a given location.
FGMatrix33 Transposed(void) const
Transposed matrix.
void InitMatrix(void)
Initialize the matrix.