-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmap_UTM.py
270 lines (213 loc) · 11.5 KB
/
map_UTM.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
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
import pandas as pd
import numpy as np
import matplotlib.pyplot as plt
import matplotlib.patches as mpatch
import pyproj
from matplotlib.patches import Rectangle
from prelude import load_flights, dist_less, make_base_dir
# Load flight files and filenames
flight_files,filenames = load_flights(2, 4, 11, 27)
# Open output file for writing
output = open('all_station_crossing_db_updated111.txt','w')
utm_proj = pyproj.Proj(proj='utm', zone='6', ellps='WGS84')
min_lon = -150.7
max_lon = -147.3
min_lat = 62.2
max_lat = 65.3
lxmin,lymin = utm_proj(min_lon, min_lat)
lxmax, lymax = utm_proj(max_lon, max_lat)
# Load the seismometer location data
seismo_data = pd.read_csv('input/all_sta.txt', sep="|")
seismo_latitudes = seismo_data['Latitude']
seismo_longitudes = seismo_data['Longitude']
sta = seismo_data['Station']
# Convert latitude and longitude to UTM coordinates
seismo_utm = [utm_proj(lon, lat) for lat, lon in zip(seismo_latitudes, seismo_longitudes)]
seismo_utm_x, seismo_utm_y = zip(*seismo_utm)
# Convert UTM coordinates to kilometers
seismo_utm_x_km = [x / 1000 for x in seismo_utm_x]
seismo_utm_y_km = [y / 1000 for y in seismo_utm_y]
seismo_utm_km = [(x, y) for x, y in zip(seismo_utm_x_km, seismo_utm_y_km)]
def closest_point_on_segment(flight_utm_x1, flight_utm_y1, flight_utm_x2, flight_utm_y2, seismo_utm_x, seismo_utm_y):
closest_point = None
dist_lim = np.Infinity
x = [flight_utm_x1, flight_utm_x2]
y = [flight_utm_y1, flight_utm_y2]
if (x[1] - x[0]) == 0:
if (y[1]-y[0]) <= 0:
ggg = -0.001
else:
ggg = 0.001
for point in np.arange(y[0], y[1], ggg):
xx = x[0]
yy = point
dist_km = np.sqrt((seismo_utm_y-yy)**2 +(seismo_utm_x-xx)**2)
if dist_km < dist_lim:
dist_lim = dist_km
closest_point = (xx,yy)
else:
continue
else:
m = (y[1]-y[0])/(x[1]-x[0])
b = y[0] - m*x[0]
if (x[1] - x[0]) <= 0:
ggg = -0.001
else:
ggg = 0.001
for point in np.arange(x[0], x[1], ggg):
xx = point
yy = m*xx + b
dist_km = np.sqrt((seismo_utm_y-yy)**2 +(seismo_utm_x-xx)**2)
if dist_km < dist_lim:
dist_lim = dist_km
closest_point = (xx,yy)
else:
continue
return closest_point, dist_lim
def find_closest_point(flight_utm, seismo_utm):
min_distance = np.Infinity
closest_point = None
for i in range(len(flight_utm) - 1):
flight_utm_x1, flight_utm_y1 = flight_utm[i]
flight_utm_x2, flight_utm_y2 = flight_utm[i + 1]
seismo_utm_x, seismo_utm_y = seismo_utm
point, d = closest_point_on_segment(flight_utm_x1, flight_utm_y1, flight_utm_x2, flight_utm_y2, seismo_utm_x, seismo_utm_y)
if point == None:
continue
elif d < min_distance:
min_distance = d
closest_point = point
index = i
else:
continue
return closest_point, min_distance, index
# Iterate over flight files
for i, flight_file in enumerate(flight_files):
print((i/len(flight_files))*100, '%') # Print progress
# Load flight data and extract relevant columns
flight_data = pd.read_csv(flight_file, sep=",")
flight_latitudes = flight_data['latitude']
flight_longitudes = flight_data['longitude']
timestamp = flight_data['snapshot_id']
alt = flight_data['altitude']
speed = flight_data['speed']
head = flight_data['heading']
fname = filenames[i]
flight_num = fname[9:18]
date = fname[0:8]
if dist_less(flight_latitudes, flight_longitudes, seismo_latitudes, seismo_longitudes) == False:
continue
else:
# Convert flight latitude and longitude to UTM coordinates
flight_utm = [utm_proj(lon, lat) for lat, lon in zip(flight_latitudes, flight_longitudes)]
flight_utm_x, flight_utm_y = zip(*flight_utm)
# Convert UTM coordinates to kilometers
flight_utm_x_km = [x / 1000 for x in flight_utm_x]
flight_utm_y_km = [y / 1000 for y in flight_utm_y]
flight_path = [(x,y) for x, y in zip(flight_utm_x_km, flight_utm_y_km)]
# Iterate over seismometer data
for s in range(len(seismo_data)):
seismometer = (seismo_utm_x_km[s], seismo_utm_y_km[s])
station = sta[s]
closest_p, d, index= find_closest_point(flight_path, seismometer)
if d <= 2:
closest_x, closest_y = closest_p
#Calculate the time of the closest point
flight_utm_x1, flight_utm_y1 = flight_path[index]
flight_utm_x2, flight_utm_y2 = flight_path[index + 1]
x_timestamp_dif_vec = flight_utm_x2 - flight_utm_x1
y_timestamp_dif_vec = flight_utm_y2 - flight_utm_y1
cx_timestamp_dif_vec = closest_x - flight_utm_x1
cy_timestamp_dif_vec = closest_y - flight_utm_y1
line_vector = (x_timestamp_dif_vec, y_timestamp_dif_vec)
cline_vector = (cx_timestamp_dif_vec, cy_timestamp_dif_vec)
line_magnitude = np.sqrt(line_vector[0] ** 2 + line_vector[1] ** 2)
cline_magnitude = np.sqrt(cline_vector[0] ** 2 + cline_vector[1] ** 2)
length_ratio = cline_magnitude / line_magnitude
closest_time = timestamp[index] + length_ratio*(timestamp[index+1] - timestamp[index])
y = [closest_y, seismometer[1]]
x = [closest_x, seismometer[0]]
yy = sum(y) / len(y)
xx = sum(x) / len(x)
min_x = int(xx - 2)
max_x = int(xx + 2)
min_y = int(yy - 2)
max_y = int(yy + 2)
if d < 0.5:
min_x = (xx - 1)
max_x = (xx + 1)
min_y = (yy - 1)
max_y = (yy + 1)
if d < 0.1:
min_x = (xx - 0.1)
max_x = (xx + 0.1)
min_y = (yy - 0.1)
max_y = (yy + 0.1)
# Create a figure with two subplots side by side
fig, axs = plt.subplots(1, 2, gridspec_kw={'width_ratios': [1, 2]})
fig.subplots_adjust(wspace=0.5) # Adjust the spacing between subplots
axs[0].set_xticks(np.arange(int(lxmin/1000)-7, int(lxmax/1000), 50))
axs[0].set_yticks(np.arange(int(lymin/1000)-1, int(lymax/1000), 50))
axs[0].set_xlabel('UTM Easting (km)')
axs[0].set_ylabel('UTM Northing (km)')
axs[0].set_aspect('equal')
axs[1].set_aspect('equal')
axs[0].grid(True, linestyle='dotted', color='gray')
axs[1].grid(True, linestyle='dotted', color='gray')
axs[0].scatter(seismo_utm_x_km, seismo_utm_y_km, c='#e41a1c', s = 3, label='seismometers')
axs[0].plot(flight_utm_x_km, flight_utm_y_km, '-', c='#377eb8', lw=1, ms = 1, label='flight path')
for i in range(1, len(flight_utm_y_km)-1, int(len(flight_utm_y_km)/5)):
direction = np.arctan2(flight_utm_y_km[i+1] - flight_utm_y_km[i], flight_utm_x_km[i+1] - flight_utm_x_km[i])
if (flight_utm_x_km[i+1] - flight_utm_x_km[i]) == 0:
continue
m = (flight_utm_y_km[i+1] - flight_utm_y_km[i])/(flight_utm_x_km[i+1] - flight_utm_x_km[i])
if m == 0:
continue
b = flight_utm_y_km[i] - m*flight_utm_x_km[i]
axs[0].quiver((flight_utm_y_km[i]-b)/m, flight_utm_y_km[i], np.cos(direction), np.sin(direction), angles='xy', color='#377eb8', headwidth = 10)
# Set labels and title
axs[0].set_xlim(int(lxmin/1000), int(lxmax/1000))
axs[0].set_ylim(int(lymin/1000), int(lymax/1000))
axs[0].tick_params(axis='both', which='major', labelsize=9)
head_avg = (head[index]+head[index+1])/2
heading = np.deg2rad(head_avg)
# Define the UTM and latitude/longitude coordinate systems
rect = Rectangle((min_x, min_y), (max_x-min_x), (max_y-min_y), ls="-", lw = 1, ec = 'k', fc="none", zorder=2.5)
axs[0].add_patch(rect)
axs[1].set_xticks(np.arange(min_x, max_x, np.round(((max_x - min_x) / 4), 1)))
axs[1].set_yticks(np.arange(min_y, max_y, np.round(((max_y - min_y) / 4), 1)))
# Draw the zoomed in map on the second subplot
axs[1].plot(x,y, '--', c='#ff7f00')
axs[1].plot(flight_utm_x_km, flight_utm_y_km, c='#377eb8',linestyle ='dotted')
axs[1].scatter(flight_utm_x_km, flight_utm_y_km, c='#377eb8', s=20)
axs[1].set_xlim(min_x, max_x)
axs[1].set_ylim(min_y, max_y)
axs[1].tick_params(axis='both', which='major', labelsize=9)
axs[1].ticklabel_format(useOffset=False, style='plain')
direction = np.arctan2(flight_utm_y_km[index+1] - flight_utm_y_km[index], flight_utm_x_km[index+1] - flight_utm_x_km[index])
axs[1].quiver(closest_x, closest_y, np.cos(direction), np.sin(direction), angles='xy', color='#377eb8', scale=8)
axs[1].quiver(closest_x, closest_y, np.cos(heading), np.sin(heading), angles='xy', scale = 8, color='#999999')
axs[1].scatter(closest_x, closest_y, c='#377eb8', s=50, zorder=3)
axs[1].scatter(seismometer[0], seismometer[1], c='#e41a1c', s=50, zorder=3)
axs[1].text(xx,yy, str(round(d, 2))+' km', fontsize=12, fontweight='bold')
# Draw dashed lines connecting the rectangle on the existing map to the zoomed-in map
con = mpatch.ConnectionPatch(xyA=(max_x, min_y), xyB=(min_x, min_y), coordsA="data", coordsB="data", axesA=axs[0], axesB=axs[1], color="black", linestyle="--")
fig.add_artist(con)
con = mpatch.ConnectionPatch(xyA=(max_x, max_y), xyB=(min_x, max_y), coordsA="data", coordsB="data", axesA=axs[0], axesB=axs[1], color="black", linestyle="--")
fig.add_artist(con)
BASE_DIR = '/scratch/irseppi/nodal_data/plane_info/map_all_UTM/' + date + '/'+flight_num + '/' + station + '/'
make_base_dir(BASE_DIR)
plt.savefig('/scratch/irseppi/nodal_data/plane_info/map_all_UTM/' + date + '/' + flight_num + '/' + station + '/map_' + flight_num + '_' + str(closest_time) + '.png')
plt.close()
alt_avg = (alt[index]+alt[index+1])/2
alt_avg_m = alt_avg * 0.3048 #convert from feet to meters
speed_avg = (speed[index]+speed[index+1])/2
speed_avg_mps = speed_avg * 0.514444 #convert from knots to meters/sec
dist_m = d * 1000
closest_x_m = closest_x * 1000
closest_y_m = closest_y * 1000
# Write data to the output file
output.write(str(date)+ ',' + str(flight_num) + ',' + str(closest_x_m) + ',' + str(closest_y_m) + ',' + str(dist_m) + ',' +str(closest_time) + ',' + str(alt_avg_m) + ',' + str(speed_avg_mps) + ',' + str(head_avg) + ',' + str(station) + ',\n')
else:
continue
output.close()