{-# LANGUAGE FlexibleContexts, TypeOperators, TypeFamilies #-}
module Geodetics.Ellipsoids (
Helmert (..),
inverseHelmert,
ECEF,
applyHelmert,
Ellipsoid (..),
WGS84 (..),
LocalEllipsoid (..),
flattening,
minorRadius,
eccentricity2,
eccentricity'2,
normal,
latitudeRadius,
meridianRadius,
primeVerticalRadius,
isometricLatitude,
Vec3,
Matrix3,
add3,
scale3,
negate3,
transform3,
invert3,
trans3,
dot3,
cross3
) where
import Data.Monoid (Monoid)
import Data.Semigroup (Semigroup, (<>))
import Numeric.Units.Dimensional
import Numeric.Units.Dimensional.Prelude
import Prelude ()
type Vec3 a = (a,a,a)
type Matrix3 a = Vec3 (Vec3 a)
scale3 :: (Num a) =>
Vec3 (Quantity d a) -> Quantity d' a -> Vec3 (Quantity (d * d') a)
scale3 (x,y,z) s = (x*s, y*s, z*s)
negate3 :: (Num a) => Vec3 (Quantity d a) -> Vec3 (Quantity d a)
negate3 (x,y,z) = (negate x, negate y, negate z)
add3 :: (Num a) => Vec3 (Quantity d a) -> Vec3 (Quantity d a) -> Vec3 (Quantity d a)
add3 (x1,y1,z1) (x2,y2,z2) = (x1+x2, y1+y2, z1+z2)
transform3 :: (Num a) =>
Matrix3 (Quantity d a) -> Vec3 (Quantity d' a) -> Vec3 (Quantity (d*d') a)
transform3 (tx,ty,tz) v = (t tx v, t ty v, t tz v)
where
t (x1,y1,z1) (x2,y2,z2) = x1*x2 + y1*y2 + z1*z2
invert3 :: (Fractional a) =>
Matrix3 (Quantity d a) -> Matrix3 (Quantity ((d*d)/(d*d*d)) a)
invert3 ((x1,y1,z1),
(x2,y2,z2),
(x3,y3,z3)) =
((det2 y2 z2 y3 z3 / det, det2 z1 y1 z3 y3 / det, det2 y1 z1 y2 z2 / det),
(det2 z2 x2 z3 x3 / det, det2 x1 z1 x3 z3 / det, det2 z1 x1 z2 x2 / det),
(det2 x2 y2 x3 y3 / det, det2 y1 x1 y3 x3 / det, det2 x1 y1 x2 y2 / det))
where
det = (x1*y2*z3 + y1*z2*x3 + z1*x2*y3) - (z1*y2*x3 + y1*x2*z3 + x1*z2*y3)
det2 a b c d = a*d - b*c
trans3 :: Matrix3 a -> Matrix3 a
trans3 ((x1,y1,z1),(x2,y2,z2),(x3,y3,z3)) = ((x1,x2,x3),(y1,y2,y3),(z1,z2,z3))
dot3 :: (Num a) =>
Vec3 (Quantity d1 a) -> Vec3 (Quantity d2 a) -> Quantity (d1 * d2) a
dot3 (x1,y1,z1) (x2,y2,z2) = x1*x2 + y1*y2 + z1*z2
cross3 :: (Num a) =>
Vec3 (Quantity d1 a) -> Vec3 (Quantity d2 a) -> Vec3 (Quantity (d1 * d2) a)
cross3 (x1,y1,z1) (x2,y2,z2) = (y1*z2 - z1*y2, z1*x2 - x1*z2, x1*y2 - y1*x2)
data Helmert = Helmert {
cX, cY, cZ :: Length Double,
helmertScale :: Dimensionless Double,
rX, rY, rZ :: Dimensionless Double } deriving (Eq, Show)
instance Semigroup Helmert where
h1 <> h2 = Helmert (cX h1 + cX h2) (cY h1 + cY h2) (cZ h1 + cZ h2)
(helmertScale h1 + helmertScale h2)
(rX h1 + rX h2) (rY h1 + rY h2) (rZ h1 + rZ h2)
instance Monoid Helmert where
mempty = Helmert (0 *~ meter) (0 *~ meter) (0 *~ meter) _0 _0 _0 _0
mappend = (<>)
inverseHelmert :: Helmert -> Helmert
inverseHelmert h = Helmert (negate $ cX h) (negate $ cY h) (negate $ cZ h)
(negate $ helmertScale h)
(negate $ rX h) (negate $ rY h) (negate $ rZ h)
type ECEF = Vec3 (Length Double)
applyHelmert:: Helmert -> ECEF -> ECEF
applyHelmert h (x,y,z) = (
cX h + s * ( x - rZ h * y + rY h * z),
cY h + s * ( rZ h * x + y - rX h * z),
cZ h + s * (negate (rY h) * x + rX h * y + z))
where
s = _1 + helmertScale h * (1e-6 *~ one)
class (Show a, Eq a) => Ellipsoid a where
majorRadius :: a -> Length Double
flatR :: a -> Dimensionless Double
helmert :: a -> Helmert
helmertToWSG84 :: a -> ECEF -> ECEF
helmertToWSG84 e = applyHelmert (helmert e)
helmertFromWSG84 :: a -> ECEF -> ECEF
helmertFromWSG84 e = applyHelmert (inverseHelmert $ helmert e)
data WGS84 = WGS84
instance Eq WGS84 where _ == _ = True
instance Show WGS84 where
show _ = "WGS84"
instance Ellipsoid WGS84 where
majorRadius _ = 6378137.0 *~ meter
flatR _ = 298.257223563 *~ one
helmert _ = mempty
helmertToWSG84 _ = id
helmertFromWSG84 _ = id
data LocalEllipsoid = LocalEllipsoid {
nameLocal :: String,
majorRadiusLocal :: Length Double,
flatRLocal :: Dimensionless Double,
helmertLocal :: Helmert } deriving (Eq)
instance Show LocalEllipsoid where
show = nameLocal
instance Ellipsoid LocalEllipsoid where
majorRadius = majorRadiusLocal
flatR = flatRLocal
helmert = helmertLocal
flattening :: (Ellipsoid e) => e -> Dimensionless Double
flattening e = _1 / flatR e
minorRadius :: (Ellipsoid e) => e -> Length Double
minorRadius e = majorRadius e * (_1 - flattening e)
eccentricity2 :: (Ellipsoid e) => e -> Dimensionless Double
eccentricity2 e = _2 * f - (f * f) where f = flattening e
eccentricity'2 :: (Ellipsoid e) => e -> Dimensionless Double
eccentricity'2 e = (f * (_2 - f)) / (_1 - f * f) where f = flattening e
normal :: (Ellipsoid e) => e -> Angle Double -> Length Double
normal e lat = majorRadius e / sqrt (_1 - eccentricity2 e * sin lat ^ pos2)
latitudeRadius :: (Ellipsoid e) => e -> Angle Double -> Length Double
latitudeRadius e lat = normal e lat * cos lat
meridianRadius :: (Ellipsoid e) => e -> Angle Double -> Length Double
meridianRadius e lat =
majorRadius e * (_1 - eccentricity2 e)
/ sqrt ((_1 - eccentricity2 e * sin lat ^ pos2) ^ pos3)
primeVerticalRadius :: (Ellipsoid e) => e -> Angle Double -> Length Double
primeVerticalRadius e lat =
majorRadius e / sqrt (_1 - eccentricity2 e * sin lat ^ pos2)
isometricLatitude :: (Ellipsoid e) => e -> Angle Double -> Angle Double
isometricLatitude ellipse lat = atanh sinLat - e * atanh (e * sinLat)
where
sinLat = sin lat
e = sqrt $ eccentricity2 ellipse