-
Notifications
You must be signed in to change notification settings - Fork 31
/
SourceInv.py
207 lines (163 loc) · 6.26 KB
/
SourceInv.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
'''
A base class for faults and datasets
Written by Z. Duputel, November 2013.
'''
# Import external stuff
import copy
import numpy as np
import matplotlib.pyplot as plt
import pyproj as pp
from pyproj import Transformer
from pyproj.aoi import AreaOfInterest
from pyproj.database import query_utm_crs_info
#class SourceInv
class SourceInv(object):
'''
Class implementing the geographical transform. This class will
be parent to almost all the classes in csi.
You can specify either an official utm zone number or provide
longitude and latitude for a custom zone.
Args:
* name : Instance Name
* utmzone : UTM zone (optional, default=None)
* lon0 : Longitude defining the center of the custom utm zone
* lat0 : Latitude defining the center of the custom utm zone
* ellps : ellipsoid (optional, default='WGS84')
'''
# ----------------------------------------------------------------------
# Initialize class #
def __init__(self,name,utmzone=None,ellps='WGS84',lon0=None, lat0=None):
# Initialization
self.name = name
# Set the utm zone
self.utmzone = utmzone
self.ellps = ellps
self.lon0 = lon0
self.lat0 = lat0
self.set_utmzone(utmzone=utmzone,
ellps = ellps,
lon0 = lon0,
lat0 = lat0)
# All done
return
# ----------------------------------------------------------------------
# ----------------------------------------------------------------------
# Lon lat Transform
def ll2xy(self, lon, lat):
'''
Do the lat/lon to UTM transform.
Input is in degrees. UTM coordinates are returned in km.
Args:
* lon : Longitude (deg)
* lat : Latitude (deg)
Returns:
* x : UTM coordinate x (km)
* y : UTM coordinate y (km)
'''
# Transpose
x, y = self.proj2utm.transform(lon, lat)
# Put it in Km
x = x/1000.
y = y/1000.
# All done
return x, y
# ----------------------------------------------------------------------
# ----------------------------------------------------------------------
# UTM transform
def xy2ll(self, x, y):
'''
Do the UTm to lat/lon transform.
Input is in km. Output is in degrees.
Args:
* x : UTM longitude (km).
* y : UTM latitude (km)
Returns:
* lon : Longitude (degrees)
* lat : Latitude (degree)
'''
# Transpose and return
return self.proj2wgs.transform(x*1000., y*1000.)
# ----------------------------------------------------------------------
# ----------------------------------------------------------------------
# Set the UTM zone
def set_utmzone(self, utmzone=None, ellps='WGS84', lon0=None, lat0=None):
'''
Sets the UTM zone in the class.
Zone can be specified via its international number or
one can specify the center of a custom UTM zone via lon0, lat0.
Kwargs:
* ellps : Reference Ellipsoid
:Method 1:
* utmzone : International UTM zone number
:Method 2:
* lon0 : Longitude of the center of the custom UTM zone (deg)
* lat0 : Latitude of the center of the custom UTM zone (deg)
'''
# WGS
self.wgs = pp.CRS('WGS84')
# Cases
if utmzone is not None:
self.utm = pp.CRS(proj='utm', zone=utmzone, ellps=ellps)
else:
assert lon0 is not None, 'Please specify a 0 longitude'
assert lat0 is not None, 'Please specify a 0 latitude'
# Find the best zone
utm_crs_list = query_utm_crs_info(
datum_name="WGS 84",
area_of_interest=AreaOfInterest(
west_lon_degree=lon0-2.,
south_lat_degree=lat0-2.,
east_lon_degree=lon0+2,
north_lat_degree=lat0+2
),
)
self.utm = pp.CRS.from_epsg(utm_crs_list[0].code)
self.code = utm_crs_list[0].code
# Make the projector
self.proj2utm = Transformer.from_crs(self.wgs, self.utm, always_xy=True)
self.proj2wgs = Transformer.from_crs(self.utm, self.wgs, always_xy=True)
# Set Geod
self.geod = pp.Geod(ellps=ellps)
# Set utmzone
self.utmzone = utmzone
self.lon0 = lon0
self.lat0 = lat0
self.ellps = ellps
# All done
return
# ----------------------------------------------------------------------
# ----------------------------------------------------------------------
# Unset the utmzone
def unset_utmzone(self):
'''
Unsets the UTM zone. Basically, TransformerLocal object (pyproj) cannot
be pickled. Therefore, if one wants to pickle the fault object (which is
important for deepcopy or multiprocessing), one needs to delete the proj
instance before. This is a temporary hack until proj fixes the bug.
I raised a bug issue in the pyproj github project (issue #1058)
'''
# delete objects
del self.proj2utm
del self.proj2wgs
del self.utm
del self.wgs
del self.geod
# Make sure clean up is done
import gc
gc.collect()
# All done
return
# ----------------------------------------------------------------------
# Checks that longitude is between 0 and 360 (kind of useless)
def _checkLongitude(self):
'''
Iterates over the longitude array and checks if longitude is between
0 and 360
'''
# Check
if len(self.lon[self.lon<0.])>0:
self.lon[self.lon<0.] += 360.
# All done
return
# ----------------------------------------------------------------------
#EOF