Coverage for pygeodesy/karney.py : 96%
 
         
         
    Hot-keys on this page
r m x p toggle line displays
j k next/prev highlighted chunk
0 (zero) top of page
1 (one) first highlighted chunk
| 
 # -*- coding: utf-8 -*- 
 
 Wrapper around Python classes C{Geodesic} and C{GeodesicLine} and several C{Math} functions from I{Karney}'s Python package U{geographiclib<https://PyPI.org/project/geographiclib>}, provided that package is installed. 
 The I{wrapped} class methods return a L{GDict} instance offering access to the C{dict} items either by C{key} or by C{attribute} name. 
 With env variable C{PYGEODESY_GEOGRAPHICLIB} left undefined or set to C{"2"}, this module and L{pygeodesy.geodesicx} will use U{GeographicLib 2.0<https://GeographicLib.SourceForge.io/C++/doc/>} transcoding, otherwise C{1.52} or older. 
 Karney-based functionality ========================== 
 1. The following classes and functions in C{pygeodesy} 
 - L{AlbersEqualArea}, L{AlbersEqualArea2}, L{AlbersEqualArea4}, L{AlbersEqualAreaCylindrical}, L{AlbersEqualAreaNorth}, L{AlbersEqualAreaSouth} -- U{AlbersEqualArea<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1AlbersEqualArea.html>} 
 - L{CassiniSoldner} -- U{CassiniSoldner<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1CassiniSoldner.html>} 
 - L{EcefKarney} -- U{Geocentric<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1Geocentric.html>} 
 - L{Elliptic} -- U{EllipticFunction<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1EllipticFunction.html>} 
 - L{EquidistantExact}, L{EquidistantGeodSolve}, L{EquidistantKarney} -- U{AzimuthalEquidistant <https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1AzimuthalEquidistant.html>} 
 - L{Etm}, L{ExactTransverseMercator} -- U{TransverseMercatorExact <https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1TransverseMercatorExact.html>} 
 - L{GeodesicAreaExact}, L{PolygonArea} -- U{PolygonArea<https://GeographicLib.SourceForge.io/ html/classGeographicLib_1_1PolygonAreaT.html>} 
 - L{GeodesicExact}, L{GeodesicLineExact} -- U{GeodesicExact<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1GeodesicExact.html>}, U{GeodesicLineExact<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1GeodesicLineExact.html>} 
 - L{GeoidKarney} -- U{Geoid<https://GeographicLib.SourceForge.io/C++/doc/geoid.html>} 
 - L{Georef} -- U{Georef<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1Georef.html>} 
 - L{GnomonicExact}, L{GnomonicGeodSolve}, L{GnomonicKarney} -- U{Gnomonic <https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1Gnomonic.html>} 
 - L{KTransverseMercator} - U{TransverseMercator <https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1TransverseMercator.html>} 
 - L{LocalCartesian}, L{Ltp} -- U{LocalCartesian<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1LocalCartesian.html>} 
 - L{Rhumb}, L{RhumbLine} -- U{Rhumb<https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1Rhumb.html>}, U{RhumbLine<https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1RhumbLine.html>}, U{TransverseMercator<https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1TransverseMercator.html>} 
 - L{Ups} -- U{PolarStereographic<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1PolarStereographic.html>} 
 - L{Utm} -- U{TransverseMercator<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1TransverseMercator.html>} 
 - L{UtmUps}, L{Epsg} -- U{UTMUPS<https://GeographicLib.SourceForge.io/C++/doc/ classGeographicLib_1_1UTMUPS.html>} 
 - L{pygeodesy.atand}, L{pygeodesy.atan2d}, L{pygeodesy.sincos2}, L{pygeodesy.sincos2d}, L{pygeodesy.tand} -- U{ Math<https://GeographicLib.SourceForge.io/C++/doc/classGeographicLib_1_1Math.html>} 
 are I{transcoded} from C++ classes in I{Karney}'s U{GeographicLib<https://GeographicLib.SourceForge.io/C++/doc/annotated.html>}. 
 2. These C{pygeodesy} modules and classes 
 - L{ellipsoidalGeodSolve}, L{ellipsoidalKarney}, L{geodsolve}, L{karney}, L{rhumbsolve} - L{EquidistantKarney}, L{FrechetKarney}, L{GeodesicSolve}, L{GeodesicLineSolve}, L{GnomonicGeodSolve}, L{GnomonicKarney}, L{HeightIDWkarney} 
 are or use I{wrappers} around I{Karney}'s Python U{geographiclib<https://PyPI.org/project/geographiclib>} C{geodesic}, C++ utility U{GeodSolve<https://GeographicLib.SourceForge.io/C++/doc/GeodSolve.1.html>} or C++ utility U{RhumbSolve<https://GeographicLib.SourceForge.io/C++/doc/RhumbSolve.1.html>}. 
 3. All C{pygeodesy} functions and methods to compute I{ellipsoidal} intersections and trilaterations 
 - L{ellipsoidalExact.intersection3}, L{ellipsoidalExact.intersections2}, L{ellipsoidalExact.nearestOn}, L{ellipsoidalExact.LatLon.intersection3}, L{ellipsoidalExact.LatLon.intersections2}, L{ellipsoidalExact.LatLon.nearestOn}, L{ellipsoidalExact.LatLon.trilaterate5} 
 - L{ellipsoidalKarney.intersection3}, L{ellipsoidalKarney.intersections2}, L{ellipsoidalKarney.nearestOn}, L{ellipsoidalKarney.LatLon.intersection3}, L{ellipsoidalKarney.LatLon.intersections2}, L{ellipsoidalKarney.LatLon.nearestOn}, L{ellipsoidalKarney.LatLon.trilaterate5} 
 - L{ellipsoidalVincenty.intersection3}, L{ellipsoidalVincenty.intersections2}, L{ellipsoidalVincenty.nearestOn}, L{ellipsoidalVincenty.LatLon.intersection3}, L{ellipsoidalVincenty.LatLon.intersections2}, L{ellipsoidalVincenty.LatLon.nearestOn}, L{ellipsoidalVincenty.LatLon.trilaterate5} 
 are implementations of I{Karney}'s solution posted under U{The B{ellipsoidal} case <https://GIS.StackExchange.com/questions/48937/calculating-intersection-of-two-circles>} and in paper U{Geodesics on an ellipsoid of revolution<https://ArXiv.org/pdf/1102.1215.pdf>} (pp 20-21, section B{14. MARITIME BOUNDARIES}). 
 4. Spherical functions 
 - L{pygeodesy.excessKarney_}, L{sphericalTrigonometry.areaOf} 
 in C{pygeodesy} are based on I{Karney}'s post U{Area of a spherical polygon <https://MathOverflow.net/questions/97711/the-area-of-spherical-polygons>}, 3rd Answer. ''' 
 _xgeographiclib, _xImportError, _xversion_info, \ _xinstanceof, _zip, isodd # PYCHOK shared # from pygeodesy.ellipsoids import Ellipsoid2 # from .datums _or # PYCHOK shared _lon2_, _m12_, _M12_, _M21_, _number_, _s12_, _S12_, \ _0_0, _180_0, _N_180_0, _360_0, _1_16th _NamedTuple, _Pass # from pygeodesy.streps import unstr # from .fmath Meter as _M, Meter2 as _M2, Number_, \ Precision_, _1mm as _TOL_M # PYCHOK shared 
 
 
 
 '''(INTERNAL) Latitude B{C{lat}}. ''' 
 
 '''(INTERNAL) Longitude B{C{lon}}. ''' 
 
 '''(INTERNAL) Get an ellipsoid from C{(B{a_..}, B{f})} or C{B{.._ellipsoid}}. ''' _ellipsoidal_datum(a_ellipsoid, name=name, raiser=raiser).ellipsoid 
 
 def _raiseX(inst, x, *args): # PYCHOK no cover '''(INTERNAL) Throw a C{GeodesicError} for C{geographiclib} issue B{C{x}} . ''' n = _DOT_(classname(inst), callername(up=2, underOK=True)) raise GeodesicError(unstr(n, *args), txt=str(x)) 
 
 '''(INTERNAL) Helper. ''' '''Convert this C{*Tuple} to a L{GDict}. 
 @kwarg updates: Optional items to apply (C{nam=value} pairs) ''' 
 
 '''3-Tuple C{(number, perimeter, area)} with the C{number} of points on the polygon or polyline, the C{perimeter} in C{meter} and the C{area} in C{meter} I{squared}. ''' 
 
 '''(INTERNAL) Overriden by C{Caps} below. ''' 
 
 
 
 
 
 _DEBUG_LINE | _ANGLE_ONLY | _SALPs_CALPs 
 # _REDUCEDLENGTH_GEODESICSCALE_DISTANCE = REDUCEDLENGTH | GEODESICSCALE | DISTANCE 
 '''I{Enum}-style masks to be bit-C{or}'ed to specify geodesic or rhumb capabilities (C{caps}) and expected results (C{outmask}). 
 C{AREA} - compute area C{S12}, 
 C{AZIMUTH} - include azimuths C{azi1} and C{azi2}, 
 C{DISTANCE} - compute distance C{s12}, 
 C{DISTANCE_IN} - allow distance C{s12} in C{.Direct}, 
 C{EMPTY} - nothing, formerly aka C{NONE}, 
 C{GEODESICSCALE} - compute geodesic scales C{M12} and C{M21}, 
 C{LINE_OFF} - Line without updates from parent geodesic or rhumb. 
 C{LATITUDE} - compute latitude C{lat2}, 
 C{LONGITUDE} - compute longitude C{lon2}, 
 C{LONG_UNROLL} - unroll C{lon2} in C{.Direct}, 
 C{REDUCEDLENGTH} - compute reduced length C{m12}, 
 C{REVERSE2} - reverse C{azi2}, 
 and C{ALL} - all of the above. 
 C{STANDARD} = C{AZIMUTH | DISTANCE | DISTANCE_IN | LATITUDE | LONGITUDE}''' 
 
 '''(INTERNAL) Base class for C{[_]Geodesic*Exact}. ''' 
 
 '''Get the capabilities (bit-or'ed C{Caps}). ''' 
 '''Check the available capabilities. 
 @arg caps: Bit-or'ed combination of L{Caps} values for all capabilities to be checked. 
 @return: C{True} if I{all} B{C{caps}} are available, C{False} otherwise (C{bool}). ''' caps &= Caps._OUT_ALL return (self.caps & caps) == caps 
 '''Get the C{debug} option (C{bool}). ''' 
 '''Set the C{debug} option (C{bool}) to include more details in L{GDict} results. ''' 
 '''(INTERNAL) Copy C{C{s}.iter} into C{B{r}._iteration}. ''' 
 
 '''9-Tuple C{(a12, lat2, lon2, azi2, s12, m12, M12, M21, S12)} with arc length C{a12}, angles C{lat2}, C{lon2} and azimuth C{azi2} in C{degrees}, distance C{s12} and reduced length C{m12} in C{meter} and area C{S12} in C{meter} I{squared}. ''' 
 
 '''Basic C{dict} with both key I{and} attribute access to the C{dict} items. 
 Results of all C{geodesic} methods are returned as a L{GDict} instance. ''' 
 '''Get the iteration number (C{int}) or C{None} if not available/applicable. ''' 
 '''Convert this L{GDict} result to a 9-tuple, like I{Karney}'s method C{geographiclib.geodesic.Geodesic._GenDirect}. 
 @kwarg dflt: Default value for missing items (C{any}). 
 @return: L{Direct9Tuple}C{(a12, lat2, lon2, azi2, s12, m12, M12, M21, S12)} ''' 
 '''Convert this L{GDict} result to a 12-Tuple, compatible with I{Karney}'s U{GeodSolve<https://GeographicLib.SourceForge.io/C++/doc/GeodSolve.1.html>} result. 
 @kwarg dflt: Default value for missing items (C{any}). 
 @return: L{GeodSolve12Tuple}C{(lat1, lon1, azi1, lat2, lon2, azi2, s12, a12, m12, M12, M21, S12)}. ''' return self._toTuple(_MODS.geodsolve.GeodSolve12Tuple, dflt) 
 '''Convert this L{GDict} result to a 10-tuple, like I{Karney}'s method C{geographiclib.geodesic.Geodesic._GenInverse}. 
 @kwarg dflt: Default value for missing items (C{any}). 
 @return: L{Inverse10Tuple}C{(a12, s12, salp1, calp1, salp2, calp2, m12, M12, M21, S12)}. ''' 
 '''DEPRECATED, used method C{toRhumb8Tuple}. 
 @return: A I{DEPRECATED} L{Rhumb7Tuple}. ''' return self._toTuple(_MODS.deprecated.Rhumb7Tuple, dflt) 
 '''Convert this L{GDict} result to a 8-tuple. 
 @kwarg dflt: Default value for missing items (C{any}). 
 @return: L{Rhumb8Tuple}C{(lat1, lon1, lat2, lon2, azi12, s12, S12, a12)}. ''' 
 '''Convert this L{GDict} result to a 8-tuple. 
 @kwarg dflt: Default value for missing items (C{any}). 
 @return: L{RhumbSolve7Tuple}C{(lat1, lon1, lat2, lon2, azi12, s12, S12)}. ''' return self._toTuple(_MODS.rhumbsolve.RhumbSolve7Tuple, dflt) 
 '''(INTERNAL) Convert this C{GDict} to an B{C{nTuple}}. ''' 
 
 '''Error raised for L{pygeodesy.geodesicx} lack of convergence or other L{pygeodesy.geodesicx} or L{pygeodesy.karney} issues. ''' 
 
 '''10-Tuple C{(a12, s12, salp1, calp1, salp2, calp2, m12, M12, M21, S12)} with arc length C{a12} in C{degrees}, distance C{s12} and reduced length C{m12} in C{meter}, area C{S12} in C{meter} I{squared} and the sines C{salp1}, C{salp2} and cosines C{calp1}, C{calp2} of the initial C{1} and final C{2} foward azimuths. ''' 
 '''Convert this C{Inverse10Tuple} to a L{GDict}. 
 @kwarg updates: Optional items to apply (C{nam=value} pairs) ''' azi2=atan2d(self.salp2, self.calp2), # PYCHOK namedTuple **updates) # PYCHOK indent 
 
 ''''(INTERNAL) Wrapper for some of I{Karney}'s U{geographiclib <https://PyPI.org/project/geographiclib>} classes. ''' 
 '''Get the I{wrapped} C{Geodesic} class, provided the U{geographiclib <https://PyPI.org/project/geographiclib>} package is installed, otherwise an C{ImportError}. ''' 
 '''I{Karney}'s U{Geodesic<https://GeographicLib.SourceForge.io/C++/doc/ python/code.html#geographiclib.geodesic.Geodesic>} wrapper. ''' 
 '''New C{Geodesic} instance. 
 @arg a_ellipsoid: An ellipsoid (L{Ellipsoid}) or datum (L{Datum}) or the equatorial radius of the ellipsoid (C{meter}). @arg f: The flattening of the ellipsoid (C{scalar}) if B{C{a_ellipsoid}) is specified as C{meter}. @kwarg name: Optional name (C{str}). ''' except (TypeError, ValueError) as x: _raiseX(self, x, *self.ellipsoid.a_f) 
 
 '''Get the C{debug} option (C{bool}). ''' return bool(self._debug) 
 '''Set the C{debug} option (C{bool}) to include more details in L{GDict} results. ''' 
 '''Return the C{Direct} result. ''' except (TypeError, ValueError) as x: _raiseX(self, x, lat1, lon1, azi1, s12, *outmask) 
 '''Return the destination lat, lon and reverse azimuth (final bearing) in C{degrees}. 
 @return: L{Destination3Tuple}C{(lat, lon, final)}. ''' 
 '''Get this geodesic's ellipsoid (C{Ellipsoid[2]}). ''' 
 '''Get the geodesic's ellipsoid I{1 - flattening} (C{float}). ''' 
 outmask=_Geodesic.STANDARD): '''(INTERNAL) Get C{._GenDirect} result as C{GDict}. ''' except (TypeError, ValueError) as x: _raiseX(self, x, lat, lon, azi, arcmode, s12_a12, outmask) 
 '''(INTERNAL) Get C{._GenInverse} result as C{GDict}. ''' except (TypeError, ValueError) as x: _raiseX(self, x, lat1, lon1, lat2, lon2, outmask) 
 '''Return the C{Inverse} result. ''' except (TypeError, ValueError) as x: _raiseX(self, x, lat1, lon1, lat2, lon2, *outmask) 
 '''Return the non-negative, I{angular} distance in C{degrees}. ''' # see .FrechetKarney.distance, .HausdorffKarney._distance # and .HeightIDWkarney._distances # XXX self.DISTANCE needed for 'a12'? 
 '''Return the distance in C{meter} and the forward and reverse azimuths (initial and final bearing) in C{degrees}. 
 @return: L{Distance3Tuple}C{(distance, initial, final)}. ''' 
 '''Set up a L{GeodesicLine} to compute several points on a single geodesic. ''' 
 # Geodesic.Direct.__doc__ = _Geodesic.Direct.__doc__ # Geodesic.Inverse.__doc__ = _Geodesic.Inverse.__doc__ # Geodesic.Line.__doc__ = _Geodesic.Line.__doc__ 
 '''Get the I{wrapped} C{GeodesicLine} class, provided the U{geographiclib <https://PyPI.org/project/geographiclib>} package is installed, otherwise an C{ImportError}. ''' 
 '''I{Karney}'s U{GeodesicLine <https://GeographicLib.SourceForge.io/C++/doc/ python/code.html#geographiclib.geodesicline.GeodesicLine>} wrapper. ''' except (TypeError, ValueError) as x: _raiseX(self, x, lat1, lon1, azi1, *caps) 
 '''Get the I{equatorial arc} (C{degrees}), the arc length between the northward equatorial crossing and point C{(lat1, lon1)}. 
 @see: U{EquatorialArc<https://GeographicLib.SourceForge.io/ C++/doc/classGeographicLib_1_1GeodesicLine.html>} ''' except AttributeError: return NAN # see .geodesicx.gxline._GeodesicLineExact 
 
 except (TypeError, ValueError) as x: _raiseX(self, x, a12, *outmask) 
 '''Get the I{equatorial azimuth} (C{degrees}), the azimuth of the geodesic line as it crosses the equator in a northward direction. 
 @see: U{EquatorialAzimuth<https://GeographicLib.SourceForge.io/ C++/doc/classGeographicLib_1_1GeodesicLine.html>} ''' except AttributeError: return NAN # see .geodesicx.gxline._GeodesicLineExact 
 
 except (TypeError, ValueError) as x: _raiseX(self, x, s12, *outmask) 
 # GeodesicLine.ArcPosition.__doc__ = _GeodesicLine.ArcPosition.__doc__ # GeodesicLine.Position.__doc__ = _GeodesicLine.Position.__doc__ 
 '''Get the I{wrapped} C{Geodesic.WGS84} I{instance} provided the U{geographiclib<https://PyPI.org/project/geographiclib>} package is installed, otherwise an C{ImportError}. ''' 
 '''Get the imported C{geographiclib}, provided the U{geographiclib <https://PyPI.org/project/geographiclib>} package is installed, otherwise an C{ImportError}. ''' 
 '''Get the C{Math} class, provided the U{geographiclib <https://PyPI.org/project/geographiclib>} package is installed, otherwise C{None}. ''' M = None # elif not _K_2_0: # XXX set 2.0? # _K_2_0 = True except (AttributeError, ImportError): M = None 
 
 
 '''I{Coarsen} a scalar by rounding small values to underflow to C{0.0}. 
 @return: Coarsened value (C{float}). 
 @see: I{Karney}'s U{Math.AngRound<https://SourceForge.net/p/ geographiclib/code/ci/release/tree/python/geographiclib/geomath.py>} ''' except AttributeError: pass else: 
 
 '''Return C{atan2(B{y}, B{x})} in C{degrees}. ''' except AttributeError: return atan2d(y, x) 
 
 '''Return C{cubic root(B{x})}. ''' except AttributeError: return cbrt(x) 
 
 '''Like C{copysign0(B{x}, B{y})}, with C{B{x} > 0}. ''' 
 
 '''Compute C{deg - deg0}, reduced to C{[-180,180]} accurately. 
 @return: 2-Tuple C{(delta_angle, residual)} in C{degrees}. ''' except AttributeError: pass fremainder( deg, _360_0)) else: d, t = _sum2(_norm180(-deg0), _norm180(deg)) d = _norm180(d) if t > 0 and d == _180_0: d = _N_180_0 d, t = _sum2(d, t) 
 
 # def _Equidistant(equidistant, exact=False, geodsolve=False): # # (INTERNAL) Get the C{EquidistantExact}, C{-GeodSolve} or # # C{-Karney} class if B{C{equidistant}} in not callable. # if equidistant is None or not callable(equidistant): # if exact: # equidistant = _MODS.azimuthal.EquidistantExact # elif geodsolve: # equidistant = _MODS.azimuthal.EquidistantGeodSolve # else: # equidistant = _MODS.azimuthal.EquidistantKarney # return equidistant 
 
 '''Replace angle in C{degrees} outside [-90,90] by NAN. 
 @return: Angle C{degrees} or NAN. ''' except AttributeError: return NAN if abs(deg) > 90 else deg 
 
 '''Check finiteness of C{x}. 
 @return: C{True} if finite. ''' try: return _wrapped.Math.isfinite(x) except AttributeError: return _math_isfinite(x) # and abs(x) <= _MAX 
 
 '''Normalize C{B{x}} and C{B{y}}. 
 @return: 2-Tuple of C{(B{x}, B{y})}, normalized. ''' except AttributeError: return norm2(x, y) 
 
 '''Reduce angle in C{degrees} to (-180,180]. 
 @return: Reduced angle C{degrees}. ''' except AttributeError: pass 
 
 '''(INTERNAL) Compute the area or perimeter of a polygon, using a L{GeodesicExact}, L{GeodesicSolve} or (if the C{geographiclib} package is installed) a C{Geodesic} or C{_wrapped.Geodesic} instance. ''' raise _ValueError(wrap=wrap) 
 
 
 # note, lon deltas are unrolled, by default pA(p0.lat, p0.lon) 
 # gP.Compute returns (number_of_points, perimeter, signed area) 
 
 '''(INTERNAL) Like C{GeographicLib.Math.hpp.polyval} but with a different signature and cascaded summation as C{karney._sum2_}. 
 @return: M{sum(cs[k] * x**(j - k - 1) for k in range(i, j)} ''' 
 
 '''Remainder of C{x / y}. 
 @return: Remainder in the range M{[-y / 2, y / 2]}, preserving signed 0.0. ''' except AttributeError: return fremainder(x, y) 
 
 
 
 else: 
 '''(INTERNAL) GeographicLin 1.52-. ''' 
 
 '''Return sine and cosine of an angle in C{degrees}. 
 @return: 2-Tuple C{(sin(B{deg}), cos(B{deg}))}. ''' except AttributeError: return sincos2d(deg) 
 
 '''Return sine and cosine of a corrected angle in C{degrees}. 
 @return: 2-Tuple C{(sin(B{deg}), cos(B{deg}))}. ''' except AttributeError: return sincos2d(deg, adeg=t) 
 
 '''Error-free summation like C{Math::sum}. 
 @return: 2-Tuple C{(B{u} + B{v}, residual)}. 
 @note: The C{residual} can be the same as B{C{u}} or B{C{v}}. 
 @see: U{Algorithm 3.1<https://www.TUHH.De/ti3/paper/rump/OgRuOi05.pdf>}. ''' except AttributeError: pass # if Algorithm_3_1: # t = (u - t) + (v + r) # elif C_CPP: # Math::sum C/C++ # r -= u # t -= v # t += r # t = -t # else: 
 
 '''Accumulate any B{C{vs}} into a previous C{_sum2(s, t)}. 
 @return: 2-Tuple C{(B{s} + B{t} + sum(B{vs}), residual)}. 
 @see: I{Karney's} C++ U{Accumulator<https://GeographicLib.SourceForge.io/ html/Accumulator_8hpp_source.html>} comments for more details and function C{_sum2} above. 
 @note: NOT "error-free", see C{pygeodesy.test/testKarney.py}. ''' # elif t: # s == 0 implies t == 0 # raise _AssertionError(t=t, txt=_not_(_0_)) else: else: 
 
 '''Return C{tan(B{x})} in C{degrees}. ''' except AttributeError: return tand(x) 
 
 '''Unroll B{C{lon2 - lon1}} like C{geodesic.Geodesic.Inverse}. 
 @return: 2-Tuple C{(B{lon2} - B{lon1}, B{lon2})} with B{C{lon2}} unrolled if B{C{wrap}} is C{True}, normalized otherwise. ''' else: lon2 = _norm180(lon2) 
 
 '''(INTERNAL) Unsign B{C{x}}. ''' 
 
 
 # **) MIT License # # Copyright (C) 2016-2022 -- mrJean1 at Gmail -- All Rights Reserved. # # Permission is hereby granted, free of charge, to any person obtaining a # copy of this software and associated documentation files (the "Software"), # to deal in the Software without restriction, including without limitation # the rights to use, copy, modify, merge, publish, distribute, sublicense, # and/or sell copies of the Software, and to permit persons to whom the # Software is furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included # in all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS # OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL # THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR # OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, # ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR # OTHER DEALINGS IN THE SOFTWARE. |