Skip to content

Commit

Permalink
vdist, vreckon: add deg= option (default true)
Browse files Browse the repository at this point in the history
track2: correct deg=False radians
  • Loading branch information
scivision committed Nov 26, 2023
1 parent 13cf5d5 commit b2767ed
Show file tree
Hide file tree
Showing 3 changed files with 135 additions and 77 deletions.
48 changes: 48 additions & 0 deletions src/pymap3d/tests/test_matlab_track2.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
#!/usr/bin/env python3
"""
Compare with Matlab Mapping toolbox reckon()
"""

from __future__ import annotations

import pytest
from pytest import approx

try:
import numpy as np
from .matlab_engine import matlab_engine, has_mapping
except ImportError:
pytest.skip("Matlab Engine not found", allow_module_level=True)


import pymap3d.vincenty


def track2(eng, lat1: float, lon1: float, lat2: float, lon2: float, npts: int, deg: bool) -> tuple:
"""Using Matlab Engine to do same thing as Pymap3d"""
d = "degrees" if deg else "radians"

lats, lons = eng.track2(
"gc", lat1, lon1, lat2, lon2, eng.wgs84Ellipsoid(), d, float(npts), nargout=2
)
return np.array(lats).squeeze(), np.array(lons).squeeze()


@pytest.mark.parametrize("deg", [True, False])
def test_track2_compare(deg):
lat1, lon1 = 0., 80.
lat2, lon2 = 0., 81.
if not deg:
lat1, lon1, lat2, lon2 = np.radians((lat1, lon1, lat2, lon2))

eng = matlab_engine()

if not has_mapping(eng):
pytest.skip("Matlab Toolbox not found")

lats, lons = pymap3d.vincenty.track2(lat1, lon1, lat2, lon2, npts=4, deg=deg)

lats_m, lons_m = track2(eng, lat1, lon1, lat2, lon2, npts=4, deg=deg)

assert lats == approx(lats_m)
assert lons == approx(lons_m)
25 changes: 21 additions & 4 deletions src/pymap3d/tests/test_vincenty.py
Original file line number Diff line number Diff line change
@@ -1,8 +1,25 @@
import numpy as np

import pymap3d.vincenty as vincenty

import pytest
from pytest import approx


def test_track2():
lats, lons = vincenty.track2(40, 80, 65, -148, npts=3)
assert lats == approx([40, 69.633139886, 65])
assert lons == approx([80, 113.06849104, -148])
@pytest.mark.parametrize("deg", [True, False])
def test_track2_unit(deg):

pytest.importorskip("numpy")

lat1, lon1 = 0., 80.
lat2, lon2 = 0., 81.
lat0 = [0., 0., 0., 0.]
lon0 = [80., 80.33333, 80.66666, 81.]
if not deg:
lat1, lon1, lat2, lon2 = np.radians((lat1, lon1, lat2, lon2))
lat0, lon0 = np.radians((lat0, lon0))

lats, lons = vincenty.track2(lat1, lon1, lat2, lon2, npts=4, deg=deg)

assert lats == approx(lat0)
assert lons == approx(lon0)
139 changes: 66 additions & 73 deletions src/pymap3d/vincenty.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,10 @@

import logging
from copy import copy
from math import nan, pi
from math import nan, pi, tau

try:
from numpy import atleast_1d
from numpy import atleast_1d, linspace
except ImportError:
pass

Expand All @@ -33,13 +33,7 @@
__all__ = ["vdist", "vreckon", "track2"]


def vdist(
Lat1,
Lon1,
Lat2,
Lon2,
ell: Ellipsoid = None,
) -> tuple:
def vdist(Lat1, Lon1, Lat2, Lon2, ell: Ellipsoid = None, deg: bool = True) -> tuple:
"""
Using the reference ellipsoid, compute the distance between two points
within a few millimeters of accuracy, compute forward azimuth,
Expand Down Expand Up @@ -113,26 +107,35 @@ def vdist(

if ell is None:
ell = Ellipsoid.from_name("wgs84")

# %% Input check:
try:
Lat1 = atleast_1d(Lat1)
Lon1 = atleast_1d(Lon1)
Lat2 = atleast_1d(Lat2)
Lon2 = atleast_1d(Lon2)
if (abs(Lat1) > 90).any() | (abs(Lat2) > 90).any():
raise ValueError("Input latitudes must be in [-90, 90] degrees.")
except NameError:
if (abs(Lat1) > 90) | (abs(Lat2) > 90):
raise ValueError("Input latitudes must be in [-90, 90] degrees.")
pass
# %% Supply WGS84 earth ellipsoid axis lengths in meters:
a = ell.semimajor_axis
b = ell.semiminor_axis
f = ell.flattening
# %% convert inputs in degrees to radians:
lat1 = radians(Lat1)
lon1 = radians(Lon1)
lat2 = radians(Lat2)
lon2 = radians(Lon2)

if deg:
Lat1 = radians(Lat1)
Lon1 = radians(Lon1)
Lat2 = radians(Lat2)
Lon2 = radians(Lon2)

# keep old variable names in case someone is using them
lat1, lon1, lat2, lon2 = Lat1, Lon1, Lat2, Lon2

try:
if (abs(lat1) > pi / 2).any() | (abs(lat2) > pi / 2).any():
raise ValueError("Input latitudes must be in [-90, 90] degrees.")
except AttributeError:
if abs(lat1) > pi / 2 or abs(lat2) > pi / 2:
raise ValueError("Input latitudes must be in [-90, 90] degrees.")
# %% correct for errors at exact poles by adjusting 0.6 millimeters:
try:
i = abs(pi / 2 - abs(lat1)) < 1e-10
Expand Down Expand Up @@ -269,23 +272,18 @@ def vdist(
numer = cos(U2) * sin(lamb)
denom = cos(U1) * sin(U2) - sin(U1) * cos(U2) * cos(lamb)
a12 = atan2(numer, denom)
a12 %= 2 * pi
a12 %= tau

az = degrees(a12)
if deg:
a12 = degrees(a12)

try:
return dist_m.squeeze()[()], az.squeeze()[()]
return dist_m.squeeze()[()], a12.squeeze()[()]
except AttributeError:
return dist_m, az
return dist_m, a12


def vreckon(
Lat1,
Lon1,
Rng,
Azim,
ell: Ellipsoid = None,
) -> tuple:
def vreckon(Lat1, Lon1, Rng, Azim, ell: Ellipsoid = None, deg: bool = True) -> tuple:
"""
This is the Vincenty "forward" solution.
Expand All @@ -310,6 +308,8 @@ def vreckon(
intial azimuth (degrees) clockwide from north.
ell : Ellipsoid, optional
reference ellipsoid
deg : bool, optional
angular units degrees (default) or radians
Results
-------
Expand Down Expand Up @@ -348,13 +348,9 @@ def vreckon(
Lon1 = atleast_1d(Lon1)
Rng = atleast_1d(Rng)
Azim = atleast_1d(Azim)
if (abs(Lat1) > 90.0).any():
raise ValueError("Input lat. must be between -90 and 90 deg., inclusive.")
if (Rng < 0.0).any():
raise ValueError("Ground distance must be positive")
except NameError:
if abs(Lat1) > 90.0:
raise ValueError("Input lat. must be between -90 and 90 deg., inclusive.")
if Rng < 0.0:
raise ValueError("Ground distance must be positive")

Expand All @@ -367,8 +363,20 @@ def vreckon(
b = 6356752.31424518 # WGS84 earth flattening coefficient definition
f = (a - b) / a

lat1 = radians(Lat1) # intial latitude in radians
lon1 = radians(Lon1) # intial longitude in radians
if deg:
Lat1 = radians(Lat1) # intial latitude in radians
Lon1 = radians(Lon1) # intial longitude in radians
Azim = radians(Azim)

# in case someone is using the old variable names
lat1, lon1 = Lat1, Lon1

try:
if (abs(lat1) > pi / 2).any():
raise ValueError("Input lat. must be between -90 and 90 deg., inclusive.")
except AttributeError:
if abs(lat1) > pi / 2:
raise ValueError("Input lat. must be between -90 and 90 deg., inclusive.")

# correct for errors at exact poles by adjusting 0.6 millimeters:
try:
Expand All @@ -378,7 +386,7 @@ def vreckon(
if abs(pi / 2 - abs(lat1)) < 1e-10:
lat1 = sign(lat1) * (pi / 2 - (1e-10))

alpha1 = radians(Azim) # inital azimuth in radians
alpha1 = Azim # inital azimuth in radians
sinAlpha1 = sin(alpha1)
cosAlpha1 = cos(alpha1)

Expand Down Expand Up @@ -448,19 +456,22 @@ def vreckon(
* (sigma + C * sinSigma * (cos2SigmaM + C * cosSigma * (-1 + 2 * cos2SigmaM * cos2SigmaM)))
)

lon2 = degrees(lon1 + L)

lon2 = lon1 + L
# Truncates angles into the [-pi pi] range
# if lon2 > pi:
# lon2 = pi*((absolute(lon2)/pi) -
# 2*ceil(((absolute(lon2)/pi)-1)/2)) * sign(lon2)

lon2 = lon2 % 360 # follow [0, 360) convention
lon2 %= tau # follow [0, 360) convention

if deg:
lat2 = degrees(lat2)
lon2 = degrees(lon2)

try:
return degrees(lat2).squeeze()[()], lon2.squeeze()[()]
return lat2.squeeze()[()], lon2.squeeze()[()]
except AttributeError:
return degrees(lat2), lon2
return lat2, lon2


def track2(
Expand Down Expand Up @@ -491,7 +502,7 @@ def track2(
npts : int, optional
number of points (default is 100)
deg : bool, optional
degrees input/output (False: radians in/out)
angular units degrees (default) or radians
Results
-------
Expand All @@ -509,49 +520,31 @@ def track2(

if npts < 2:
raise ValueError("npts must be greater than 1")

if npts == 2:
return [lat1, lat2], [lon1, lon2]

if deg:
rlat1 = radians(lat1)
rlon1 = radians(lon1)
rlat2 = radians(lat2)
rlon2 = radians(lon2)
else:
rlat1, rlon1, rlat2, rlon2 = lat1, lon1, lat2, lon2
lat1 = radians(lat1)
lon1 = radians(lon1)
lat2 = radians(lat2)
lon2 = radians(lon2)

gcarclen = 2.0 * asin(
sqrt(
(sin((rlat1 - rlat2) / 2)) ** 2
+ cos(rlat1) * cos(rlat2) * (sin((rlon1 - rlon2) / 2)) ** 2
)
sqrt((sin((lat1 - lat2) / 2)) ** 2 + cos(lat1) * cos(lat2) * (sin((lon1 - lon2) / 2)) ** 2)
)
# check to see if points are antipodal (if so, route is undefined).
if abs(gcarclen - pi) < 1e-12:
raise ValueError(
"cannot compute intermediate points on a great circle whose endpoints are antipodal"
)

distance, azimuth = vdist(lat1, lon1, lat2, lon2)
incdist = distance / (npts - 1)

latpt = lat1
lonpt = lon1
lons = [lonpt]
lats = [latpt]
for _ in range(npts - 2):
latptnew, lonptnew = vreckon(latpt, lonpt, incdist, azimuth)
azimuth = vdist(latptnew, lonptnew, lat2, lon2, ell=ell)[1]
lats.append(latptnew)
lons.append(lonptnew)
latpt = latptnew
lonpt = lonptnew
lons.append(lon2)
lats.append(lat2)

if not deg:
lats = list(map(radians, lats))
lons = list(map(radians, lons))
distance, azimuth = vdist(lat1, lon1, lat2, lon2, ell, deg=False)
dists = linspace(0, distance, npts)

lats, lons = vreckon(lat1, lon1, dists, azimuth, ell, deg=False)

if deg:
lats = degrees(lats)
lons = degrees(lons)

return lats, lons

0 comments on commit b2767ed

Please sign in to comment.