You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							232 lines
						
					
					
						
							6.5 KiB
						
					
					
				
			
		
		
	
	
							232 lines
						
					
					
						
							6.5 KiB
						
					
					
				import math
 | 
						|
from utm.error import OutOfRangeError
 | 
						|
 | 
						|
__all__ = ['to_latlon', 'from_latlon']
 | 
						|
 | 
						|
K0 = 0.9996
 | 
						|
 | 
						|
E = 0.00669438
 | 
						|
E2 = E * E
 | 
						|
E3 = E2 * E
 | 
						|
E_P2 = E / (1.0 - E)
 | 
						|
 | 
						|
SQRT_E = math.sqrt(1 - E)
 | 
						|
_E = (1 - SQRT_E) / (1 + SQRT_E)
 | 
						|
_E2 = _E * _E
 | 
						|
_E3 = _E2 * _E
 | 
						|
_E4 = _E3 * _E
 | 
						|
_E5 = _E4 * _E
 | 
						|
 | 
						|
M1 = (1 - E / 4 - 3 * E2 / 64 - 5 * E3 / 256)
 | 
						|
M2 = (3 * E / 8 + 3 * E2 / 32 + 45 * E3 / 1024)
 | 
						|
M3 = (15 * E2 / 256 + 45 * E3 / 1024)
 | 
						|
M4 = (35 * E3 / 3072)
 | 
						|
 | 
						|
P2 = (3. / 2 * _E - 27. / 32 * _E3 + 269. / 512 * _E5)
 | 
						|
P3 = (21. / 16 * _E2 - 55. / 32 * _E4)
 | 
						|
P4 = (151. / 96 * _E3 - 417. / 128 * _E5)
 | 
						|
P5 = (1097. / 512 * _E4)
 | 
						|
 | 
						|
R = 6378137
 | 
						|
 | 
						|
ZONE_LETTERS = "CDEFGHJKLMNPQRSTUVWXX"
 | 
						|
 | 
						|
 | 
						|
def to_latlon(easting, northing, zone_number, zone_letter=None, northern=None, strict=True):
 | 
						|
    """This function convert an UTM coordinate into Latitude and Longitude
 | 
						|
 | 
						|
        Parameters
 | 
						|
        ----------
 | 
						|
        easting: int
 | 
						|
            Easting value of UTM coordinate
 | 
						|
 | 
						|
        northing: int
 | 
						|
            Northing value of UTM coordinate
 | 
						|
 | 
						|
        zone number: int
 | 
						|
            Zone Number is represented with global map numbers of an UTM Zone
 | 
						|
            Numbers Map. More information see utmzones [1]_
 | 
						|
 | 
						|
        zone_letter: str
 | 
						|
            Zone Letter can be represented as string values. Where UTM Zone
 | 
						|
            Designators can be accessed in [1]_
 | 
						|
 | 
						|
        northern: bool
 | 
						|
            You can set True or False to set this parameter. Default is None
 | 
						|
 | 
						|
 | 
						|
       .. _[1]: http://www.jaworski.ca/utmzones.htm
 | 
						|
 | 
						|
    """
 | 
						|
    if not zone_letter and northern is None:
 | 
						|
        raise ValueError('either zone_letter or northern needs to be set')
 | 
						|
 | 
						|
    elif zone_letter and northern is not None:
 | 
						|
        raise ValueError('set either zone_letter or northern, but not both')
 | 
						|
 | 
						|
    if strict:
 | 
						|
        if not 100000 <= easting < 1000000:
 | 
						|
            raise OutOfRangeError('easting out of range (must be between 100.000 m and 999.999 m)')
 | 
						|
        if not 0 <= northing <= 10000000:
 | 
						|
            raise OutOfRangeError('northing out of range (must be between 0 m and 10.000.000 m)')
 | 
						|
    if not 1 <= zone_number <= 60:
 | 
						|
        raise OutOfRangeError('zone number out of range (must be between 1 and 60)')
 | 
						|
 | 
						|
    if zone_letter:
 | 
						|
        zone_letter = zone_letter.upper()
 | 
						|
 | 
						|
        if not 'C' <= zone_letter <= 'X' or zone_letter in ['I', 'O']:
 | 
						|
            raise OutOfRangeError('zone letter out of range (must be between C and X)')
 | 
						|
 | 
						|
        northern = (zone_letter >= 'N')
 | 
						|
 | 
						|
    x = easting - 500000
 | 
						|
    y = northing
 | 
						|
 | 
						|
    if not northern:
 | 
						|
        y -= 10000000
 | 
						|
 | 
						|
    m = y / K0
 | 
						|
    mu = m / (R * M1)
 | 
						|
 | 
						|
    p_rad = (mu +
 | 
						|
             P2 * math.sin(2 * mu) +
 | 
						|
             P3 * math.sin(4 * mu) +
 | 
						|
             P4 * math.sin(6 * mu) +
 | 
						|
             P5 * math.sin(8 * mu))
 | 
						|
 | 
						|
    p_sin = math.sin(p_rad)
 | 
						|
    p_sin2 = p_sin * p_sin
 | 
						|
 | 
						|
    p_cos = math.cos(p_rad)
 | 
						|
 | 
						|
    p_tan = p_sin / p_cos
 | 
						|
    p_tan2 = p_tan * p_tan
 | 
						|
    p_tan4 = p_tan2 * p_tan2
 | 
						|
 | 
						|
    ep_sin = 1 - E * p_sin2
 | 
						|
    ep_sin_sqrt = math.sqrt(1 - E * p_sin2)
 | 
						|
 | 
						|
    n = R / ep_sin_sqrt
 | 
						|
    r = (1 - E) / ep_sin
 | 
						|
 | 
						|
    c = _E * p_cos**2
 | 
						|
    c2 = c * c
 | 
						|
 | 
						|
    d = x / (n * K0)
 | 
						|
    d2 = d * d
 | 
						|
    d3 = d2 * d
 | 
						|
    d4 = d3 * d
 | 
						|
    d5 = d4 * d
 | 
						|
    d6 = d5 * d
 | 
						|
 | 
						|
    latitude = (p_rad - (p_tan / r) *
 | 
						|
                (d2 / 2 -
 | 
						|
                 d4 / 24 * (5 + 3 * p_tan2 + 10 * c - 4 * c2 - 9 * E_P2)) +
 | 
						|
                 d6 / 720 * (61 + 90 * p_tan2 + 298 * c + 45 * p_tan4 - 252 * E_P2 - 3 * c2))
 | 
						|
 | 
						|
    longitude = (d -
 | 
						|
                 d3 / 6 * (1 + 2 * p_tan2 + c) +
 | 
						|
                 d5 / 120 * (5 - 2 * c + 28 * p_tan2 - 3 * c2 + 8 * E_P2 + 24 * p_tan4)) / p_cos
 | 
						|
 | 
						|
    return (math.degrees(latitude),
 | 
						|
            math.degrees(longitude) + zone_number_to_central_longitude(zone_number))
 | 
						|
 | 
						|
 | 
						|
def from_latlon(latitude, longitude, force_zone_number=None):
 | 
						|
    """This function convert Latitude and Longitude to UTM coordinate
 | 
						|
 | 
						|
        Parameters
 | 
						|
        ----------
 | 
						|
        latitude: float
 | 
						|
            Latitude between 80 deg S and 84 deg N, e.g. (-80.0 to 84.0)
 | 
						|
 | 
						|
        longitude: float
 | 
						|
            Longitude between 180 deg W and 180 deg E, e.g. (-180.0 to 180.0).
 | 
						|
 | 
						|
        force_zone number: int
 | 
						|
            Zone Number is represented with global map numbers of an UTM Zone
 | 
						|
            Numbers Map. You may force conversion including one UTM Zone Number.
 | 
						|
            More information see utmzones [1]_
 | 
						|
 | 
						|
       .. _[1]: http://www.jaworski.ca/utmzones.htm
 | 
						|
    """
 | 
						|
    if not -80.0 <= latitude <= 84.0:
 | 
						|
        raise OutOfRangeError('latitude out of range (must be between 80 deg S and 84 deg N)')
 | 
						|
    if not -180.0 <= longitude <= 180.0:
 | 
						|
        raise OutOfRangeError('longitude out of range (must be between 180 deg W and 180 deg E)')
 | 
						|
 | 
						|
    lat_rad = math.radians(latitude)
 | 
						|
    lat_sin = math.sin(lat_rad)
 | 
						|
    lat_cos = math.cos(lat_rad)
 | 
						|
 | 
						|
    lat_tan = lat_sin / lat_cos
 | 
						|
    lat_tan2 = lat_tan * lat_tan
 | 
						|
    lat_tan4 = lat_tan2 * lat_tan2
 | 
						|
 | 
						|
    if force_zone_number is None:
 | 
						|
        zone_number = latlon_to_zone_number(latitude, longitude)
 | 
						|
    else:
 | 
						|
        zone_number = force_zone_number
 | 
						|
 | 
						|
    zone_letter = latitude_to_zone_letter(latitude)
 | 
						|
 | 
						|
    lon_rad = math.radians(longitude)
 | 
						|
    central_lon = zone_number_to_central_longitude(zone_number)
 | 
						|
    central_lon_rad = math.radians(central_lon)
 | 
						|
 | 
						|
    n = R / math.sqrt(1 - E * lat_sin**2)
 | 
						|
    c = E_P2 * lat_cos**2
 | 
						|
 | 
						|
    a = lat_cos * (lon_rad - central_lon_rad)
 | 
						|
    a2 = a * a
 | 
						|
    a3 = a2 * a
 | 
						|
    a4 = a3 * a
 | 
						|
    a5 = a4 * a
 | 
						|
    a6 = a5 * a
 | 
						|
 | 
						|
    m = R * (M1 * lat_rad -
 | 
						|
             M2 * math.sin(2 * lat_rad) +
 | 
						|
             M3 * math.sin(4 * lat_rad) -
 | 
						|
             M4 * math.sin(6 * lat_rad))
 | 
						|
 | 
						|
    easting = K0 * n * (a +
 | 
						|
                        a3 / 6 * (1 - lat_tan2 + c) +
 | 
						|
                        a5 / 120 * (5 - 18 * lat_tan2 + lat_tan4 + 72 * c - 58 * E_P2)) + 500000
 | 
						|
 | 
						|
    northing = K0 * (m + n * lat_tan * (a2 / 2 +
 | 
						|
                                        a4 / 24 * (5 - lat_tan2 + 9 * c + 4 * c**2) +
 | 
						|
                                        a6 / 720 * (61 - 58 * lat_tan2 + lat_tan4 + 600 * c - 330 * E_P2)))
 | 
						|
 | 
						|
    if latitude < 0:
 | 
						|
        northing += 10000000
 | 
						|
 | 
						|
    return easting, northing, zone_number, zone_letter
 | 
						|
 | 
						|
 | 
						|
def latitude_to_zone_letter(latitude):
 | 
						|
    if -80 <= latitude <= 84:
 | 
						|
        return ZONE_LETTERS[int(latitude + 80) >> 3]
 | 
						|
    else:
 | 
						|
        return None
 | 
						|
 | 
						|
 | 
						|
def latlon_to_zone_number(latitude, longitude):
 | 
						|
    if 56 <= latitude < 64 and 3 <= longitude < 12:
 | 
						|
        return 32
 | 
						|
 | 
						|
    if 72 <= latitude <= 84 and longitude >= 0:
 | 
						|
        if longitude <= 9:
 | 
						|
            return 31
 | 
						|
        elif longitude <= 21:
 | 
						|
            return 33
 | 
						|
        elif longitude <= 33:
 | 
						|
            return 35
 | 
						|
        elif longitude <= 42:
 | 
						|
            return 37
 | 
						|
 | 
						|
    return int((longitude + 180) / 6) + 1
 | 
						|
 | 
						|
 | 
						|
def zone_number_to_central_longitude(zone_number):
 | 
						|
    return (zone_number - 1) * 6 - 180 + 3
 | 
						|
 |