jord 0.16.0

Geographical Position Calculations (Ellipsoidal and Spherical models)
Documentation
use crate::{
    surface::Surface, Angle, Cartesian3DVector, GeocentricPosition, GeodeticPosition, Length,
    NVector, Vec3,
};

/// An ellipsoid.
#[derive(PartialEq, Clone, Copy, Debug, Default)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] // codecov:ignore:this
pub struct Ellipsoid {
    equatorial_radius: Length,
    polar_radius: Length,
    eccentricity: f64,
    flattening: f64,
}

impl Ellipsoid {
    /// [World Geodetic](https://en.wikipedia.org/wiki/World_Geodetic_System) 84 Ellipsoid.
    pub const WGS84: Ellipsoid = Ellipsoid {
        equatorial_radius: Length::from_metres(6_378_137.0f64),
        polar_radius: Length::from_metres(6_356_752.314245179f64),
        eccentricity: 0.08181919084262157f64,
        flattening: 0.0033528106647474805f64,
    };

    /// Geodetic Reference System 1980 Ellipsoid.
    pub const GRS80: Ellipsoid = Ellipsoid {
        equatorial_radius: Length::from_metres(6_378_137.0f64),
        polar_radius: Length::from_metres(6_356_752.314140356f64),
        eccentricity: 0.08181919104281514f64,
        flattening: 0.003352810681182319f64,
    };

    /// [World Geodetic](https://en.wikipedia.org/wiki/World_Geodetic_System) 72 Ellipsoid.
    pub const WGS72: Ellipsoid = Ellipsoid {
        equatorial_radius: Length::from_metres(6_378_135.0f64),
        polar_radius: Length::from_metres(6_356_750.520016094f64),
        eccentricity: 0.08181881066274845f64,
        flattening: 0.003352779454167505,
    };

    /// [Mars Orbiter Laser Altimeter Ellipsoid](https://tharsis.gsfc.nasa.gov/geodesy.html).
    pub const MOLA: Ellipsoid = Ellipsoid {
        equatorial_radius: Length::from_metres(3_396_200f64),
        polar_radius: Length::from_metres(3_376_198.822143698f64),
        eccentricity: 0.10836918094475001f64,
        flattening: 0.005889281507656065f64,
    };

    /// Creates a new ellipsoid from the given equatorial radius (semi-major axis A) and
    /// inverse (or reciprocal) flattening.
    pub fn new(equatorial_radius: Length, inverse_flattening: f64) -> Self {
        let a = equatorial_radius.as_metres();
        let f = 1.0 / inverse_flattening;
        let b = a * (1.0 - f);
        let e = (1.0 - (b * b) / (a * a)).sqrt();
        Ellipsoid {
            equatorial_radius,
            polar_radius: Length::from_metres(b),
            eccentricity: e,
            flattening: f,
        }
    }

    /// Returns the equatorial radius (or semi-major axis A) of this ellipsoid.
    #[inline]
    pub fn equatorial_radius(&self) -> Length {
        self.equatorial_radius
    }

    /// Returns the polar radius (or semi-minor axis B) of this ellipsoid.
    #[inline]
    pub fn polar_radius(&self) -> Length {
        self.polar_radius
    }

    /// Returns the eccentricity of this ellipsoid.
    #[inline]
    pub fn eccentricity(&self) -> f64 {
        self.eccentricity
    }

    /// Returns the flattening of this ellipsoid.
    #[inline]
    pub fn flattening(&self) -> f64 {
        self.flattening
    }

    /// Returns the geocentric radius at the given geodetic latitude: the distance from the Earth's center
    /// to a position on the spheroid surface at geodetic latitude.
    ///
    /// See: [Location-dependent radii](https://en.wikipedia.org/wiki/Earth_radius#Location-dependent_radii)
    ///
    /// # Examples
    ///
    /// ```
    /// use jord::{Angle, Length};
    /// use jord::ellipsoidal::Ellipsoid;
    ///
    /// assert_eq!(Ellipsoid::WGS84.equatorial_radius(), Ellipsoid::WGS84.geocentric_radius(Angle::ZERO));
    /// assert_eq!(Ellipsoid::WGS84.polar_radius(), Ellipsoid::WGS84.geocentric_radius(Angle::from_degrees(90.0)));
    /// assert_eq!(
    ///     Length::from_metres(6_367_490.0),
    ///     Ellipsoid::WGS84.geocentric_radius(Angle::from_degrees(45.0)).round_m()
    /// );
    /// ```
    pub fn geocentric_radius(&self, latitude: Angle) -> Length {
        let cos_lat = latitude.as_radians().cos();
        let sin_lat = latitude.as_radians().sin();
        let a = self.equatorial_radius.as_metres();
        let b = self.polar_radius.as_metres();
        let f1 = a * a * cos_lat;
        let f2 = b * b * sin_lat;
        let f3 = a * cos_lat;
        let f4 = b * sin_lat;
        let r = (((f1 * f1) + (f2 * f2)) / ((f3 * f3) + (f4 * f4))).sqrt();
        Length::from_metres(r)
    }

    /// Returns the radius of the parallel of the given geodetic latitude.
    ///
    /// See: [Radius of the Earth](https://www.oc.nps.edu/oc2902w/geodesy/radiigeo.pdf)
    ///
    /// # Examples
    ///
    /// ```
    /// use jord::{Angle, Length};
    /// use jord::ellipsoidal::Ellipsoid;
    ///
    /// assert_eq!(Ellipsoid::WGS84.equatorial_radius(), Ellipsoid::WGS84.latitude_radius(Angle::ZERO));
    /// assert_eq!(Length::ZERO, Ellipsoid::WGS84.latitude_radius(Angle::from_degrees(90.0)));
    /// assert_eq!(Length::ZERO, Ellipsoid::WGS84.latitude_radius(Angle::from_degrees(-90.0)));
    /// ```
    pub fn latitude_radius(&self, latitude: Angle) -> Length {
        if latitude == Angle::QUARTER_CIRCLE || latitude == Angle::NEG_QUARTER_CIRCLE {
            Length::ZERO
        } else {
            self.prime_vertical_radius(latitude) * latitude.as_radians().cos()
        }
    }

    /// Returns the radius of curvature of the ellipsoid perpendicular to the meridian at the given geodetic latitude (often denoted `N`).
    ///
    /// See: [Radius of the Earth](https://www.oc.nps.edu/oc2902w/geodesy/radiigeo.pdf)
    ///
    /// # Examples
    ///
    /// ```
    /// use jord::{Angle, Length};
    /// use jord::ellipsoidal::Ellipsoid;
    ///
    /// assert_eq!(Ellipsoid::WGS84.equatorial_radius(), Ellipsoid::WGS84.prime_vertical_radius(Angle::ZERO));
    /// assert_eq!(
    ///     Length::from_metres(6388838.29),
    ///     Ellipsoid::WGS84.prime_vertical_radius(Angle::from_degrees(45.0)).round_mm()
    /// );
    /// ```
    pub fn prime_vertical_radius(&self, latitude: Angle) -> Length {
        let e2: f64 = self.eccentricity * self.eccentricity;
        let sin_lat = latitude.as_radians().sin();
        let sin_lat2 = sin_lat * sin_lat;
        let r = self.equatorial_radius.as_metres() / (1.0 - e2 * sin_lat2).sqrt();
        Length::from_metres(r)
    }

    /// Radius of curvature in the meridian at the given geodetic latitude (often denoted `M`).
    ///
    /// See: [Radius of the Earth](https://www.oc.nps.edu/oc2902w/geodesy/radiigeo.pdf)
    pub fn meridian_radius(&self, latitude: Angle) -> Length {
        let e2: f64 = self.eccentricity * self.eccentricity;
        let sin_lat = latitude.as_radians().sin();
        let sin_lat2 = sin_lat * sin_lat;
        let r =
            self.equatorial_radius.as_metres() * (1.0 - e2) / (1.0 - e2 * sin_lat2).powf(3.0 / 2.0);
        Length::from_metres(r)
    }

    /// Returns the mean radius (arithmetic mean) of this ellipsoid.
    ///
    /// # Examples
    ///
    /// ```
    /// use jord::Length;
    /// use jord::ellipsoidal::Ellipsoid;
    ///
    /// assert_eq!(Length::from_metres(6_371_008.8), Ellipsoid::WGS84.mean_radius().round_dm());
    /// ```
    pub fn mean_radius(&self) -> Length {
        let a = self.equatorial_radius();
        let b = self.polar_radius();
        (2.0 * a + b) / 3.0
    }

    /// Returns the volumetric radius of this ellipsoid: the radius of sphere of same volume.
    ///
    /// # Examples
    ///
    /// ```
    /// use jord::Length;
    /// use jord::spherical::Sphere;
    /// use jord::ellipsoidal::Ellipsoid;
    ///
    /// let r = (Ellipsoid::WGS84.volumetric_radius().as_metres() * 10.0).round() / 10.0;
    /// assert_eq!(Sphere::EARTH.radius().as_metres(), r);
    /// ```
    pub fn volumetric_radius(&self) -> Length {
        let a = self.equatorial_radius().as_metres();
        let b = self.polar_radius().as_metres();
        let r = (a * a * b).cbrt();
        Length::from_metres(r)
    }
}

impl Surface for Ellipsoid {
    fn geodetic_to_geocentric_position(&self, pos: GeodeticPosition) -> GeocentricPosition {
        let nv = pos.horizontal_position().as_vec3();
        let nx = nv.x();
        let ny = nv.y();
        let nz = nv.z();
        let a = self.equatorial_radius.as_metres();
        let b = self.polar_radius.as_metres();
        let m = (a * a) / (b * b);
        let n = b / ((nx * nx * m) + (ny * ny * m) + (nz * nz)).sqrt();
        let h = pos.height().as_metres();
        let cx = n * m * nx + h * nx;
        let cy = n * m * ny + h * ny;
        let cz = n * nz + h * nz;
        GeocentricPosition::from_metres(cx, cy, cz)
    }

    fn geocentric_to_geodetic_position(&self, pos: GeocentricPosition) -> GeodeticPosition {
        let pv = pos.as_metres();
        let px = pv.x();
        let py = pv.y();
        let pz = pv.z();
        let e2 = self.eccentricity * self.eccentricity;
        let e4 = e2 * e2;
        let a = self.equatorial_radius.as_metres();
        let a2 = a * a;
        let p = (px * px + py * py) / a2;
        let q = ((1.0 - e2) / a2) * (pz * pz);
        let r = (p + q - e4) / 6.0;
        let s = (e4 * p * q) / (4.0 * r * r * r);
        let t = (1.0 + s + (s * (2.0 + s)).sqrt()).powf(1.0 / 3.0);
        let u = r * (1.0 + t + 1.0 / t);
        let v = (u * u + q * e4).sqrt();
        let w = e2 * (u + v - q) / (2.0 * v);
        let k = (u + v + w * w).sqrt() - w;
        let d = k * (px * px + py * py).sqrt() / (k + e2);
        let h = ((k + e2 - 1.0) / k) * (d * d + pz * pz).sqrt();

        let fs = 1.0 / (d * d + pz * pz).sqrt();
        let fa = k / (k + e2);
        let nx = fs * fa * px;
        let ny = fs * fa * py;
        let nz = fs * pz;
        GeodeticPosition::new(NVector::new(Vec3::new(nx, ny, nz)), Length::from_metres(h))
    }
}

#[cfg(test)]
mod tests {
    use crate::{spherical::Sphere, Angle, Length};

    use super::Ellipsoid;

    #[test]
    fn wgs84() {
        let wgs84 = Ellipsoid::new(Length::from_metres(6_378_137.0), 298.257223563);
        assert_eq!(
            Ellipsoid::WGS84.equatorial_radius(),
            wgs84.equatorial_radius()
        );
        assert_eq!(Ellipsoid::WGS84.polar_radius(), wgs84.polar_radius());
        assert_eq!(Ellipsoid::WGS84.eccentricity(), wgs84.eccentricity());
        assert_eq!(Ellipsoid::WGS84.flattening(), wgs84.flattening());
    }

    #[test]
    fn grs80() {
        let grs80 = Ellipsoid::new(Length::from_metres(6_378_137.0), 298.257222101);
        assert_eq!(
            Ellipsoid::GRS80.equatorial_radius(),
            grs80.equatorial_radius()
        );
        assert_eq!(Ellipsoid::GRS80.polar_radius(), grs80.polar_radius());
        assert_eq!(Ellipsoid::GRS80.eccentricity(), grs80.eccentricity());
        assert_eq!(Ellipsoid::GRS80.flattening(), grs80.flattening());
    }

    #[test]
    fn wgs72() {
        let wgs72 = Ellipsoid::new(Length::from_metres(6_378_135.0), 298.26);
        assert_eq!(
            Ellipsoid::WGS72.equatorial_radius(),
            wgs72.equatorial_radius()
        );
        assert_eq!(Ellipsoid::WGS72.polar_radius(), wgs72.polar_radius());
        assert_eq!(Ellipsoid::WGS72.eccentricity(), wgs72.eccentricity());
        assert_eq!(Ellipsoid::WGS72.flattening(), wgs72.flattening());
    }

    #[test]
    fn mola() {
        let mola = Ellipsoid::new(Length::from_metres(3_396_200.0), 169.8);
        assert_eq!(
            Ellipsoid::MOLA.equatorial_radius(),
            mola.equatorial_radius()
        );
        assert_eq!(Ellipsoid::MOLA.polar_radius(), mola.polar_radius());
        assert_eq!(Ellipsoid::MOLA.eccentricity(), mola.eccentricity());
        assert_eq!(Ellipsoid::MOLA.flattening(), mola.flattening());
    }

    #[test]
    fn geocentric_radius() {
        assert_eq!(
            Ellipsoid::WGS84.equatorial_radius(),
            Ellipsoid::WGS84.geocentric_radius(Angle::ZERO)
        );
        assert_eq!(
            Ellipsoid::WGS84.polar_radius(),
            Ellipsoid::WGS84.geocentric_radius(Angle::from_degrees(90.0))
        );
        assert_eq!(
            Length::from_metres(6_367_490.0),
            Ellipsoid::WGS84
                .geocentric_radius(Angle::from_degrees(45.0))
                .round_m()
        );
    }

    #[test]
    fn latitude_radius() {
        assert_eq!(
            Ellipsoid::WGS84.equatorial_radius(),
            Ellipsoid::WGS84.latitude_radius(Angle::ZERO)
        );
        assert_eq!(
            Length::ZERO,
            Ellipsoid::WGS84.latitude_radius(Angle::from_degrees(90.0))
        );
        assert_eq!(
            Length::ZERO,
            Ellipsoid::WGS84.latitude_radius(Angle::from_degrees(-90.0))
        );
    }

    #[test]
    fn prime_vertical_radius() {
        assert_eq!(
            Ellipsoid::WGS84.equatorial_radius(),
            Ellipsoid::WGS84.prime_vertical_radius(Angle::ZERO)
        );
        assert_eq!(
            Length::from_metres(6388838.29),
            Ellipsoid::WGS84
                .prime_vertical_radius(Angle::from_degrees(45.0))
                .round_mm()
        );
    }

    #[test]
    fn mean_radius() {
        assert_eq!(
            Length::from_metres(6_371_008.8),
            Ellipsoid::WGS84.mean_radius().round_dm()
        );
    }

    #[test]
    fn volumetric_radius() {
        let r = (Ellipsoid::WGS84.volumetric_radius().as_metres() * 10.0).round() / 10.0;
        assert_eq!(Sphere::EARTH.radius().as_metres(), r);
    }
}