rexx-things/samples/oorexx/distance.rex
2025-03-12 20:50:48 +00:00

91 lines
4.9 KiB
Rexx
Executable File

/* ---------------------------------------------------------------- */
/* Calculates the distance between (possibly a series) of */
/* 2 latitude/longitude pairs. May be used in a pipe. */
/* ---------------------------------------------------------------- */
/* */
/* Originally by Ruurd J. Idenburg */
/* */
/* */
/* No copyright, no licence, no guarantees or warrantees, be it */
/* explicit, implicit or whatever. Usage is totally and completely */
/* at the users own risk, the author shall not be liable for any */
/* damages whatsoever, for any reason whatsoever. */
/* */
/* Please keep this comment block intact when modifying this code */
/* and add a note with date and a description. */
/* */
/* ---------------------------------------------------------------- */
/* Parameter(s): */
/* */
/* None */
/* */
/* Result: */
/* */
/* For each pair of latitude/longitude the distance in thousands */
/* of a kilometer to the previous pair is going to STDOUT */
/* (usually the console). */
/* */
/* ---------------------------------------------------------------- */
/* 2007/12/31 - Initial version approximately. */
/* 2011/12/31 - Isolate calculus to allow use outside a pipe. */
/* 2021/05/20 - Updated shebang and ::requires iso RxMathLoadFunc */
/* ---------------------------------------------------------------- */
Call RxFuncAdd "MathLoadFuncs","rxmath","MathLoadFuncs"
Call MathLoadFuncs
/* ---------------------------------------------------------------- */
/* The following formula's can be used to calculate the distance */
/* between two points on earth: */
/* */
/* d=R*acos(sin(lat1)*sin(lat2)+cos(lat1)*cos(lat2)*cos(lon1-lon2)) */
/* */
/* or */
/* */
/* d=R*2*asin(sqrt((sin((lat1-lat2)/2))^2 + cos(lat1)*cos(lat2)* */
/* (sin((lon1-lon2)/2)^2)) */
/* */
/* where */
/* */
/* R -- the radius of the earth, commonly taken to be: 6378,137 km */
/* lat1/lat2 -- latitude in degrees (NOT degrees minutes seconds) */
/* lon1/lon2 -- longitude in degrees (NOT degrees minutes seconds */
/* */
/* The first formula is more subject to possible rounding errors */
/* for short distances and since the route description produced by */
/* Google Earth/Maps often has points within a couple of meters */
/* this code uses the second formula. */
/* */
/* ---------------------------------------------------------------- */
/* -oldradius = 6366710 */
googleradius = 6378137 /* -- Equatorial earth radius used by Google */
/* Doesn't seem necessary for Windows, might be necessary for Linux */
Do Until Lines()>0
Call SysSleep 0.1
End
Parse Value Linein() With flat flon tlat tlon
If (tlat<>'' & tlon<>'') Then Say Distance(flat,flon,tlat,tlon)
Do While Lines()>0
Parse Value Linein() With tlat tlon .
Say Distance(flat,flon,tlat,tlon)
flat = tlat
flon = tlon
End
Exit
Distance: Procedure Expose googleradius
Parse Arg flat, flon, tlat, tlon
sinlat = RxCalcSin((flat-tlat)/2)
cosplat = RxCalcCos(flat)
coslati = RxCalcCos(tlat)
sinlon = RxCalcSin((flon-tlon)/2)
sinlatp = (sinlat)**2
sinlonp = (sinlon)**2
radians = 2*RxCalcArcSin(RxCalcSqrt(sinlatp + cosplat*coslati*sinlonp),,'R')
distance = (radians*googleradius)~format(,0)/1000 'km'
Return distance /* -- returns distance in thousands of a Km */
/* ::requires 'rxmath' LIBRARY */