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.
		
		
		
		
			
				
					233 lines
				
				6.5 KiB
			
		
		
			
		
	
	
					233 lines
				
				6.5 KiB
			| 
											8 years ago
										 | 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
 |