-
Notifications
You must be signed in to change notification settings - Fork 11
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
5 changed files
with
94 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,5 +6,6 @@ members = [ | |
"nusamai-geojson", | ||
"nusamai-plateau", | ||
"nusamai-mvt", | ||
"nusamai-projection" | ||
] | ||
resolver = "2" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
[package] | ||
name = "nusamai-projection" | ||
version = "0.1.0" | ||
edition = "2021" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
//! Conversion between geographic and geocentric (cartesian) coordinate systems. | ||
use crate::ellipsoid::Ellipsoid; | ||
|
||
/// Convert from geographic to geocentric coordinate system. | ||
pub fn geographic_to_geocentric<E: Ellipsoid>(lng: f64, lat: f64, height: f64) -> (f64, f64, f64) { | ||
let (lng_rad, lat_rad) = (lng.to_radians(), lat.to_radians()); | ||
let tan_psi = (1. - E::E_SQ) * lat_rad.tan(); | ||
let z = E::A * (1. / (1. / (tan_psi * tan_psi) + 1. / ((1. - E::F) * (1. - E::F)))).sqrt(); | ||
let r = E::A * (1. / (1. + (tan_psi * tan_psi) / ((1. - E::F) * (1. - E::F)))).sqrt(); | ||
let x = r * lng_rad.cos(); | ||
let y = r * lng_rad.sin(); | ||
let dhz = lat_rad.sin(); | ||
let dhx = lat_rad.cos() * lng_rad.cos(); | ||
let dhy = lat_rad.cos() * lng_rad.sin(); | ||
(x + dhx * height, y + dhy * height, z + dhz * height) | ||
} | ||
|
||
/// Convert from geocentric to geographic coordinate system. | ||
pub fn geocentric_to_geographic<E: Ellipsoid>(x: f64, y: f64, z: f64) -> (f64, f64, f64) { | ||
println!("not implemented: {:?}", (x, y, z)); | ||
todo!(); | ||
} | ||
|
||
#[cfg(test)] | ||
mod tests { | ||
use super::*; | ||
use crate::ellipsoid::WGS84; | ||
|
||
#[test] | ||
fn fixtures() { | ||
{ | ||
let (x, y, z) = geographic_to_geocentric::<WGS84>(140., 37., 50.); | ||
assert!((x - -3906851.9770472576).abs() < 1e-9); | ||
assert!((y - 3278238.0530045824).abs() < 1e-9); | ||
assert!((z - 3817423.251099322).abs() < 1e-9); | ||
} | ||
|
||
// north pole | ||
{ | ||
let height = 150.; | ||
let (x, y, z) = geographic_to_geocentric::<WGS84>(123., 90., 150.); | ||
assert!((x - 0.).abs() < 1e-9); | ||
assert!((y - 0.).abs() < 1e-9); | ||
assert!((z - (WGS84::B + height)).abs() < 1e-9); | ||
} | ||
|
||
// null island | ||
{ | ||
let height = 100.; | ||
let (x, y, z) = geographic_to_geocentric::<WGS84>(0., 0., height); | ||
assert!((x - (WGS84::A + height)).abs() < 1e-9); | ||
assert!((y - 0.).abs() < 1e-9); | ||
assert!((z - 0.).abs() < 1e-9); | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
/// Ellipsoid parameters | ||
pub trait Ellipsoid { | ||
/// Semi-major axis | ||
const A: f64; | ||
/// Inverse flattening | ||
const INV_F: f64; | ||
|
||
/// Flattening | ||
const F: f64 = 1. / Self::INV_F; | ||
/// Semi-minor axis | ||
const B: f64 = Self::A * (1. - Self::F); | ||
/// Eccentricity squared | ||
const E_SQ: f64 = Self::F * (2. - Self::F); | ||
} | ||
|
||
/// WGS84 Eliipsoid | ||
pub struct WGS84 {} | ||
impl Ellipsoid for WGS84 { | ||
const A: f64 = 6378137.; | ||
const INV_F: f64 = 298.257223563; | ||
} | ||
|
||
/// GRS80 Eliipsoid | ||
pub struct GRS80 {} | ||
impl Ellipsoid for GRS80 { | ||
const A: f64 = 6378137.; | ||
const INV_F: f64 = 298.257222101; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
pub mod cartesian; | ||
pub mod ellipsoid; | ||
|
||
pub use ellipsoid::*; |