Spaces:
Sleeping
Sleeping
Upload 23 files
Browse files- FUTURE_AQI_v1.json +1 -0
- FUTURE_AQI_v1.weights.h5 +3 -0
- app.py +244 -0
- app3.py +248 -0
- app_map.py +246 -0
- app_test.py +340 -0
- aqi_data.csv +15 -0
- aqi_data_actual_api.csv +15 -0
- requirement.txt +116 -0
- scaler_X_cpcb_4.pkl +3 -0
- scaler_y_cpcb_4.pkl +3 -0
- static/aqi_forecast_map.html +177 -0
- static/aqi_forecast_vs_actual_map.html +0 -0
- static/aqi_forecast_vs_actual_map_horizontal_bars.html +0 -0
- static/aqi_forecast_with_legend.html +203 -0
- static/aqi_map_with_legend.html +222 -0
- static/multiple_wind_speed_map.html +101 -0
- surat_AQI_testing_2024.csv +0 -0
- surat_sample_test.csv +204 -0
- templates/aqi_forecast_with_legend.html +187 -0
- templates/index.html +134 -0
- templates/multiple_wind_speed_map.html +101 -0
- templates/results.html +45 -0
FUTURE_AQI_v1.json
ADDED
@@ -0,0 +1 @@
|
|
|
|
|
1 |
+
{"module": "keras", "class_name": "Sequential", "config": {"name": "sequential_1", "trainable": true, "dtype": {"module": "keras", "class_name": "DTypePolicy", "config": {"name": "float32"}, "registered_name": null}, "layers": [{"module": "keras.layers", "class_name": "InputLayer", "config": {"batch_shape": [null, 22], "dtype": "float32", "sparse": false, "name": "input_layer_1"}, "registered_name": null}, {"module": "keras.layers", "class_name": "Dense", "config": {"name": "dense_4", "trainable": true, "dtype": {"module": "keras", "class_name": "DTypePolicy", "config": {"name": "float32"}, "registered_name": null}, "units": 128, "activation": "relu", "use_bias": true, "kernel_initializer": {"module": "keras.initializers", "class_name": "GlorotUniform", "config": {"seed": null}, "registered_name": null}, "bias_initializer": {"module": "keras.initializers", "class_name": "Zeros", "config": {}, "registered_name": null}, "kernel_regularizer": null, "bias_regularizer": null, "kernel_constraint": null, "bias_constraint": null}, "registered_name": null, "build_config": {"input_shape": [null, 22]}}, {"module": "keras.layers", "class_name": "Dense", "config": {"name": "dense_5", "trainable": true, "dtype": {"module": "keras", "class_name": "DTypePolicy", "config": {"name": "float32"}, "registered_name": null}, "units": 256, "activation": "relu", "use_bias": true, "kernel_initializer": {"module": "keras.initializers", "class_name": "GlorotUniform", "config": {"seed": null}, "registered_name": null}, "bias_initializer": {"module": "keras.initializers", "class_name": "Zeros", "config": {}, "registered_name": null}, "kernel_regularizer": {"module": "keras.regularizers", "class_name": "L2", "config": {"l2": 0.001}, "registered_name": null}, "bias_regularizer": null, "kernel_constraint": null, "bias_constraint": null}, "registered_name": null, "build_config": {"input_shape": [null, 128]}}, {"module": "keras.layers", "class_name": "Dropout", "config": {"name": "dropout_1", "trainable": true, "dtype": {"module": "keras", "class_name": "DTypePolicy", "config": {"name": "float32"}, "registered_name": null}, "rate": 0.3, "seed": null, "noise_shape": null}, "registered_name": null}, {"module": "keras.layers", "class_name": "Dense", "config": {"name": "dense_6", "trainable": true, "dtype": {"module": "keras", "class_name": "DTypePolicy", "config": {"name": "float32"}, "registered_name": null}, "units": 128, "activation": "relu", "use_bias": true, "kernel_initializer": {"module": "keras.initializers", "class_name": "GlorotUniform", "config": {"seed": null}, "registered_name": null}, "bias_initializer": {"module": "keras.initializers", "class_name": "Zeros", "config": {}, "registered_name": null}, "kernel_regularizer": {"module": "keras.regularizers", "class_name": "L2", "config": {"l2": 0.001}, "registered_name": null}, "bias_regularizer": null, "kernel_constraint": null, "bias_constraint": null}, "registered_name": null, "build_config": {"input_shape": [null, 256]}}, {"module": "keras.layers", "class_name": "Dense", "config": {"name": "dense_7", "trainable": true, "dtype": {"module": "keras", "class_name": "DTypePolicy", "config": {"name": "float32"}, "registered_name": null}, "units": 3, "activation": "linear", "use_bias": true, "kernel_initializer": {"module": "keras.initializers", "class_name": "GlorotUniform", "config": {"seed": null}, "registered_name": null}, "bias_initializer": {"module": "keras.initializers", "class_name": "Zeros", "config": {}, "registered_name": null}, "kernel_regularizer": null, "bias_regularizer": null, "kernel_constraint": null, "bias_constraint": null}, "registered_name": null, "build_config": {"input_shape": [null, 128]}}], "build_input_shape": [null, 22]}, "registered_name": null, "build_config": {"input_shape": [null, 22]}, "compile_config": {"optimizer": {"module": "keras.optimizers", "class_name": "Adam", "config": {"name": "adam", "learning_rate": 0.00010000000474974513, "weight_decay": null, "clipnorm": null, "global_clipnorm": null, "clipvalue": null, "use_ema": false, "ema_momentum": 0.99, "ema_overwrite_frequency": null, "loss_scale_factor": null, "gradient_accumulation_steps": null, "beta_1": 0.9, "beta_2": 0.999, "epsilon": 1e-07, "amsgrad": false}, "registered_name": null}, "loss": "mae", "loss_weights": null, "metrics": ["mae"], "weighted_metrics": null, "run_eagerly": false, "steps_per_execution": 1, "jit_compile": false}}
|
FUTURE_AQI_v1.weights.h5
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:73a4b6288d471be366832d6337e25ac55798beb7025943c025ed3b3c5f7e347d
|
3 |
+
size 860320
|
app.py
ADDED
@@ -0,0 +1,244 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import matplotlib
|
2 |
+
matplotlib.use('Agg') # Set non-GUI backend for Matplotlib
|
3 |
+
|
4 |
+
import tensorflow as tf
|
5 |
+
import numpy as np
|
6 |
+
import pandas as pd
|
7 |
+
import matplotlib.pyplot as plt
|
8 |
+
from flask import Flask, render_template, request
|
9 |
+
from keras.models import model_from_json
|
10 |
+
import folium
|
11 |
+
import json
|
12 |
+
import requests
|
13 |
+
from pathlib import Path
|
14 |
+
from io import BytesIO
|
15 |
+
import base64
|
16 |
+
|
17 |
+
app = Flask(__name__)
|
18 |
+
|
19 |
+
# Load the model
|
20 |
+
def load_model(name):
|
21 |
+
with open(f"{name}.json", "r") as json_file:
|
22 |
+
loaded_model_json = json_file.read()
|
23 |
+
loaded_model = model_from_json(loaded_model_json)
|
24 |
+
|
25 |
+
weights_file = f"{name}.weights.h5"
|
26 |
+
if not Path(weights_file).is_file():
|
27 |
+
raise FileNotFoundError(f"Weight file {weights_file} not found.")
|
28 |
+
|
29 |
+
loaded_model.load_weights(weights_file)
|
30 |
+
print("Loaded model from disk")
|
31 |
+
return loaded_model
|
32 |
+
|
33 |
+
#MAP CREATING
|
34 |
+
# Function to create map
|
35 |
+
def create_map(df, predicted_wind_speed, map_file):
|
36 |
+
# Extract latitude, longitude, and actual wind speed
|
37 |
+
latitude = df['lat'].iloc[0]
|
38 |
+
longitude = df['lon'].iloc[0]
|
39 |
+
actual_wind_speed = df['wind_kph'].iloc[0]
|
40 |
+
|
41 |
+
# Ensure single numeric values for plotting
|
42 |
+
if isinstance(actual_wind_speed, (list, np.ndarray)):
|
43 |
+
actual_wind_speed = actual_wind_speed[0]
|
44 |
+
if isinstance(predicted_wind_speed, (list, np.ndarray)):
|
45 |
+
predicted_wind_speed = predicted_wind_speed[0]
|
46 |
+
|
47 |
+
# Create folium map
|
48 |
+
m = folium.Map(location=[latitude, longitude], zoom_start=10)
|
49 |
+
|
50 |
+
# Create a bar chart comparing actual and predicted wind speeds
|
51 |
+
fig, ax = plt.subplots(figsize=(3, 3))
|
52 |
+
bars = ax.bar(['Actual', 'Predicted'], [actual_wind_speed, predicted_wind_speed], color=['blue', 'green'])
|
53 |
+
ax.set_ylabel('Wind Speed (km/h)')
|
54 |
+
ax.set_title('Wind Speed Comparison')
|
55 |
+
|
56 |
+
# Add labels inside each bar
|
57 |
+
for bar in bars:
|
58 |
+
yval = bar.get_height() # Get the height of each bar (wind speed value)
|
59 |
+
# Position the label inside the bar, slightly adjusted to avoid overlap with the top
|
60 |
+
ax.text(bar.get_x() + bar.get_width() / 2, yval / 2, round(yval, 2), ha='center', va='center', color='white')
|
61 |
+
|
62 |
+
# Adjust layout
|
63 |
+
plt.tight_layout()
|
64 |
+
|
65 |
+
# Save chart to a BytesIO buffer in PNG format and encode to base64
|
66 |
+
buffer = BytesIO()
|
67 |
+
plt.savefig(buffer, format="png")
|
68 |
+
plt.close(fig)
|
69 |
+
buffer.seek(0)
|
70 |
+
image_base64 = base64.b64encode(buffer.read()).decode('utf-8')
|
71 |
+
|
72 |
+
# HTML for the popup with base64 image
|
73 |
+
popup_html = f'<img src="data:image/png;base64,{image_base64}" alt="Wind Speed Comparison">'
|
74 |
+
popup = folium.Popup(html=popup_html, max_width=300)
|
75 |
+
|
76 |
+
# Add a marker to the map with popup containing the chart
|
77 |
+
folium.Marker(
|
78 |
+
location=[latitude, longitude],
|
79 |
+
popup=popup,
|
80 |
+
tooltip=df['location'].iloc[0]
|
81 |
+
).add_to(m)
|
82 |
+
|
83 |
+
# Save the folium map as an HTML file
|
84 |
+
m.save(map_file)
|
85 |
+
|
86 |
+
model = load_model('Wind_Model_New_v1')
|
87 |
+
|
88 |
+
# Route to input form
|
89 |
+
@app.route('/')
|
90 |
+
def index():
|
91 |
+
return render_template('index.html')
|
92 |
+
|
93 |
+
# Prediction route
|
94 |
+
@app.route('/predict', methods=['POST'])
|
95 |
+
def predict():
|
96 |
+
latitude = float(request.form['latitude'])
|
97 |
+
longitude = float(request.form['longitude'])
|
98 |
+
|
99 |
+
api_key = "846ca0bb2fa144cdb7195352240711"
|
100 |
+
base_url = "http://api.weatherapi.com/v1/current.json"
|
101 |
+
params = {"key": api_key, "q": f"{latitude},{longitude}"}
|
102 |
+
response = requests.get(base_url, params=params)
|
103 |
+
|
104 |
+
if response.status_code == 200:
|
105 |
+
current_weather = response.json()
|
106 |
+
extracted_data = {
|
107 |
+
'location': current_weather['location']['name'],
|
108 |
+
'lat': latitude,
|
109 |
+
'lon': longitude,
|
110 |
+
'date': current_weather['location']['localtime'].split()[0],
|
111 |
+
'time': current_weather['location']['localtime'],
|
112 |
+
'temp_c': current_weather['current']['temp_c'],
|
113 |
+
'temp_f': current_weather['current']['temp_f'],
|
114 |
+
'is_day': current_weather['current']['is_day'],
|
115 |
+
'condition': current_weather['current']['condition']['text'],
|
116 |
+
'wind_mph': current_weather['current']['wind_mph'],
|
117 |
+
'wind_kph': current_weather['current']['wind_kph'],
|
118 |
+
'humidity': current_weather['current']['humidity'],
|
119 |
+
'cloud': current_weather['current']['cloud'],
|
120 |
+
'feelslike_c': current_weather['current']['feelslike_c'],
|
121 |
+
'feelslike_f': current_weather['current']['feelslike_f'],
|
122 |
+
'windchill_c': current_weather['current']['windchill_c'],
|
123 |
+
'windchill_f': current_weather['current']['windchill_f'],
|
124 |
+
'heatindex_c': current_weather['current']['heatindex_c'],
|
125 |
+
'heatindex_f': current_weather['current']['heatindex_f'],
|
126 |
+
'dewpoint_c': current_weather['current']['dewpoint_c'],
|
127 |
+
'dewpoint_f': current_weather['current']['dewpoint_f'],
|
128 |
+
'vis_km': current_weather['current']['vis_km'],
|
129 |
+
'vis_miles': current_weather['current']['vis_miles'],
|
130 |
+
'gust_mph': current_weather['current']['gust_mph'],
|
131 |
+
'gust_kph': current_weather['current']['gust_kph'],
|
132 |
+
'uv': current_weather['current']['uv']
|
133 |
+
}
|
134 |
+
df = pd.DataFrame([extracted_data])
|
135 |
+
print("dataframe",df)
|
136 |
+
df['time'] = pd.to_datetime(df['time'])
|
137 |
+
df['day'] = df['time'].dt.day
|
138 |
+
df['month'] = df['time'].dt.month
|
139 |
+
df['hour'] = df['time'].dt.hour
|
140 |
+
|
141 |
+
dx=df[['temp_c', 'temp_f', 'is_day','wind_mph', 'wind_kph', 'humidity', 'cloud', 'feelslike_c',
|
142 |
+
'feelslike_f', 'windchill_c', 'windchill_f', 'heatindex_c','heatindex_f', 'dewpoint_c',
|
143 |
+
'dewpoint_f', 'vis_km', 'vis_miles','gust_mph', 'gust_kph', 'uv', 'day', 'month', 'hour']]
|
144 |
+
x=dx.drop(['wind_mph', 'wind_kph'],axis=1)
|
145 |
+
x_unk = np.array(x)
|
146 |
+
x_unk = x_unk / 100
|
147 |
+
predictions = model.predict(x_unk)
|
148 |
+
predicted_wind_speed = predictions[0][0]
|
149 |
+
print("predictions",predictions)
|
150 |
+
map_file = "static/multiple_wind_speed_map.html"
|
151 |
+
create_map(df, predicted_wind_speed, map_file)
|
152 |
+
|
153 |
+
return render_template('results.html',
|
154 |
+
actual=df['wind_kph'][0],
|
155 |
+
predicted=predicted_wind_speed,
|
156 |
+
map_file='multiple_wind_speed_map.html')
|
157 |
+
else:
|
158 |
+
return "Failed to retrieve weather data."
|
159 |
+
|
160 |
+
# forcast
|
161 |
+
@app.route('/forcast', methods=['POST'])
|
162 |
+
def forcast():
|
163 |
+
latitude = float(request.form['latitude'])
|
164 |
+
longitude = float(request.form['longitude'])
|
165 |
+
|
166 |
+
api_key = "846ca0bb2fa144cdb7195352240711"
|
167 |
+
base_url = "http://api.weatherapi.com/v1/forecast.json"
|
168 |
+
params = {
|
169 |
+
"key": api_key,
|
170 |
+
"q": f"{latitude},{longitude}",
|
171 |
+
"days": 1 # Fetch forecast for 1 day
|
172 |
+
}
|
173 |
+
response = requests.get(base_url, params=params)
|
174 |
+
extracted_data = {}
|
175 |
+
if response.status_code == 200:
|
176 |
+
data = response.json()
|
177 |
+
for location in data:
|
178 |
+
location_name = data['location']['name']
|
179 |
+
forecast_days = data['forecast']['forecastday']
|
180 |
+
# Loop through each day in forecastdays
|
181 |
+
for day in forecast_days:
|
182 |
+
date = day['date']
|
183 |
+
hours = day['hour']
|
184 |
+
# Extract hourly data for each hour
|
185 |
+
for hour_data in hours:
|
186 |
+
# Add extra details (date, location) for each hourly data
|
187 |
+
extracted_data={
|
188 |
+
"location": location_name,
|
189 |
+
"lat": latitude,
|
190 |
+
"lon": longitude,
|
191 |
+
"date": date,
|
192 |
+
"time": hour_data["time"],
|
193 |
+
"temp_c": hour_data["temp_c"],
|
194 |
+
"temp_f": hour_data["temp_f"],
|
195 |
+
"is_day": hour_data["is_day"],
|
196 |
+
"condition": hour_data["condition"]["text"],
|
197 |
+
"wind_mph": hour_data["wind_mph"],
|
198 |
+
"wind_kph": hour_data["wind_kph"],
|
199 |
+
"humidity": hour_data["humidity"],
|
200 |
+
"cloud": hour_data["cloud"],
|
201 |
+
"feelslike_c": hour_data["feelslike_c"],
|
202 |
+
"feelslike_f": hour_data["feelslike_f"],
|
203 |
+
"windchill_c": hour_data["windchill_c"],
|
204 |
+
"windchill_f": hour_data["windchill_f"],
|
205 |
+
"heatindex_c": hour_data["heatindex_c"],
|
206 |
+
"heatindex_f": hour_data["heatindex_f"],
|
207 |
+
"dewpoint_c": hour_data["dewpoint_c"],
|
208 |
+
"dewpoint_f": hour_data["dewpoint_f"],
|
209 |
+
"vis_km": hour_data["vis_km"],
|
210 |
+
"vis_miles": hour_data["vis_miles"],
|
211 |
+
"gust_mph": hour_data["gust_mph"],
|
212 |
+
"gust_kph": hour_data["gust_kph"],
|
213 |
+
"uv": hour_data["uv"]
|
214 |
+
}
|
215 |
+
|
216 |
+
# Convert extracted data into a DataFrame
|
217 |
+
df = pd.DataFrame([extracted_data])
|
218 |
+
print("dataframe", df)
|
219 |
+
# Extracting additional date/time features
|
220 |
+
df['time']=pd.to_datetime(df['time'])
|
221 |
+
df['datetime']=pd.to_datetime(df['time'])
|
222 |
+
df['day'] = df['datetime'].dt.day # Extracts day of the month
|
223 |
+
df['month'] = df['datetime'].dt.month # Extracts month
|
224 |
+
df['hour'] = df['datetime'].dt.hour
|
225 |
+
# Prepare features for prediction
|
226 |
+
dx=df[['temp_c', 'temp_f', 'is_day','wind_mph', 'wind_kph', 'humidity', 'cloud', 'feelslike_c','feelslike_f', 'windchill_c', 'windchill_f', 'heatindex_c','heatindex_f', 'dewpoint_c', 'dewpoint_f', 'vis_km', 'vis_miles','gust_mph', 'gust_kph', 'uv', 'day', 'month', 'hour']]
|
227 |
+
x = dx.drop(['wind_mph', 'wind_kph'], axis=1)
|
228 |
+
x_unk = np.array(x)
|
229 |
+
x_unk = x_unk / 110
|
230 |
+
predictions = model.predict(x_unk)
|
231 |
+
predicted_wind_speed = predictions[0][0]
|
232 |
+
print("predictions",predictions)
|
233 |
+
map_file = "static/multiple_wind_speed_map.html"
|
234 |
+
create_map(df, predicted_wind_speed, map_file)
|
235 |
+
|
236 |
+
return render_template('results.html',
|
237 |
+
actual=df['wind_kph'][0],
|
238 |
+
predicted=predicted_wind_speed,
|
239 |
+
map_file='multiple_wind_speed_map.html')
|
240 |
+
else:
|
241 |
+
return "Failed to retrieve weather data."
|
242 |
+
|
243 |
+
if __name__ == '__main__':
|
244 |
+
app.run(debug=True)
|
app3.py
ADDED
@@ -0,0 +1,248 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from flask import Flask, render_template, request, redirect, url_for
|
2 |
+
import pandas as pd
|
3 |
+
import numpy as np
|
4 |
+
import joblib
|
5 |
+
import requests
|
6 |
+
from keras.models import model_from_json
|
7 |
+
import folium
|
8 |
+
import matplotlib.pyplot as plt
|
9 |
+
from io import BytesIO
|
10 |
+
import base64
|
11 |
+
import os
|
12 |
+
from collections import defaultdict
|
13 |
+
|
14 |
+
app = Flask(__name__)
|
15 |
+
|
16 |
+
# Load model and scalers
|
17 |
+
def load_model(name):
|
18 |
+
with open(f"{name}.json", "r") as json_file:
|
19 |
+
loaded_model_json = json_file.read()
|
20 |
+
model = model_from_json(loaded_model_json)
|
21 |
+
model.load_weights(f"{name}.weights.h5")
|
22 |
+
return model
|
23 |
+
|
24 |
+
model = load_model("FUTURE_AQI_v1")
|
25 |
+
scaler_X = joblib.load('scaler_X_cpcb_4.pkl')
|
26 |
+
scaler_y = joblib.load('scaler_y_cpcb_4.pkl')
|
27 |
+
|
28 |
+
API_KEY = "26daca1b78f44099a755b921be4bfcf1"
|
29 |
+
|
30 |
+
@app.route('/')
|
31 |
+
def index():
|
32 |
+
return render_template('index.html')
|
33 |
+
|
34 |
+
@app.route('/forecast', methods=['POST'])
|
35 |
+
def forecast():
|
36 |
+
# Get user input
|
37 |
+
latitude = float(request.form['latitude'])
|
38 |
+
longitude = float(request.form['longitude'])
|
39 |
+
|
40 |
+
# Fetch current AQI from API
|
41 |
+
current_url = f"https://api.weatherbit.io/v2.0/current/airquality?lat={latitude}&lon={longitude}&key={API_KEY}"
|
42 |
+
response = requests.get(current_url)
|
43 |
+
if response.status_code == 200:
|
44 |
+
current_data = response.json()['data'][0]
|
45 |
+
|
46 |
+
# Prepare input for the model
|
47 |
+
now = pd.to_datetime("now")
|
48 |
+
input_data = pd.DataFrame([{
|
49 |
+
'PM2.5': current_data['pm25'],
|
50 |
+
'PM10': current_data['pm10'],
|
51 |
+
'NO2': current_data['no2'],
|
52 |
+
'SO2': current_data['so2'],
|
53 |
+
'CO': current_data['co'],
|
54 |
+
'AQI': current_data['aqi'],
|
55 |
+
'Day': now.day,
|
56 |
+
'Month': now.month,
|
57 |
+
'Hour': now.hour
|
58 |
+
}])
|
59 |
+
|
60 |
+
# Scale and predict
|
61 |
+
input_scaled = scaler_X.transform(input_data)
|
62 |
+
predictions = model.predict(input_scaled)
|
63 |
+
predictions_actual = scaler_y.inverse_transform(predictions)
|
64 |
+
|
65 |
+
# Fetch forecasted AQI from API
|
66 |
+
forecast_url = f"https://api.weatherbit.io/v2.0/forecast/airquality?lat={latitude}&lon={longitude}&key={API_KEY}"
|
67 |
+
response = requests.get(forecast_url)
|
68 |
+
forecast_data = response.json()['data']
|
69 |
+
|
70 |
+
grouped_aqi = defaultdict(list)
|
71 |
+
for entry in forecast_data:
|
72 |
+
date = entry['datetime'].split(':')[0]
|
73 |
+
grouped_aqi[date].append(entry['aqi'])
|
74 |
+
|
75 |
+
api_predictions = {date: max(values) for date, values in grouped_aqi.items()}
|
76 |
+
|
77 |
+
# Save results to CSV
|
78 |
+
forecast_df = pd.DataFrame([{
|
79 |
+
**input_data.iloc[0],
|
80 |
+
'lat': latitude,
|
81 |
+
'lon': longitude,
|
82 |
+
'AQI_step_1': predictions_actual[0, 0],
|
83 |
+
'AQI_step_2': predictions_actual[0, 1],
|
84 |
+
'AQI_step_3': predictions_actual[0, 2]
|
85 |
+
}])
|
86 |
+
forecast_df.to_csv('aqi_data.csv', mode='a', header=False, index=False)
|
87 |
+
|
88 |
+
api_df = pd.DataFrame([{
|
89 |
+
'AQI_currrent_API': current_data['aqi'],
|
90 |
+
'AQI_step_1_API': api_predictions.get(list(api_predictions.keys())[0], None),
|
91 |
+
'AQI_step_2_API': api_predictions.get(list(api_predictions.keys())[1], None),
|
92 |
+
'AQI_step_3_API': api_predictions.get(list(api_predictions.keys())[2], None)
|
93 |
+
}])
|
94 |
+
api_df.to_csv('aqi_data_actual_api.csv', mode='a', header=False, index=False)
|
95 |
+
|
96 |
+
# Generate updated map and return
|
97 |
+
generate_map()
|
98 |
+
return redirect(url_for('result'))
|
99 |
+
|
100 |
+
@app.route('/result')
|
101 |
+
def result():
|
102 |
+
return render_template('result.html')
|
103 |
+
|
104 |
+
def generate_map():
|
105 |
+
# Load data
|
106 |
+
df1 = pd.read_csv('aqi_data.csv')
|
107 |
+
df2 = pd.read_csv('aqi_data_actual_api.csv')
|
108 |
+
data = pd.concat([df1, df2], axis=1)
|
109 |
+
|
110 |
+
# Create Folium map
|
111 |
+
# Create the Folium map
|
112 |
+
map_center = [data['lat'].mean(), data['lon'].mean()]
|
113 |
+
m = folium.Map(location=map_center, zoom_start=10)
|
114 |
+
|
115 |
+
# AQI Color Legend
|
116 |
+
legend_html = """
|
117 |
+
<div style="
|
118 |
+
position: fixed;
|
119 |
+
bottom: 20px; left: 20px; width: 350px; height: 225px;
|
120 |
+
background-color: white;
|
121 |
+
z-index:9999; font-size:14px; border:2px solid grey;
|
122 |
+
padding: 10px; overflow-y: auto;">
|
123 |
+
<b>AQI Color Legend</b>
|
124 |
+
<table style="width: 100%; border-collapse: collapse; text-align: left;">
|
125 |
+
<thead>
|
126 |
+
<tr style="border-bottom: 2px solid grey;">
|
127 |
+
<th style="padding: 5px;">Color</th>
|
128 |
+
<th style="padding: 5px;">Remark</th>
|
129 |
+
<th style="padding: 5px;">Range</th>
|
130 |
+
</tr>
|
131 |
+
</thead>
|
132 |
+
<tbody>
|
133 |
+
<tr>
|
134 |
+
<td><i style="background:green; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
135 |
+
<td>Good</td>
|
136 |
+
<td>0-50</td>
|
137 |
+
</tr>
|
138 |
+
<tr>
|
139 |
+
<td><i style="background:yellow; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
140 |
+
<td>Moderate</td>
|
141 |
+
<td>51-100</td>
|
142 |
+
</tr>
|
143 |
+
<tr>
|
144 |
+
<td><i style="background:orange; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
145 |
+
<td>Unhealthy for Sensitive Groups</td>
|
146 |
+
<td>101-150</td>
|
147 |
+
</tr>
|
148 |
+
<tr>
|
149 |
+
<td><i style="background:red; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
150 |
+
<td>Unhealthy</td>
|
151 |
+
<td>151-200</td>
|
152 |
+
</tr>
|
153 |
+
<tr>
|
154 |
+
<td><i style="background:purple; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
155 |
+
<td>Very Unhealthy</td>
|
156 |
+
<td>201-300</td>
|
157 |
+
</tr>
|
158 |
+
<tr>
|
159 |
+
<td><i style="background:maroon; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
160 |
+
<td>Hazardous</td>
|
161 |
+
<td>301+</td>
|
162 |
+
</tr>
|
163 |
+
</tbody>
|
164 |
+
</table>
|
165 |
+
</div>
|
166 |
+
"""
|
167 |
+
m.get_root().html.add_child(folium.Element(legend_html))
|
168 |
+
|
169 |
+
for _, row in data.iterrows():
|
170 |
+
popup_html = create_plot(row)
|
171 |
+
color = get_color_for_aqi(row['AQI_step_1'])
|
172 |
+
folium.Marker(
|
173 |
+
location=[row["lat"], row["lon"]],
|
174 |
+
popup=folium.Popup(html=popup_html, max_width=500),
|
175 |
+
icon=folium.Icon(color=color)
|
176 |
+
).add_to(m)
|
177 |
+
|
178 |
+
# Save the map
|
179 |
+
m.save('static/aqi_forecast_with_legend.html')
|
180 |
+
|
181 |
+
def create_plot(data):
|
182 |
+
# Bar plot generation logic (same as before)
|
183 |
+
fig, ax = plt.subplots(figsize=(5, 2))
|
184 |
+
categories = ['DAY 1', 'DAY 2', 'DAY 3']
|
185 |
+
actual_values = [data['AQI_step_1'], data['AQI_step_2'], data['AQI_step_3']]
|
186 |
+
api_values = [data['AQI_step_1_API'], data['AQI_step_2_API'], data['AQI_step_3_API']]
|
187 |
+
|
188 |
+
bar_width = 0.35
|
189 |
+
index = range(len(categories))
|
190 |
+
|
191 |
+
# Plot horizontal bars
|
192 |
+
bars_actual = ax.barh(index, actual_values, bar_width, label="Model AQI", color='blue')
|
193 |
+
bars_api = ax.barh([i + bar_width for i in index], api_values, bar_width, label="API AQI", color='green')
|
194 |
+
|
195 |
+
# Add values to each bar
|
196 |
+
max_value = 0 # Track the maximum value for axis limit adjustment
|
197 |
+
for bar in bars_actual:
|
198 |
+
value = bar.get_width()
|
199 |
+
ax.text(value + 2, bar.get_y() + bar.get_height() / 2,
|
200 |
+
f'{value:.1f}', va='center', fontsize=10)
|
201 |
+
max_value = max(max_value, value)
|
202 |
+
for bar in bars_api:
|
203 |
+
value = bar.get_width()
|
204 |
+
ax.text(value + 2, bar.get_y() + bar.get_height() / 2,
|
205 |
+
f'{value:.1f}', va='center', fontsize=10)
|
206 |
+
max_value = max(max_value, value)
|
207 |
+
|
208 |
+
# Adjust x-axis limits to accommodate annotations
|
209 |
+
ax.set_xlim(0, max_value * 1.2)
|
210 |
+
|
211 |
+
# Customize y-ticks and labels
|
212 |
+
ax.set_yticks([i + bar_width / 2 for i in index])
|
213 |
+
ax.set_yticklabels(categories)
|
214 |
+
ax.set_xlabel('AQI')
|
215 |
+
ax.set_title('AQI Comparison')
|
216 |
+
|
217 |
+
# Place legend outside the plot area
|
218 |
+
ax.legend(loc='center left', bbox_to_anchor=(1, 0.5), frameon=False)
|
219 |
+
|
220 |
+
plt.tight_layout()
|
221 |
+
|
222 |
+
# Save the plot to a PNG image in memory
|
223 |
+
buffer = BytesIO()
|
224 |
+
plt.savefig(buffer, format="png", bbox_inches='tight')
|
225 |
+
plt.close(fig)
|
226 |
+
buffer.seek(0)
|
227 |
+
|
228 |
+
# Encode the image to base64 to embed it in the HTML
|
229 |
+
image_base64 = base64.b64encode(buffer.read()).decode()
|
230 |
+
return f'<img src="data:image/png;base64,{image_base64}">'
|
231 |
+
|
232 |
+
def get_color_for_aqi(aqi_value):
|
233 |
+
# Color logic (same as before)
|
234 |
+
if aqi_value <= 50:
|
235 |
+
return 'green'
|
236 |
+
elif aqi_value <= 100:
|
237 |
+
return 'yellow'
|
238 |
+
elif aqi_value <= 150:
|
239 |
+
return 'orange'
|
240 |
+
elif aqi_value <= 200:
|
241 |
+
return 'red'
|
242 |
+
elif aqi_value <= 300:
|
243 |
+
return 'purple'
|
244 |
+
else:
|
245 |
+
return 'maroon'
|
246 |
+
|
247 |
+
if __name__ == '__main__':
|
248 |
+
app.run(debug=True)
|
app_map.py
ADDED
@@ -0,0 +1,246 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from flask import Flask, render_template, request, redirect, url_for
|
2 |
+
import pandas as pd
|
3 |
+
import numpy as np
|
4 |
+
import joblib
|
5 |
+
import requests
|
6 |
+
from keras.models import model_from_json
|
7 |
+
import folium
|
8 |
+
from collections import defaultdict
|
9 |
+
import matplotlib
|
10 |
+
matplotlib.use('Agg') # Use non-GUI backend
|
11 |
+
import matplotlib.pyplot as plt
|
12 |
+
from io import BytesIO
|
13 |
+
import base64
|
14 |
+
import branca
|
15 |
+
|
16 |
+
app = Flask(__name__)
|
17 |
+
|
18 |
+
# Load model and scalers
|
19 |
+
def load_model(name):
|
20 |
+
with open(f"{name}.json", "r") as json_file:
|
21 |
+
loaded_model_json = json_file.read()
|
22 |
+
model = model_from_json(loaded_model_json)
|
23 |
+
model.load_weights(f"{name}.weights.h5")
|
24 |
+
return model
|
25 |
+
|
26 |
+
model = load_model("FUTURE_AQI_v1")
|
27 |
+
scaler_X = joblib.load('scaler_X_cpcb_4.pkl')
|
28 |
+
scaler_y = joblib.load('scaler_y_cpcb_4.pkl')
|
29 |
+
|
30 |
+
API_KEY = "26daca1b78f44099a755b921be4bfcf1"
|
31 |
+
|
32 |
+
# Route to display the form and the map
|
33 |
+
@app.route('/', methods=['GET', 'POST'])
|
34 |
+
def index():
|
35 |
+
map_html = generate_map()
|
36 |
+
if request.method == 'POST':
|
37 |
+
# Get user input from form
|
38 |
+
latitude = float(request.form['latitude'])
|
39 |
+
longitude = float(request.form['longitude'])
|
40 |
+
|
41 |
+
# Fetch current AQI from API
|
42 |
+
current_url = f"https://api.weatherbit.io/v2.0/current/airquality?lat={latitude}&lon={longitude}&key={API_KEY}"
|
43 |
+
response = requests.get(current_url)
|
44 |
+
if response.status_code == 200:
|
45 |
+
current_data = response.json()['data'][0]
|
46 |
+
|
47 |
+
# Prepare input for the model
|
48 |
+
now = pd.to_datetime("now")
|
49 |
+
input_data = pd.DataFrame([{
|
50 |
+
'PM2.5': current_data['pm25'],
|
51 |
+
'PM10': current_data['pm10'],
|
52 |
+
'NO2': current_data['no2'],
|
53 |
+
'SO2': current_data['so2'],
|
54 |
+
'CO': current_data['co'],
|
55 |
+
'AQI': current_data['aqi'],
|
56 |
+
'Day': now.day,
|
57 |
+
'Month': now.month,
|
58 |
+
'Hour': now.hour
|
59 |
+
}])
|
60 |
+
|
61 |
+
# Scale and predict
|
62 |
+
input_scaled = scaler_X.transform(input_data)
|
63 |
+
predictions = model.predict(input_scaled)
|
64 |
+
predictions_actual = scaler_y.inverse_transform(predictions)
|
65 |
+
|
66 |
+
# Save results to CSV
|
67 |
+
forecast_df = pd.DataFrame([{
|
68 |
+
**input_data.iloc[0],
|
69 |
+
'lat': latitude,
|
70 |
+
'lon': longitude,
|
71 |
+
'AQI_step_1': predictions_actual[0, 0],
|
72 |
+
'AQI_step_2': predictions_actual[0, 1],
|
73 |
+
'AQI_step_3': predictions_actual[0, 2]
|
74 |
+
}])
|
75 |
+
forecast_df.to_csv('aqi_data.csv', mode='a', header=False, index=False)
|
76 |
+
|
77 |
+
# Generate the updated map
|
78 |
+
map_html = generate_map()
|
79 |
+
|
80 |
+
return render_template('aqi_forecast_with_legend.html', map_html=map_html)
|
81 |
+
|
82 |
+
# Route to handle the forecast data submission and AQI prediction
|
83 |
+
@app.route('/forecast', methods=['POST'])
|
84 |
+
def forecast():
|
85 |
+
# Get user input from form
|
86 |
+
latitude = float(request.form['latitude'])
|
87 |
+
longitude = float(request.form['longitude'])
|
88 |
+
|
89 |
+
# Fetch current AQI from API
|
90 |
+
current_url = f"https://api.weatherbit.io/v2.0/current/airquality?lat={latitude}&lon={longitude}&key={API_KEY}"
|
91 |
+
response = requests.get(current_url)
|
92 |
+
if response.status_code == 200:
|
93 |
+
current_data = response.json()['data'][0]
|
94 |
+
|
95 |
+
# Prepare input for the model
|
96 |
+
now = pd.to_datetime("now")
|
97 |
+
input_data = pd.DataFrame([{
|
98 |
+
'PM2.5': current_data['pm25'],
|
99 |
+
'PM10': current_data['pm10'],
|
100 |
+
'NO2': current_data['no2'],
|
101 |
+
'SO2': current_data['so2'],
|
102 |
+
'CO': current_data['co'],
|
103 |
+
'AQI': current_data['aqi'],
|
104 |
+
'Day': now.day,
|
105 |
+
'Month': now.month,
|
106 |
+
'Hour': now.hour
|
107 |
+
}])
|
108 |
+
|
109 |
+
# Scale and predict
|
110 |
+
input_scaled = scaler_X.transform(input_data)
|
111 |
+
predictions = model.predict(input_scaled)
|
112 |
+
predictions_actual = scaler_y.inverse_transform(predictions)
|
113 |
+
|
114 |
+
# Fetch forecasted AQI from API
|
115 |
+
forecast_url = f"https://api.weatherbit.io/v2.0/forecast/airquality?lat={latitude}&lon={longitude}&key={API_KEY}"
|
116 |
+
response = requests.get(forecast_url)
|
117 |
+
forecast_data = response.json()['data']
|
118 |
+
|
119 |
+
# Group AQI data by date
|
120 |
+
grouped_aqi = defaultdict(list)
|
121 |
+
for entry in forecast_data:
|
122 |
+
date = entry['datetime'].split(':')[0]
|
123 |
+
grouped_aqi[date].append(entry['aqi'])
|
124 |
+
|
125 |
+
# Get the maximum AQI forecasted for the next days
|
126 |
+
api_predictions = {date: max(values) for date, values in grouped_aqi.items()}
|
127 |
+
|
128 |
+
# Save the forecasted AQI results to CSV
|
129 |
+
api_df = pd.DataFrame([{
|
130 |
+
'AQI_currrent_API': current_data['aqi'],
|
131 |
+
'AQI_step_1_API': api_predictions.get(list(api_predictions.keys())[0], None),
|
132 |
+
'AQI_step_2_API': api_predictions.get(list(api_predictions.keys())[1], None),
|
133 |
+
'AQI_step_3_API': api_predictions.get(list(api_predictions.keys())[2], None)
|
134 |
+
}])
|
135 |
+
api_df.to_csv('aqi_data_actual_api.csv', mode='a', header=False, index=False)
|
136 |
+
|
137 |
+
# Save the predicted results to CSV
|
138 |
+
forecast_df = pd.DataFrame([{
|
139 |
+
**input_data.iloc[0],
|
140 |
+
'lat': latitude,
|
141 |
+
'lon': longitude,
|
142 |
+
'AQI_step_1': predictions_actual[0, 0],
|
143 |
+
'AQI_step_2': predictions_actual[0, 1],
|
144 |
+
'AQI_step_3': predictions_actual[0, 2]
|
145 |
+
}])
|
146 |
+
forecast_df.to_csv('aqi_data.csv', mode='a', header=False, index=False)
|
147 |
+
|
148 |
+
# Generate the updated map
|
149 |
+
map_html = generate_map()
|
150 |
+
|
151 |
+
return render_template('aqi_forecast_with_legend.html', map_html=map_html)
|
152 |
+
|
153 |
+
def generate_map(output_file='templates/aqi_forecast_with_legend.html'):
|
154 |
+
# Load data
|
155 |
+
df1 = pd.read_csv('aqi_data.csv', on_bad_lines='skip')
|
156 |
+
df2 = pd.read_csv('aqi_data_actual_api.csv', on_bad_lines='skip')
|
157 |
+
data = pd.concat([df1, df2], axis=1)
|
158 |
+
|
159 |
+
# Create Folium map
|
160 |
+
map_center = [data['lat'].mean(), data['lon'].mean()]
|
161 |
+
m = folium.Map(location=map_center, zoom_start=10)
|
162 |
+
|
163 |
+
# Add AQI data points
|
164 |
+
for _, row in data.iterrows():
|
165 |
+
popup_html = create_plot(row)
|
166 |
+
color = get_color_for_aqi(row['AQI_step_1'])
|
167 |
+
if color: # Ensure a valid color is returned
|
168 |
+
folium.Marker(
|
169 |
+
location=[row["lat"], row["lon"]],
|
170 |
+
popup=folium.Popup(html=popup_html, max_width=500),
|
171 |
+
icon=folium.Icon(color=color)
|
172 |
+
).add_to(m)
|
173 |
+
|
174 |
+
# Return map as an HTML string
|
175 |
+
map_html = m._repr_html_()
|
176 |
+
|
177 |
+
# Save the map
|
178 |
+
#m.save(output_file)
|
179 |
+
return map_html
|
180 |
+
|
181 |
+
def get_color_for_aqi(aqi_value):
|
182 |
+
if aqi_value <= 50:
|
183 |
+
return 'green'
|
184 |
+
elif aqi_value <= 100:
|
185 |
+
return 'lightgreen'
|
186 |
+
elif aqi_value <= 150:
|
187 |
+
return 'orange'
|
188 |
+
elif aqi_value <= 200:
|
189 |
+
return 'red'
|
190 |
+
elif aqi_value <= 300:
|
191 |
+
return 'purple'
|
192 |
+
else:
|
193 |
+
return 'gray' # Supported Folium color
|
194 |
+
|
195 |
+
def create_plot(data):
|
196 |
+
fig, ax = plt.subplots(figsize=(5, 2))
|
197 |
+
categories = ['DAY 1', 'DAY 2', 'DAY 3']
|
198 |
+
actual_values = [data['AQI_step_1'], data['AQI_step_2'], data['AQI_step_3']]
|
199 |
+
api_values = [data['AQI_step_1_API'], data['AQI_step_2_API'], data['AQI_step_3_API']]
|
200 |
+
|
201 |
+
bar_width = 0.35
|
202 |
+
index = range(len(categories))
|
203 |
+
|
204 |
+
# Plot horizontal bars
|
205 |
+
bars_actual = ax.barh(index, actual_values, bar_width, label="Model AQI", color='blue')
|
206 |
+
bars_api = ax.barh([i + bar_width for i in index], api_values, bar_width, label="API AQI", color='green')
|
207 |
+
|
208 |
+
# Add values to each bar
|
209 |
+
max_value = 0 # Track the maximum value for axis limit adjustment
|
210 |
+
for bar in bars_actual:
|
211 |
+
value = bar.get_width()
|
212 |
+
ax.text(value + 2, bar.get_y() + bar.get_height() / 2,
|
213 |
+
f'{value:.1f}', va='center', fontsize=10)
|
214 |
+
max_value = max(max_value, value)
|
215 |
+
for bar in bars_api:
|
216 |
+
value = bar.get_width()
|
217 |
+
ax.text(value + 2, bar.get_y() + bar.get_height() / 2,
|
218 |
+
f'{value:.1f}', va='center', fontsize=10)
|
219 |
+
max_value = max(max_value, value)
|
220 |
+
|
221 |
+
# Adjust x-axis limits to accommodate annotations
|
222 |
+
ax.set_xlim(0, max_value * 1.2)
|
223 |
+
|
224 |
+
# Customize y-ticks and labels
|
225 |
+
ax.set_yticks([i + bar_width / 2 for i in index])
|
226 |
+
ax.set_yticklabels(categories)
|
227 |
+
ax.set_xlabel('AQI')
|
228 |
+
ax.set_title('AQI Comparison')
|
229 |
+
|
230 |
+
# Place legend outside the plot area
|
231 |
+
ax.legend(loc='center left', bbox_to_anchor=(1, 0.5), frameon=False)
|
232 |
+
|
233 |
+
plt.tight_layout()
|
234 |
+
|
235 |
+
# Save the plot to a PNG image in memory
|
236 |
+
buffer = BytesIO()
|
237 |
+
plt.savefig(buffer, format="png", bbox_inches='tight')
|
238 |
+
plt.close(fig)
|
239 |
+
buffer.seek(0)
|
240 |
+
|
241 |
+
# Encode the image to base64 to embed it in the HTML
|
242 |
+
image_base64 = base64.b64encode(buffer.read()).decode()
|
243 |
+
return f'<img src="data:image/png;base64,{image_base64}">'
|
244 |
+
|
245 |
+
if __name__ == '__main__':
|
246 |
+
app.run(debug=True)
|
app_test.py
ADDED
@@ -0,0 +1,340 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import tensorflow as tf
|
2 |
+
import numpy as np
|
3 |
+
import pandas as pd
|
4 |
+
import matplotlib.pyplot as plt
|
5 |
+
from pathlib import Path
|
6 |
+
from keras.models import model_from_json
|
7 |
+
import tensorflow as tf
|
8 |
+
import matplotlib.pyplot as plt
|
9 |
+
import joblib
|
10 |
+
import requests
|
11 |
+
import json
|
12 |
+
from datetime import datetime
|
13 |
+
|
14 |
+
def load_model(name):
|
15 |
+
# Load JSON and create model
|
16 |
+
json_file = open("%s.json" % name, "r")
|
17 |
+
loaded_model_json = json_file.read()
|
18 |
+
json_file.close()
|
19 |
+
loaded_model = model_from_json(loaded_model_json)
|
20 |
+
|
21 |
+
# Check if the weights file exists before loading
|
22 |
+
weights_file = f"{name}.weights.h5"
|
23 |
+
if not Path(weights_file).is_file():
|
24 |
+
raise FileNotFoundError(f"Weight file {weights_file} not found.")
|
25 |
+
|
26 |
+
# Load weights into the new model
|
27 |
+
loaded_model.load_weights(weights_file)
|
28 |
+
print("Loaded model from disk")
|
29 |
+
return loaded_model
|
30 |
+
|
31 |
+
model = load_model("3_day_forecast_AQI_v5")
|
32 |
+
|
33 |
+
####################################
|
34 |
+
|
35 |
+
# Load the scalers
|
36 |
+
scaler_X = joblib.load('scaler_X_AQI.pkl')
|
37 |
+
scaler_y = joblib.load('scaler_y_AQI.pkl')
|
38 |
+
|
39 |
+
import requests
|
40 |
+
import pandas as pd
|
41 |
+
import joblib
|
42 |
+
import os
|
43 |
+
from datetime import datetime
|
44 |
+
# delhi 28.639638713652012, 77.19002000205269
|
45 |
+
# bhopal 23.23731292701139, 77.44433463788636
|
46 |
+
# ahemdabad 23.0364012974141, 72.58238347964425
|
47 |
+
# ankleshwar 21.62880896774956, 73.0043990197163
|
48 |
+
# jamnagar 22.3033564155508, 70.8012921707898
|
49 |
+
# 21.22050672027795 72.83355967457062#
|
50 |
+
# 21.236796371788703, 72.8665479925569
|
51 |
+
|
52 |
+
# Define API parameters
|
53 |
+
api_key = "26daca1b78f44099a755b921be4bfcf1" # Your WeatherAPI key
|
54 |
+
latitude = 21.236796371788703 # Example latitude
|
55 |
+
longitude = 72.8665479925569 # # Example longitude
|
56 |
+
base_url = f"https://api.weatherbit.io/v2.0/current/airquality?lat={latitude}&lon={longitude}&key={api_key}"
|
57 |
+
|
58 |
+
# Make the API request
|
59 |
+
response = requests.get(base_url)
|
60 |
+
if response.status_code == 200:
|
61 |
+
data = response.json()
|
62 |
+
|
63 |
+
# Extract forecast data
|
64 |
+
dx = [data['data'][0]]
|
65 |
+
test = pd.DataFrame(dx)
|
66 |
+
|
67 |
+
# Add time-based features
|
68 |
+
now = datetime.now()
|
69 |
+
current_time = now.strftime("%Y-%m-%d %H:%M:%S")
|
70 |
+
test = test[['pm25', 'pm10', 'no2', 'so2', 'co', 'aqi']]
|
71 |
+
test['Date'] = pd.to_datetime(current_time)
|
72 |
+
test['Day'] = test['Date'].dt.day
|
73 |
+
test['Month'] = test['Date'].dt.month
|
74 |
+
test['Hour'] = test['Date'].dt.hour
|
75 |
+
test = test[['pm25', 'pm10', 'no2', 'so2', 'co', 'aqi', 'Day', 'Month', 'Hour']]
|
76 |
+
test.columns = ['PM2.5', 'PM10', 'NO2', 'SO2', 'CO', 'AQI', 'Day', 'Month', 'Hour']
|
77 |
+
|
78 |
+
# Load scalers
|
79 |
+
scaler_X = joblib.load('scaler_X_AQI.pkl')
|
80 |
+
scaler_y = joblib.load('scaler_y_AQI.pkl')
|
81 |
+
|
82 |
+
# Standardize data and make predictions
|
83 |
+
data_normalized = scaler_X.transform(test)
|
84 |
+
prediction_Test = model.predict(data_normalized)
|
85 |
+
predictions_actual = scaler_y.inverse_transform(prediction_Test)
|
86 |
+
test['lat']= latitude
|
87 |
+
test['lon']= longitude
|
88 |
+
# Create a DataFrame for predictions
|
89 |
+
pred = pd.DataFrame(predictions_actual, columns=['AQI_step_1', 'AQI_step_2', 'AQI_step_3'])
|
90 |
+
df = pd.concat([test, pred], axis=1)
|
91 |
+
|
92 |
+
# Define the CSV file path
|
93 |
+
csv_file_path = "aqi_data.csv"
|
94 |
+
|
95 |
+
# Create the CSV file with headers if it doesn't exist
|
96 |
+
if not os.path.exists(csv_file_path):
|
97 |
+
columns = ['PM2.5', 'PM10', 'NO2', 'SO2', 'CO', 'AQI', 'Day', 'Month', 'Hour', 'lat', 'lon', 'AQI_step_1', 'AQI_step_2', 'AQI_step_3']
|
98 |
+
df_empty = pd.DataFrame(columns=columns)
|
99 |
+
df_empty.to_csv(csv_file_path, index=False)
|
100 |
+
|
101 |
+
# Append new data to the existing CSV
|
102 |
+
df.to_csv(csv_file_path, mode='a', index=False, header=False)
|
103 |
+
|
104 |
+
print(f"Data appended to {csv_file_path}")
|
105 |
+
|
106 |
+
|
107 |
+
####################################
|
108 |
+
|
109 |
+
|
110 |
+
import requests
|
111 |
+
import json
|
112 |
+
from datetime import datetime
|
113 |
+
|
114 |
+
# Define API parameters
|
115 |
+
api_key = "26daca1b78f44099a755b921be4bfcf1" # Your WeatherAPI key
|
116 |
+
# 21.195069775800516,72.79324648126439
|
117 |
+
# 21.22050672027795, 72.83355967457062
|
118 |
+
# 28.639638713652012,77.19002000205269
|
119 |
+
# 23.23731292701139,77.44433463788636,
|
120 |
+
# 23.0364012974141,72.58238347964425,
|
121 |
+
# 21.62880896774956,73.0043990197163,
|
122 |
+
# jamnagar 22.3033564155508, 70.8012921707898
|
123 |
+
# new delhi 28.619913380208967, 77.20633325621425
|
124 |
+
|
125 |
+
latitude = 21.236796371788703 # Example latitude
|
126 |
+
longitude = 72.8665479925569 # # Example longitude
|
127 |
+
|
128 |
+
base_url = f"https://api.weatherbit.io/v2.0/forecast/airquality?lat={latitude}&lon={longitude}&key={api_key}"
|
129 |
+
|
130 |
+
# Make the API request
|
131 |
+
response = requests.get(base_url)
|
132 |
+
|
133 |
+
if response.status_code == 200:
|
134 |
+
# Parse the returned JSON data
|
135 |
+
data = response.json()
|
136 |
+
|
137 |
+
data
|
138 |
+
|
139 |
+
data=data['data']
|
140 |
+
|
141 |
+
from collections import defaultdict
|
142 |
+
|
143 |
+
# Group AQI values by date (ignoring the hour)
|
144 |
+
grouped_aqi = defaultdict(list)
|
145 |
+
|
146 |
+
for entry in data:
|
147 |
+
# Extract date part only from the datetime (before the colon)
|
148 |
+
date = entry['datetime'].split(':')[0]
|
149 |
+
aqi = entry['aqi']
|
150 |
+
grouped_aqi[date].append(aqi)
|
151 |
+
|
152 |
+
# Convert defaultdict to a regular dictionary
|
153 |
+
grouped_aqi = dict(grouped_aqi)
|
154 |
+
|
155 |
+
# Display the result
|
156 |
+
print(grouped_aqi)
|
157 |
+
index=11
|
158 |
+
key= grouped_aqi.keys()
|
159 |
+
samp={}
|
160 |
+
samp.clear()
|
161 |
+
for i in key:
|
162 |
+
print(i)
|
163 |
+
ls=grouped_aqi[i]
|
164 |
+
if index<len(ls):
|
165 |
+
print(ls[11])
|
166 |
+
samp[i]=ls[11]
|
167 |
+
else:
|
168 |
+
print(ls[-1])
|
169 |
+
samp[i]=ls[-1]
|
170 |
+
print(samp)
|
171 |
+
df = pd.DataFrame([samp])
|
172 |
+
df.columns=['AQI_currrent','AQI_step_1', 'AQI_step_2', 'AQI_step_3']
|
173 |
+
print(df)
|
174 |
+
# Define the CSV file path
|
175 |
+
csv_file_path = "aqi_data_actual_api.csv"
|
176 |
+
|
177 |
+
# Create the CSV file with headers if it doesn't exist
|
178 |
+
if not os.path.exists(csv_file_path):
|
179 |
+
columns = ['AQI_currrent_API','AQI_step_1_API', 'AQI_step_2_API', 'AQI_step_3_API']
|
180 |
+
df_empty = pd.DataFrame(columns=columns)
|
181 |
+
df_empty.to_csv(csv_file_path, index=False)
|
182 |
+
|
183 |
+
# Append new data to the existing CSV
|
184 |
+
df.to_csv(csv_file_path, mode='a', index=False, header=False)
|
185 |
+
|
186 |
+
|
187 |
+
##########################################################
|
188 |
+
|
189 |
+
|
190 |
+
import folium
|
191 |
+
import matplotlib.pyplot as plt
|
192 |
+
from io import BytesIO
|
193 |
+
import base64
|
194 |
+
import pandas as pd
|
195 |
+
|
196 |
+
|
197 |
+
df1= pd.read_csv('aqi_data.csv')
|
198 |
+
df2= pd.read_csv('aqi_data_actual_api.csv')
|
199 |
+
data = pd.concat([df1,df2],axis=1)
|
200 |
+
data = data.head(3)
|
201 |
+
# Create the Folium map
|
202 |
+
map_center = [data['lat'].mean(), data['lon'].mean()]
|
203 |
+
m = folium.Map(location=map_center, zoom_start=10)
|
204 |
+
|
205 |
+
# AQI Color Legend
|
206 |
+
legend_html = """
|
207 |
+
<div style="
|
208 |
+
position: fixed;
|
209 |
+
bottom: 20px; left: 20px; width: 350px; height: 225px;
|
210 |
+
background-color: white;
|
211 |
+
z-index:9999; font-size:14px; border:2px solid grey;
|
212 |
+
padding: 10px; overflow-y: auto;">
|
213 |
+
<b>AQI Color Legend</b>
|
214 |
+
<table style="width: 100%; border-collapse: collapse; text-align: left;">
|
215 |
+
<thead>
|
216 |
+
<tr style="border-bottom: 2px solid grey;">
|
217 |
+
<th style="padding: 5px;">Color</th>
|
218 |
+
<th style="padding: 5px;">Remark</th>
|
219 |
+
<th style="padding: 5px;">Range</th>
|
220 |
+
</tr>
|
221 |
+
</thead>
|
222 |
+
<tbody>
|
223 |
+
<tr>
|
224 |
+
<td><i style="background:green; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
225 |
+
<td>Good</td>
|
226 |
+
<td>0-50</td>
|
227 |
+
</tr>
|
228 |
+
<tr>
|
229 |
+
<td><i style="background:yellow; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
230 |
+
<td>Moderate</td>
|
231 |
+
<td>51-100</td>
|
232 |
+
</tr>
|
233 |
+
<tr>
|
234 |
+
<td><i style="background:orange; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
235 |
+
<td>Unhealthy for Sensitive Groups</td>
|
236 |
+
<td>101-150</td>
|
237 |
+
</tr>
|
238 |
+
<tr>
|
239 |
+
<td><i style="background:red; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
240 |
+
<td>Unhealthy</td>
|
241 |
+
<td>151-200</td>
|
242 |
+
</tr>
|
243 |
+
<tr>
|
244 |
+
<td><i style="background:purple; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
245 |
+
<td>Very Unhealthy</td>
|
246 |
+
<td>201-300</td>
|
247 |
+
</tr>
|
248 |
+
<tr>
|
249 |
+
<td><i style="background:maroon; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
250 |
+
<td>Hazardous</td>
|
251 |
+
<td>301+</td>
|
252 |
+
</tr>
|
253 |
+
</tbody>
|
254 |
+
</table>
|
255 |
+
</div>
|
256 |
+
"""
|
257 |
+
# Add the legend to the map
|
258 |
+
legend = folium.Element(legend_html)
|
259 |
+
m.get_root().html.add_child(legend)
|
260 |
+
|
261 |
+
# Function to generate a horizontal bar plot
|
262 |
+
def create_aqi_comparison_plot(data):
|
263 |
+
fig, ax = plt.subplots(figsize=(5, 2))
|
264 |
+
categories = ['DAY 1', 'DAY 2', 'DAY 3']
|
265 |
+
actual_values = [data['AQI_step_1'], data['AQI_step_2'], data['AQI_step_3']]
|
266 |
+
api_values = [data['AQI_step_1_API'], data['AQI_step_2_API'], data['AQI_step_3_API']]
|
267 |
+
|
268 |
+
bar_width = 0.35
|
269 |
+
index = range(len(categories))
|
270 |
+
|
271 |
+
# Plot horizontal bars
|
272 |
+
bars_actual = ax.barh(index, actual_values, bar_width, label="Model AQI", color='blue')
|
273 |
+
bars_api = ax.barh([i + bar_width for i in index], api_values, bar_width, label="API AQI", color='green')
|
274 |
+
|
275 |
+
# Add values to each bar
|
276 |
+
max_value = 0 # Track the maximum value for axis limit adjustment
|
277 |
+
for bar in bars_actual:
|
278 |
+
value = bar.get_width()
|
279 |
+
ax.text(value + 2, bar.get_y() + bar.get_height() / 2,
|
280 |
+
f'{value:.1f}', va='center', fontsize=10)
|
281 |
+
max_value = max(max_value, value)
|
282 |
+
for bar in bars_api:
|
283 |
+
value = bar.get_width()
|
284 |
+
ax.text(value + 2, bar.get_y() + bar.get_height() / 2,
|
285 |
+
f'{value:.1f}', va='center', fontsize=10)
|
286 |
+
max_value = max(max_value, value)
|
287 |
+
|
288 |
+
# Adjust x-axis limits to accommodate annotations
|
289 |
+
ax.set_xlim(0, max_value * 1.2)
|
290 |
+
|
291 |
+
# Customize y-ticks and labels
|
292 |
+
ax.set_yticks([i + bar_width / 2 for i in index])
|
293 |
+
ax.set_yticklabels(categories)
|
294 |
+
ax.set_xlabel('AQI')
|
295 |
+
ax.set_title('AQI Comparison')
|
296 |
+
|
297 |
+
# Place legend outside the plot area
|
298 |
+
ax.legend(loc='center left', bbox_to_anchor=(1, 0.5), frameon=False)
|
299 |
+
|
300 |
+
plt.tight_layout()
|
301 |
+
|
302 |
+
# Save the plot to a PNG image in memory
|
303 |
+
buffer = BytesIO()
|
304 |
+
plt.savefig(buffer, format="png", bbox_inches='tight')
|
305 |
+
plt.close(fig)
|
306 |
+
buffer.seek(0)
|
307 |
+
|
308 |
+
# Encode the image to base64 to embed it in the HTML
|
309 |
+
image_base64 = base64.b64encode(buffer.read()).decode()
|
310 |
+
return f'<img src="data:image/png;base64,{image_base64}">'
|
311 |
+
|
312 |
+
# Function to determine AQI marker color
|
313 |
+
def get_color_for_aqi(aqi_value):
|
314 |
+
if aqi_value <= 50:
|
315 |
+
return 'green'
|
316 |
+
elif aqi_value <= 100:
|
317 |
+
return 'yellow'
|
318 |
+
elif aqi_value <= 150:
|
319 |
+
return 'orange'
|
320 |
+
elif aqi_value <= 200:
|
321 |
+
return 'red'
|
322 |
+
elif aqi_value <= 300:
|
323 |
+
return 'purple'
|
324 |
+
else:
|
325 |
+
return 'maroon'
|
326 |
+
|
327 |
+
# Add markers with AQI comparison plot
|
328 |
+
for _, row in data.iterrows():
|
329 |
+
color = get_color_for_aqi(row['AQI_step_1'])
|
330 |
+
popup_html = create_aqi_comparison_plot(row)
|
331 |
+
folium.Marker(
|
332 |
+
location=[row["lat"], row["lon"]],
|
333 |
+
popup=folium.Popup(html=popup_html, max_width=500),
|
334 |
+
#tooltip=row["name"],
|
335 |
+
icon=folium.Icon(color=color)
|
336 |
+
).add_to(m)
|
337 |
+
|
338 |
+
# Save the map
|
339 |
+
m.save("aqi_forecast_with_legend.html")
|
340 |
+
m
|
aqi_data.csv
ADDED
@@ -0,0 +1,15 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
PM2.5,PM10,NO2,SO2,CO,AQI,Day,Month,Hour,lat,lon,AQI_step_1,AQI_step_2,AQI_step_3
|
2 |
+
87,89,2,127,123.1396,192,20,11,11,21.195069775800516,72.79324648126439,174.50804,175.36728,176.44917
|
3 |
+
86,89,7,65,138.3132,191,20,11,17,21.22050672027795,72.83355967457062,171.93951,172.93735,174.26135
|
4 |
+
86,89,7,65,138.3132,191,20,11,17,21.236796371788703,72.8665479925569,171.93951,172.93735,174.26135
|
5 |
+
84.0,88.0,3.0,89.0,132.4772,188.0,21.0,11.0,13.0,21.236796371788703,72.8665479925569,169.29886,170.1785,171.38548
|
6 |
+
205.0,256.0,8.0,99.0,151.1524,205.0,21.0,11.0,13.0,28.639638713652012,77.19002000205269,178.70003,179.95726,181.62497
|
7 |
+
54.0,58.0,2.0,40.0,124.8904,150.0,21.0,11.0,15.0,21.634633971544297,72.99221466910912,143.35014,143.9174,144.78899
|
8 |
+
106.0,116.0,2.0,80.0,170.4112,217.0,21.0,11.0,15.0,26.846962942608172,80.90966630609816,196.3583,198.081,200.372
|
9 |
+
35.0,37.0,3.0,40.0,141.2312,101.0,21.0,11.0,15.0,31.335876535998896,75.565248172424,111.138115,111.64326,112.21634
|
10 |
+
34.0,36.0,5.0,95.0,139.4804,101.0,21.0,11.0,16.0,30.921956088524542,75.85192818745244,112.60642,113.12772,113.72198
|
11 |
+
74.0,83.0,1.0,27.0,113.2184,175.0,21.0,11.0,16.0,23.238552725422572,77.43189201233218,157.19151,157.69708,158.61711
|
12 |
+
89.0,111.0,3.0,27.0,104.4644,195.0,21.0,11.0,16.0,22.622719257719115,88.36320858480713,173.02531,173.67334,174.52841
|
13 |
+
29.0,30.0,3.0,36.0,114.3856,87.0,21.0,11.0,16.0,11.301601687488285,77.19399206074418,94.49498,94.79229,95.16263
|
14 |
+
15.0,17.0,0.0,2.0,82.8712,56.0,21.0,11.0,16.0,10.569512756592735,72.63877516454528,56.11896,56.067623,56.15389
|
15 |
+
15.0,17.0,0.0,2.0,82.8712,56.0,21.0,11.0,16.0,10.569512756592735,72.63877516454528,56.11896,56.067623,56.15389
|
aqi_data_actual_api.csv
ADDED
@@ -0,0 +1,15 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
AQI_currrent_API,AQI_step_1_API,AQI_step_2_API,AQI_step_3_API
|
2 |
+
172,177,176,177
|
3 |
+
171,176,178,188
|
4 |
+
171,176,178,188
|
5 |
+
188,189,203,225
|
6 |
+
205,264,272,257
|
7 |
+
150,150,173,167
|
8 |
+
217,229,265,217
|
9 |
+
101,241,252,262
|
10 |
+
101,252,264,268
|
11 |
+
175,191,184,167
|
12 |
+
195,253,274,273
|
13 |
+
87,163,163,153
|
14 |
+
56,56,72,93
|
15 |
+
56,56,72,93
|
requirement.txt
ADDED
@@ -0,0 +1,116 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
absl-py==2.1.0
|
2 |
+
accelerate==1.1.1
|
3 |
+
aiohappyeyeballs==2.4.3
|
4 |
+
aiohttp==3.11.2
|
5 |
+
aiosignal==1.3.1
|
6 |
+
asttokens==2.4.1
|
7 |
+
astunparse==1.6.3
|
8 |
+
async-timeout==5.0.1
|
9 |
+
attrs==24.2.0
|
10 |
+
blinker==1.9.0
|
11 |
+
branca==0.8.0
|
12 |
+
certifi==2024.8.30
|
13 |
+
charset-normalizer==3.4.0
|
14 |
+
click==8.1.7
|
15 |
+
colorama==0.4.6
|
16 |
+
comm==0.2.2
|
17 |
+
contourpy==1.3.0
|
18 |
+
cycler==0.12.1
|
19 |
+
datasets==3.1.0
|
20 |
+
debugpy==1.8.7
|
21 |
+
decorator==5.1.1
|
22 |
+
dill==0.3.8
|
23 |
+
exceptiongroup==1.2.2
|
24 |
+
executing==2.1.0
|
25 |
+
filelock==3.16.1
|
26 |
+
Flask==3.0.3
|
27 |
+
flatbuffers==24.3.25
|
28 |
+
folium==0.18.0
|
29 |
+
fonttools==4.54.1
|
30 |
+
frozenlist==1.5.0
|
31 |
+
fsspec==2024.9.0
|
32 |
+
gast==0.6.0
|
33 |
+
google-pasta==0.2.0
|
34 |
+
grpcio==1.67.1
|
35 |
+
h5py==3.12.1
|
36 |
+
huggingface-hub==0.26.2
|
37 |
+
idna==3.10
|
38 |
+
ipykernel==6.29.5
|
39 |
+
ipython==8.29.0
|
40 |
+
itsdangerous==2.2.0
|
41 |
+
jedi==0.19.1
|
42 |
+
Jinja2==3.1.4
|
43 |
+
joblib==1.4.2
|
44 |
+
jupyter_client==8.6.3
|
45 |
+
jupyter_core==5.7.2
|
46 |
+
keras==3.6.0
|
47 |
+
kiwisolver==1.4.7
|
48 |
+
libclang==18.1.1
|
49 |
+
Markdown==3.7
|
50 |
+
markdown-it-py==3.0.0
|
51 |
+
MarkupSafe==3.0.2
|
52 |
+
matplotlib==3.9.2
|
53 |
+
matplotlib-inline==0.1.7
|
54 |
+
mdurl==0.1.2
|
55 |
+
ml-dtypes==0.4.1
|
56 |
+
mpmath==1.3.0
|
57 |
+
multidict==6.1.0
|
58 |
+
multiprocess==0.70.16
|
59 |
+
namex==0.0.8
|
60 |
+
nest-asyncio==1.6.0
|
61 |
+
networkx==3.4.2
|
62 |
+
numpy==2.0.2
|
63 |
+
opt_einsum==3.4.0
|
64 |
+
optree==0.13.0
|
65 |
+
packaging==24.1
|
66 |
+
pandas==2.2.3
|
67 |
+
parso==0.8.4
|
68 |
+
pillow==11.0.0
|
69 |
+
platformdirs==4.3.6
|
70 |
+
prompt_toolkit==3.0.48
|
71 |
+
propcache==0.2.0
|
72 |
+
protobuf==5.28.3
|
73 |
+
psutil==6.1.0
|
74 |
+
pure_eval==0.2.3
|
75 |
+
pyarrow==18.0.0
|
76 |
+
Pygments==2.18.0
|
77 |
+
pyparsing==3.2.0
|
78 |
+
python-dateutil==2.9.0.post0
|
79 |
+
pytz==2024.2
|
80 |
+
pywin32==308
|
81 |
+
PyYAML==6.0.2
|
82 |
+
pyzmq==26.2.0
|
83 |
+
regex==2024.11.6
|
84 |
+
requests==2.32.3
|
85 |
+
rich==13.9.4
|
86 |
+
safetensors==0.4.5
|
87 |
+
scikit-learn==1.5.2
|
88 |
+
scipy==1.14.1
|
89 |
+
seaborn==0.13.2
|
90 |
+
sentencepiece==0.2.0
|
91 |
+
six==1.16.0
|
92 |
+
stack-data==0.6.3
|
93 |
+
sympy==1.13.1
|
94 |
+
tensorboard==2.18.0
|
95 |
+
tensorboard-data-server==0.7.2
|
96 |
+
tensorflow==2.18.0
|
97 |
+
termcolor==2.5.0
|
98 |
+
tf_keras==2.18.0
|
99 |
+
threadpoolctl==3.5.0
|
100 |
+
tokenizers==0.20.3
|
101 |
+
torch==2.5.1
|
102 |
+
torchaudio==2.5.1
|
103 |
+
torchvision==0.20.1
|
104 |
+
tornado==6.4.1
|
105 |
+
tqdm==4.67.0
|
106 |
+
traitlets==5.14.3
|
107 |
+
transformers==4.46.2
|
108 |
+
typing_extensions==4.12.2
|
109 |
+
tzdata==2024.2
|
110 |
+
urllib3==2.2.3
|
111 |
+
wcwidth==0.2.13
|
112 |
+
Werkzeug==3.1.2
|
113 |
+
wrapt==1.16.0
|
114 |
+
xxhash==3.5.0
|
115 |
+
xyzservices==2024.9.0
|
116 |
+
yarl==1.17.1
|
scaler_X_cpcb_4.pkl
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:427ce69431d9fecc00f9f31668e87e575dff645ecfd188dc146d7a41728d1dd7
|
3 |
+
size 1703
|
scaler_y_cpcb_4.pkl
ADDED
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
1 |
+
version https://git-lfs.github.com/spec/v1
|
2 |
+
oid sha256:e2f48ec4bcaf215123873f416b60d9ffe080045913719a945e115131fe3b6eeb
|
3 |
+
size 1023
|
static/aqi_forecast_map.html
ADDED
@@ -0,0 +1,177 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html>
|
3 |
+
<head>
|
4 |
+
|
5 |
+
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
|
6 |
+
|
7 |
+
<script>
|
8 |
+
L_NO_TOUCH = false;
|
9 |
+
L_DISABLE_3D = false;
|
10 |
+
</script>
|
11 |
+
|
12 |
+
<style>html, body {width: 100%;height: 100%;margin: 0;padding: 0;}</style>
|
13 |
+
<style>#map {position:absolute;top:0;bottom:0;right:0;left:0;}</style>
|
14 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.js"></script>
|
15 |
+
<script src="https://code.jquery.com/jquery-3.7.1.min.js"></script>
|
16 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js"></script>
|
17 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js"></script>
|
18 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.css"/>
|
19 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css"/>
|
20 |
+
<link rel="stylesheet" href="https://netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap-glyphicons.css"/>
|
21 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/[email protected]/css/all.min.css"/>
|
22 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css"/>
|
23 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/python-visualization/folium/folium/templates/leaflet.awesome.rotate.min.css"/>
|
24 |
+
|
25 |
+
<meta name="viewport" content="width=device-width,
|
26 |
+
initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
27 |
+
<style>
|
28 |
+
#map_f96d7c068d70a2dfb86669b83d95543c {
|
29 |
+
position: relative;
|
30 |
+
width: 100.0%;
|
31 |
+
height: 100.0%;
|
32 |
+
left: 0.0%;
|
33 |
+
top: 0.0%;
|
34 |
+
}
|
35 |
+
.leaflet-container { font-size: 1rem; }
|
36 |
+
</style>
|
37 |
+
|
38 |
+
</head>
|
39 |
+
<body>
|
40 |
+
|
41 |
+
|
42 |
+
<div class="folium-map" id="map_f96d7c068d70a2dfb86669b83d95543c" ></div>
|
43 |
+
|
44 |
+
</body>
|
45 |
+
<script>
|
46 |
+
|
47 |
+
|
48 |
+
var map_f96d7c068d70a2dfb86669b83d95543c = L.map(
|
49 |
+
"map_f96d7c068d70a2dfb86669b83d95543c",
|
50 |
+
{
|
51 |
+
center: [21.2, 72.85],
|
52 |
+
crs: L.CRS.EPSG3857,
|
53 |
+
zoom: 10,
|
54 |
+
zoomControl: true,
|
55 |
+
preferCanvas: false,
|
56 |
+
}
|
57 |
+
);
|
58 |
+
|
59 |
+
|
60 |
+
|
61 |
+
|
62 |
+
|
63 |
+
var tile_layer_2fced53561d97c46a3851d3ffbd06d32 = L.tileLayer(
|
64 |
+
"https://tile.openstreetmap.org/{z}/{x}/{y}.png",
|
65 |
+
{"attribution": "\u0026copy; \u003ca href=\"https://www.openstreetmap.org/copyright\"\u003eOpenStreetMap\u003c/a\u003e contributors", "detectRetina": false, "maxNativeZoom": 19, "maxZoom": 19, "minZoom": 0, "noWrap": false, "opacity": 1, "subdomains": "abc", "tms": false}
|
66 |
+
);
|
67 |
+
|
68 |
+
|
69 |
+
tile_layer_2fced53561d97c46a3851d3ffbd06d32.addTo(map_f96d7c068d70a2dfb86669b83d95543c);
|
70 |
+
|
71 |
+
|
72 |
+
var marker_6e96a811704e3a9dcaf0853acb663dee = L.marker(
|
73 |
+
[21.195069775800516, 72.79324648126439],
|
74 |
+
{}
|
75 |
+
).addTo(map_f96d7c068d70a2dfb86669b83d95543c);
|
76 |
+
|
77 |
+
|
78 |
+
var icon_9d8e2b2ea6b79f929f60c2483341d300 = L.AwesomeMarkers.icon(
|
79 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
80 |
+
);
|
81 |
+
marker_6e96a811704e3a9dcaf0853acb663dee.setIcon(icon_9d8e2b2ea6b79f929f60c2483341d300);
|
82 |
+
|
83 |
+
|
84 |
+
var popup_d299bc7f89956c3e85741cba3f9cf026 = L.popup({"maxWidth": 500});
|
85 |
+
|
86 |
+
|
87 |
+
|
88 |
+
var html_9c6b4cf4128fd1a18f384a46847adc48 = $(`<div id="html_9c6b4cf4128fd1a18f384a46847adc48" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
89 |
+
popup_d299bc7f89956c3e85741cba3f9cf026.setContent(html_9c6b4cf4128fd1a18f384a46847adc48);
|
90 |
+
|
91 |
+
|
92 |
+
|
93 |
+
marker_6e96a811704e3a9dcaf0853acb663dee.bindPopup(popup_d299bc7f89956c3e85741cba3f9cf026)
|
94 |
+
;
|
95 |
+
|
96 |
+
|
97 |
+
|
98 |
+
|
99 |
+
marker_6e96a811704e3a9dcaf0853acb663dee.bindTooltip(
|
100 |
+
`<div>
|
101 |
+
Location 1
|
102 |
+
</div>`,
|
103 |
+
{"sticky": true}
|
104 |
+
);
|
105 |
+
|
106 |
+
|
107 |
+
var marker_d41a1d89753128a0d648fe9a95670476 = L.marker(
|
108 |
+
[21.3, 72.9],
|
109 |
+
{}
|
110 |
+
).addTo(map_f96d7c068d70a2dfb86669b83d95543c);
|
111 |
+
|
112 |
+
|
113 |
+
var icon_49ac172dbb970324b9bc82293c12de6c = L.AwesomeMarkers.icon(
|
114 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
115 |
+
);
|
116 |
+
marker_d41a1d89753128a0d648fe9a95670476.setIcon(icon_49ac172dbb970324b9bc82293c12de6c);
|
117 |
+
|
118 |
+
|
119 |
+
var popup_d0739edffb2fb3b36cd822eab4c31669 = L.popup({"maxWidth": 500});
|
120 |
+
|
121 |
+
|
122 |
+
|
123 |
+
var html_1d63111637e0a5d02c02ff5848c3a1b1 = $(`<div id="html_1d63111637e0a5d02c02ff5848c3a1b1" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
124 |
+
popup_d0739edffb2fb3b36cd822eab4c31669.setContent(html_1d63111637e0a5d02c02ff5848c3a1b1);
|
125 |
+
|
126 |
+
|
127 |
+
|
128 |
+
marker_d41a1d89753128a0d648fe9a95670476.bindPopup(popup_d0739edffb2fb3b36cd822eab4c31669)
|
129 |
+
;
|
130 |
+
|
131 |
+
|
132 |
+
|
133 |
+
|
134 |
+
marker_d41a1d89753128a0d648fe9a95670476.bindTooltip(
|
135 |
+
`<div>
|
136 |
+
Location 2
|
137 |
+
</div>`,
|
138 |
+
{"sticky": true}
|
139 |
+
);
|
140 |
+
|
141 |
+
|
142 |
+
var marker_2b4dd178b28f8dc64a6b58d3b2195568 = L.marker(
|
143 |
+
[21.1, 72.8],
|
144 |
+
{}
|
145 |
+
).addTo(map_f96d7c068d70a2dfb86669b83d95543c);
|
146 |
+
|
147 |
+
|
148 |
+
var icon_904d715c3fdcf1bd3e2085c62379e5c8 = L.AwesomeMarkers.icon(
|
149 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
150 |
+
);
|
151 |
+
marker_2b4dd178b28f8dc64a6b58d3b2195568.setIcon(icon_904d715c3fdcf1bd3e2085c62379e5c8);
|
152 |
+
|
153 |
+
|
154 |
+
var popup_17f841170c403bf21955f085ac0694c3 = L.popup({"maxWidth": 500});
|
155 |
+
|
156 |
+
|
157 |
+
|
158 |
+
var html_ac0992d15caf4cab8b21de240372b018 = $(`<div id="html_ac0992d15caf4cab8b21de240372b018" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
159 |
+
popup_17f841170c403bf21955f085ac0694c3.setContent(html_ac0992d15caf4cab8b21de240372b018);
|
160 |
+
|
161 |
+
|
162 |
+
|
163 |
+
marker_2b4dd178b28f8dc64a6b58d3b2195568.bindPopup(popup_17f841170c403bf21955f085ac0694c3)
|
164 |
+
;
|
165 |
+
|
166 |
+
|
167 |
+
|
168 |
+
|
169 |
+
marker_2b4dd178b28f8dc64a6b58d3b2195568.bindTooltip(
|
170 |
+
`<div>
|
171 |
+
Location 3
|
172 |
+
</div>`,
|
173 |
+
{"sticky": true}
|
174 |
+
);
|
175 |
+
|
176 |
+
</script>
|
177 |
+
</html>
|
static/aqi_forecast_vs_actual_map.html
ADDED
The diff for this file is too large to render.
See raw diff
|
|
static/aqi_forecast_vs_actual_map_horizontal_bars.html
ADDED
The diff for this file is too large to render.
See raw diff
|
|
static/aqi_forecast_with_legend.html
ADDED
@@ -0,0 +1,203 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html>
|
3 |
+
<head>
|
4 |
+
|
5 |
+
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
|
6 |
+
|
7 |
+
<script>
|
8 |
+
L_NO_TOUCH = false;
|
9 |
+
L_DISABLE_3D = false;
|
10 |
+
</script>
|
11 |
+
|
12 |
+
<style>html, body {width: 100%;height: 100%;margin: 0;padding: 0;}</style>
|
13 |
+
<style>#map {position:absolute;top:0;bottom:0;right:0;left:0;}</style>
|
14 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.js"></script>
|
15 |
+
<script src="https://code.jquery.com/jquery-3.7.1.min.js"></script>
|
16 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js"></script>
|
17 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js"></script>
|
18 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.css"/>
|
19 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css"/>
|
20 |
+
<link rel="stylesheet" href="https://netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap-glyphicons.css"/>
|
21 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/[email protected]/css/all.min.css"/>
|
22 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css"/>
|
23 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/python-visualization/folium/folium/templates/leaflet.awesome.rotate.min.css"/>
|
24 |
+
|
25 |
+
<meta name="viewport" content="width=device-width,
|
26 |
+
initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
27 |
+
<style>
|
28 |
+
#map_2c2025c360c3ed95ce6339b20a693bdd {
|
29 |
+
position: relative;
|
30 |
+
width: 100.0%;
|
31 |
+
height: 100.0%;
|
32 |
+
left: 0.0%;
|
33 |
+
top: 0.0%;
|
34 |
+
}
|
35 |
+
.leaflet-container { font-size: 1rem; }
|
36 |
+
</style>
|
37 |
+
|
38 |
+
</head>
|
39 |
+
<body>
|
40 |
+
<meta charset="UTF-8">
|
41 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
42 |
+
<title>AQI Forecast Map</title>
|
43 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.0/css/bootstrap.min.css">
|
44 |
+
<style>
|
45 |
+
#legend {
|
46 |
+
position: fixed;
|
47 |
+
bottom: 20px;
|
48 |
+
left: 20px;
|
49 |
+
width: 300px;
|
50 |
+
background-color: rgba(255, 255, 255, 0.8);
|
51 |
+
z-index: 9999;
|
52 |
+
font-size: 12px;
|
53 |
+
border-radius: 8px;
|
54 |
+
padding: 10px;
|
55 |
+
box-shadow: 2px 2px 8px rgba(0, 0, 0, 0.3);
|
56 |
+
}
|
57 |
+
</style>
|
58 |
+
|
59 |
+
<!-- Navigation Bar -->
|
60 |
+
<nav class="navbar navbar-expand-lg navbar-dark bg-dark">
|
61 |
+
<div class="container-fluid">
|
62 |
+
<a class="navbar-brand" href="#">AQI Forecaster</a>
|
63 |
+
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav" aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
|
64 |
+
<span class="navbar-toggler-icon"></span>
|
65 |
+
</button>
|
66 |
+
<div class="collapse navbar-collapse" id="navbarNav">
|
67 |
+
<form class="d-flex ms-auto" method="POST" action="{{ url_for('forecast') }}">
|
68 |
+
<input class="form-control me-2" type="text" name="latitude" placeholder="Enter Latitude" required>
|
69 |
+
<input class="form-control me-2" type="text" name="longitude" placeholder="Enter Longitude" required>
|
70 |
+
<button class="btn btn-outline-success" type="submit">Update Map</button>
|
71 |
+
</form>
|
72 |
+
</div>
|
73 |
+
</div>
|
74 |
+
</nav>
|
75 |
+
|
76 |
+
<!-- AQI Legend -->
|
77 |
+
<div id="legend">
|
78 |
+
<b>AQI Color Legend</b>
|
79 |
+
<ul style="list-style: none; padding: 0; margin: 5px;">
|
80 |
+
<li><i style="background:green; width:15px; height:15px; display:inline-block; margin-right:5px;"></i> Good (0-50)</li>
|
81 |
+
<li><i style="background:yellow; width:15px; height:15px; display:inline-block; margin-right:5px;"></i> Moderate (51-100)</li>
|
82 |
+
<li><i style="background:orange; width:15px; height:15px; display:inline-block; margin-right:5px;"></i> Unhealthy for Sensitive Groups (101-150)</li>
|
83 |
+
<li><i style="background:red; width:15px; height:15px; display:inline-block; margin-right:5px;"></i> Unhealthy (151-200)</li>
|
84 |
+
<li><i style="background:purple; width:15px; height:15px; display:inline-block; margin-right:5px;"></i> Very Unhealthy (201-300)</li>
|
85 |
+
<li><i style="background:maroon; width:15px; height:15px; display:inline-block; margin-right:5px;"></i> Hazardous (301+)</li>
|
86 |
+
</ul>
|
87 |
+
</div>
|
88 |
+
|
89 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.0/js/bootstrap.bundle.min.js"></script>
|
90 |
+
|
91 |
+
|
92 |
+
<div class="folium-map" id="map_2c2025c360c3ed95ce6339b20a693bdd" ></div>
|
93 |
+
|
94 |
+
</body>
|
95 |
+
<script>
|
96 |
+
|
97 |
+
|
98 |
+
var map_2c2025c360c3ed95ce6339b20a693bdd = L.map(
|
99 |
+
"map_2c2025c360c3ed95ce6339b20a693bdd",
|
100 |
+
{
|
101 |
+
center: [21.21745762262239, 72.83111804946397],
|
102 |
+
crs: L.CRS.EPSG3857,
|
103 |
+
zoom: 10,
|
104 |
+
zoomControl: true,
|
105 |
+
preferCanvas: false,
|
106 |
+
}
|
107 |
+
);
|
108 |
+
|
109 |
+
|
110 |
+
|
111 |
+
|
112 |
+
|
113 |
+
var tile_layer_1ed9d8a583c9ed0cbc8410660e53a246 = L.tileLayer(
|
114 |
+
"https://tile.openstreetmap.org/{z}/{x}/{y}.png",
|
115 |
+
{"attribution": "\u0026copy; \u003ca href=\"https://www.openstreetmap.org/copyright\"\u003eOpenStreetMap\u003c/a\u003e contributors", "detectRetina": false, "maxNativeZoom": 19, "maxZoom": 19, "minZoom": 0, "noWrap": false, "opacity": 1, "subdomains": "abc", "tms": false}
|
116 |
+
);
|
117 |
+
|
118 |
+
|
119 |
+
tile_layer_1ed9d8a583c9ed0cbc8410660e53a246.addTo(map_2c2025c360c3ed95ce6339b20a693bdd);
|
120 |
+
|
121 |
+
|
122 |
+
var marker_38b108b3eb17f246af1847ce3fde7155 = L.marker(
|
123 |
+
[21.19506977580052, 72.79324648126439],
|
124 |
+
{}
|
125 |
+
).addTo(map_2c2025c360c3ed95ce6339b20a693bdd);
|
126 |
+
|
127 |
+
|
128 |
+
var icon_48e797f3cf504d64ca1c73448a103ea4 = L.AwesomeMarkers.icon(
|
129 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "red", "prefix": "glyphicon"}
|
130 |
+
);
|
131 |
+
marker_38b108b3eb17f246af1847ce3fde7155.setIcon(icon_48e797f3cf504d64ca1c73448a103ea4);
|
132 |
+
|
133 |
+
|
134 |
+
var popup_0baf097a03f16cda6bcec6973beb4d7e = L.popup({"maxWidth": 500});
|
135 |
+
|
136 |
+
|
137 |
+
|
138 |
+
var html_7a3296080251cbbfd4b2ec8815f1ae4e = $(`<div id="html_7a3296080251cbbfd4b2ec8815f1ae4e" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
139 |
+
popup_0baf097a03f16cda6bcec6973beb4d7e.setContent(html_7a3296080251cbbfd4b2ec8815f1ae4e);
|
140 |
+
|
141 |
+
|
142 |
+
|
143 |
+
marker_38b108b3eb17f246af1847ce3fde7155.bindPopup(popup_0baf097a03f16cda6bcec6973beb4d7e)
|
144 |
+
;
|
145 |
+
|
146 |
+
|
147 |
+
|
148 |
+
|
149 |
+
var marker_408f797d9baf56a8137906ff78f3eff5 = L.marker(
|
150 |
+
[21.22050672027795, 72.83355967457062],
|
151 |
+
{}
|
152 |
+
).addTo(map_2c2025c360c3ed95ce6339b20a693bdd);
|
153 |
+
|
154 |
+
|
155 |
+
var icon_331cf9c1763cef84374ab13767deef94 = L.AwesomeMarkers.icon(
|
156 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "red", "prefix": "glyphicon"}
|
157 |
+
);
|
158 |
+
marker_408f797d9baf56a8137906ff78f3eff5.setIcon(icon_331cf9c1763cef84374ab13767deef94);
|
159 |
+
|
160 |
+
|
161 |
+
var popup_3e38d54f4f14f205f64d254fc7ff5ec2 = L.popup({"maxWidth": 500});
|
162 |
+
|
163 |
+
|
164 |
+
|
165 |
+
var html_19ea0d1ccb60191cfa575a17ae6bc14a = $(`<div id="html_19ea0d1ccb60191cfa575a17ae6bc14a" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
166 |
+
popup_3e38d54f4f14f205f64d254fc7ff5ec2.setContent(html_19ea0d1ccb60191cfa575a17ae6bc14a);
|
167 |
+
|
168 |
+
|
169 |
+
|
170 |
+
marker_408f797d9baf56a8137906ff78f3eff5.bindPopup(popup_3e38d54f4f14f205f64d254fc7ff5ec2)
|
171 |
+
;
|
172 |
+
|
173 |
+
|
174 |
+
|
175 |
+
|
176 |
+
var marker_93a2124fa21d15be6000355f1e3d3307 = L.marker(
|
177 |
+
[21.236796371788703, 72.8665479925569],
|
178 |
+
{}
|
179 |
+
).addTo(map_2c2025c360c3ed95ce6339b20a693bdd);
|
180 |
+
|
181 |
+
|
182 |
+
var icon_2a83ba72c46f59895db0b7de760b0a07 = L.AwesomeMarkers.icon(
|
183 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "red", "prefix": "glyphicon"}
|
184 |
+
);
|
185 |
+
marker_93a2124fa21d15be6000355f1e3d3307.setIcon(icon_2a83ba72c46f59895db0b7de760b0a07);
|
186 |
+
|
187 |
+
|
188 |
+
var popup_0f78979bbd5a3a2c30a209b0b346f20b = L.popup({"maxWidth": 500});
|
189 |
+
|
190 |
+
|
191 |
+
|
192 |
+
var html_7a8f516c8442020b7cbc117f95c0b7b6 = $(`<div id="html_7a8f516c8442020b7cbc117f95c0b7b6" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
193 |
+
popup_0f78979bbd5a3a2c30a209b0b346f20b.setContent(html_7a8f516c8442020b7cbc117f95c0b7b6);
|
194 |
+
|
195 |
+
|
196 |
+
|
197 |
+
marker_93a2124fa21d15be6000355f1e3d3307.bindPopup(popup_0f78979bbd5a3a2c30a209b0b346f20b)
|
198 |
+
;
|
199 |
+
|
200 |
+
|
201 |
+
|
202 |
+
</script>
|
203 |
+
</html>
|
static/aqi_map_with_legend.html
ADDED
@@ -0,0 +1,222 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html>
|
3 |
+
<head>
|
4 |
+
|
5 |
+
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
|
6 |
+
|
7 |
+
<script>
|
8 |
+
L_NO_TOUCH = false;
|
9 |
+
L_DISABLE_3D = false;
|
10 |
+
</script>
|
11 |
+
|
12 |
+
<style>html, body {width: 100%;height: 100%;margin: 0;padding: 0;}</style>
|
13 |
+
<style>#map {position:absolute;top:0;bottom:0;right:0;left:0;}</style>
|
14 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.js"></script>
|
15 |
+
<script src="https://code.jquery.com/jquery-3.7.1.min.js"></script>
|
16 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js"></script>
|
17 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js"></script>
|
18 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.css"/>
|
19 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css"/>
|
20 |
+
<link rel="stylesheet" href="https://netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap-glyphicons.css"/>
|
21 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/[email protected]/css/all.min.css"/>
|
22 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css"/>
|
23 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/python-visualization/folium/folium/templates/leaflet.awesome.rotate.min.css"/>
|
24 |
+
|
25 |
+
<meta name="viewport" content="width=device-width,
|
26 |
+
initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
27 |
+
<style>
|
28 |
+
#map_a150e0fb4cc7276810a35f632619f3a1 {
|
29 |
+
position: relative;
|
30 |
+
width: 100.0%;
|
31 |
+
height: 100.0%;
|
32 |
+
left: 0.0%;
|
33 |
+
top: 0.0%;
|
34 |
+
}
|
35 |
+
.leaflet-container { font-size: 1rem; }
|
36 |
+
</style>
|
37 |
+
|
38 |
+
</head>
|
39 |
+
<body>
|
40 |
+
|
41 |
+
|
42 |
+
<div style="
|
43 |
+
position: fixed;
|
44 |
+
bottom: 50px; left: 50px; width: 300px; height: 200px;
|
45 |
+
background-color: white;
|
46 |
+
z-index:9999; font-size:14px; border:2px solid grey;
|
47 |
+
padding: 10px; overflow-y: auto;">
|
48 |
+
<b>AQI Color Legend</b>
|
49 |
+
<table style="width: 100%; border-collapse: collapse; text-align: left;">
|
50 |
+
<thead>
|
51 |
+
<tr style="border-bottom: 2px solid grey;">
|
52 |
+
<th style="padding: 5px;">Color</th>
|
53 |
+
<th style="padding: 5px;">Remark</th>
|
54 |
+
<th style="padding: 5px;">Range</th>
|
55 |
+
</tr>
|
56 |
+
</thead>
|
57 |
+
<tbody>
|
58 |
+
<tr>
|
59 |
+
<td style="padding: 5px;"><i style="background:green; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
60 |
+
<td style="padding: 5px;">Good</td>
|
61 |
+
<td style="padding: 5px;">0-50</td>
|
62 |
+
</tr>
|
63 |
+
<tr>
|
64 |
+
<td style="padding: 5px;"><i style="background:yellow; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
65 |
+
<td style="padding: 5px;">Moderate</td>
|
66 |
+
<td style="padding: 5px;">51-100</td>
|
67 |
+
</tr>
|
68 |
+
<tr>
|
69 |
+
<td style="padding: 5px;"><i style="background:orange; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
70 |
+
<td style="padding: 5px;">Unhealthy for Sensitive Groups</td>
|
71 |
+
<td style="padding: 5px;">101-150</td>
|
72 |
+
</tr>
|
73 |
+
<tr>
|
74 |
+
<td style="padding: 5px;"><i style="background:red; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
75 |
+
<td style="padding: 5px;">Unhealthy</td>
|
76 |
+
<td style="padding: 5px;">151-200</td>
|
77 |
+
</tr>
|
78 |
+
<tr>
|
79 |
+
<td style="padding: 5px;"><i style="background:purple; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
80 |
+
<td style="padding: 5px;">Very Unhealthy</td>
|
81 |
+
<td style="padding: 5px;">201-300</td>
|
82 |
+
</tr>
|
83 |
+
<tr>
|
84 |
+
<td style="padding: 5px;"><i style="background:maroon; width:15px; height:15px; display:inline-block; border:1px solid black;"></i></td>
|
85 |
+
<td style="padding: 5px;">Hazardous</td>
|
86 |
+
<td style="padding: 5px;">301+</td>
|
87 |
+
</tr>
|
88 |
+
</tbody>
|
89 |
+
</table>
|
90 |
+
</div>
|
91 |
+
|
92 |
+
<div class="folium-map" id="map_a150e0fb4cc7276810a35f632619f3a1" ></div>
|
93 |
+
|
94 |
+
</body>
|
95 |
+
<script>
|
96 |
+
|
97 |
+
|
98 |
+
var map_a150e0fb4cc7276810a35f632619f3a1 = L.map(
|
99 |
+
"map_a150e0fb4cc7276810a35f632619f3a1",
|
100 |
+
{
|
101 |
+
center: [21.2130139551601, 72.81864929625287],
|
102 |
+
crs: L.CRS.EPSG3857,
|
103 |
+
zoom: 10,
|
104 |
+
zoomControl: true,
|
105 |
+
preferCanvas: false,
|
106 |
+
}
|
107 |
+
);
|
108 |
+
|
109 |
+
|
110 |
+
|
111 |
+
|
112 |
+
|
113 |
+
var tile_layer_63bd45a9cd619b6645de023d33db15a9 = L.tileLayer(
|
114 |
+
"https://tile.openstreetmap.org/{z}/{x}/{y}.png",
|
115 |
+
{"attribution": "\u0026copy; \u003ca href=\"https://www.openstreetmap.org/copyright\"\u003eOpenStreetMap\u003c/a\u003e contributors", "detectRetina": false, "maxNativeZoom": 19, "maxZoom": 19, "minZoom": 0, "noWrap": false, "opacity": 1, "subdomains": "abc", "tms": false}
|
116 |
+
);
|
117 |
+
|
118 |
+
|
119 |
+
tile_layer_63bd45a9cd619b6645de023d33db15a9.addTo(map_a150e0fb4cc7276810a35f632619f3a1);
|
120 |
+
|
121 |
+
|
122 |
+
var marker_2429860b4cbaf5615ea1017c44b19f22 = L.marker(
|
123 |
+
[21.195069775800516, 72.79324648126439],
|
124 |
+
{}
|
125 |
+
).addTo(map_a150e0fb4cc7276810a35f632619f3a1);
|
126 |
+
|
127 |
+
|
128 |
+
var icon_67d84c6204b0790d039646f6c1e2da39 = L.AwesomeMarkers.icon(
|
129 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
130 |
+
);
|
131 |
+
marker_2429860b4cbaf5615ea1017c44b19f22.setIcon(icon_67d84c6204b0790d039646f6c1e2da39);
|
132 |
+
|
133 |
+
|
134 |
+
marker_2429860b4cbaf5615ea1017c44b19f22.bindTooltip(
|
135 |
+
`<div>
|
136 |
+
Location 1
|
137 |
+
</div>`,
|
138 |
+
{"sticky": true}
|
139 |
+
);
|
140 |
+
|
141 |
+
|
142 |
+
var marker_b7a6e3483466a57f4c675d8f1855957d = L.marker(
|
143 |
+
[21.3, 72.9],
|
144 |
+
{}
|
145 |
+
).addTo(map_a150e0fb4cc7276810a35f632619f3a1);
|
146 |
+
|
147 |
+
|
148 |
+
var icon_312c040d6b5b62945c0a9095b367df86 = L.AwesomeMarkers.icon(
|
149 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
150 |
+
);
|
151 |
+
marker_b7a6e3483466a57f4c675d8f1855957d.setIcon(icon_312c040d6b5b62945c0a9095b367df86);
|
152 |
+
|
153 |
+
|
154 |
+
marker_b7a6e3483466a57f4c675d8f1855957d.bindTooltip(
|
155 |
+
`<div>
|
156 |
+
Location 2
|
157 |
+
</div>`,
|
158 |
+
{"sticky": true}
|
159 |
+
);
|
160 |
+
|
161 |
+
|
162 |
+
var marker_7ae55386c3f54f1e32b3a75aa8a3c475 = L.marker(
|
163 |
+
[21.1, 72.8],
|
164 |
+
{}
|
165 |
+
).addTo(map_a150e0fb4cc7276810a35f632619f3a1);
|
166 |
+
|
167 |
+
|
168 |
+
var icon_3dbc85d72a24050ac1f5c0e977d415fe = L.AwesomeMarkers.icon(
|
169 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
170 |
+
);
|
171 |
+
marker_7ae55386c3f54f1e32b3a75aa8a3c475.setIcon(icon_3dbc85d72a24050ac1f5c0e977d415fe);
|
172 |
+
|
173 |
+
|
174 |
+
marker_7ae55386c3f54f1e32b3a75aa8a3c475.bindTooltip(
|
175 |
+
`<div>
|
176 |
+
Location 3
|
177 |
+
</div>`,
|
178 |
+
{"sticky": true}
|
179 |
+
);
|
180 |
+
|
181 |
+
|
182 |
+
var marker_2d8020c07cc33b3b6500310ab4224ec2 = L.marker(
|
183 |
+
[21.25, 72.85],
|
184 |
+
{}
|
185 |
+
).addTo(map_a150e0fb4cc7276810a35f632619f3a1);
|
186 |
+
|
187 |
+
|
188 |
+
var icon_a4e9c615cf66ea065d533cfb5956086b = L.AwesomeMarkers.icon(
|
189 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
190 |
+
);
|
191 |
+
marker_2d8020c07cc33b3b6500310ab4224ec2.setIcon(icon_a4e9c615cf66ea065d533cfb5956086b);
|
192 |
+
|
193 |
+
|
194 |
+
marker_2d8020c07cc33b3b6500310ab4224ec2.bindTooltip(
|
195 |
+
`<div>
|
196 |
+
Location 4
|
197 |
+
</div>`,
|
198 |
+
{"sticky": true}
|
199 |
+
);
|
200 |
+
|
201 |
+
|
202 |
+
var marker_a32fd1d6158d26403dfdfd275aeee4fe = L.marker(
|
203 |
+
[21.22, 72.75],
|
204 |
+
{}
|
205 |
+
).addTo(map_a150e0fb4cc7276810a35f632619f3a1);
|
206 |
+
|
207 |
+
|
208 |
+
var icon_903f2288884967fe826c17b2e3de0ae6 = L.AwesomeMarkers.icon(
|
209 |
+
{"extraClasses": "fa-rotate-0", "icon": "info-sign", "iconColor": "white", "markerColor": "orange", "prefix": "glyphicon"}
|
210 |
+
);
|
211 |
+
marker_a32fd1d6158d26403dfdfd275aeee4fe.setIcon(icon_903f2288884967fe826c17b2e3de0ae6);
|
212 |
+
|
213 |
+
|
214 |
+
marker_a32fd1d6158d26403dfdfd275aeee4fe.bindTooltip(
|
215 |
+
`<div>
|
216 |
+
Location 5
|
217 |
+
</div>`,
|
218 |
+
{"sticky": true}
|
219 |
+
);
|
220 |
+
|
221 |
+
</script>
|
222 |
+
</html>
|
static/multiple_wind_speed_map.html
ADDED
@@ -0,0 +1,101 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html>
|
3 |
+
<head>
|
4 |
+
|
5 |
+
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
|
6 |
+
|
7 |
+
<script>
|
8 |
+
L_NO_TOUCH = false;
|
9 |
+
L_DISABLE_3D = false;
|
10 |
+
</script>
|
11 |
+
|
12 |
+
<style>html, body {width: 100%;height: 100%;margin: 0;padding: 0;}</style>
|
13 |
+
<style>#map {position:absolute;top:0;bottom:0;right:0;left:0;}</style>
|
14 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.js"></script>
|
15 |
+
<script src="https://code.jquery.com/jquery-3.7.1.min.js"></script>
|
16 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js"></script>
|
17 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js"></script>
|
18 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.css"/>
|
19 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css"/>
|
20 |
+
<link rel="stylesheet" href="https://netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap-glyphicons.css"/>
|
21 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/[email protected]/css/all.min.css"/>
|
22 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css"/>
|
23 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/python-visualization/folium/folium/templates/leaflet.awesome.rotate.min.css"/>
|
24 |
+
|
25 |
+
<meta name="viewport" content="width=device-width,
|
26 |
+
initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
27 |
+
<style>
|
28 |
+
#map_f07b2c63ab23eb37554ee9e8f9b994bd {
|
29 |
+
position: relative;
|
30 |
+
width: 100.0%;
|
31 |
+
height: 100.0%;
|
32 |
+
left: 0.0%;
|
33 |
+
top: 0.0%;
|
34 |
+
}
|
35 |
+
.leaflet-container { font-size: 1rem; }
|
36 |
+
</style>
|
37 |
+
|
38 |
+
</head>
|
39 |
+
<body>
|
40 |
+
|
41 |
+
|
42 |
+
<div class="folium-map" id="map_f07b2c63ab23eb37554ee9e8f9b994bd" ></div>
|
43 |
+
|
44 |
+
</body>
|
45 |
+
<script>
|
46 |
+
|
47 |
+
|
48 |
+
var map_f07b2c63ab23eb37554ee9e8f9b994bd = L.map(
|
49 |
+
"map_f07b2c63ab23eb37554ee9e8f9b994bd",
|
50 |
+
{
|
51 |
+
center: [21.172654967610296, 72.8284030862854],
|
52 |
+
crs: L.CRS.EPSG3857,
|
53 |
+
zoom: 10,
|
54 |
+
zoomControl: true,
|
55 |
+
preferCanvas: false,
|
56 |
+
}
|
57 |
+
);
|
58 |
+
|
59 |
+
|
60 |
+
|
61 |
+
|
62 |
+
|
63 |
+
var tile_layer_450ea641254663ee4a9f020cc91927af = L.tileLayer(
|
64 |
+
"https://tile.openstreetmap.org/{z}/{x}/{y}.png",
|
65 |
+
{"attribution": "\u0026copy; \u003ca href=\"https://www.openstreetmap.org/copyright\"\u003eOpenStreetMap\u003c/a\u003e contributors", "detectRetina": false, "maxNativeZoom": 19, "maxZoom": 19, "minZoom": 0, "noWrap": false, "opacity": 1, "subdomains": "abc", "tms": false}
|
66 |
+
);
|
67 |
+
|
68 |
+
|
69 |
+
tile_layer_450ea641254663ee4a9f020cc91927af.addTo(map_f07b2c63ab23eb37554ee9e8f9b994bd);
|
70 |
+
|
71 |
+
|
72 |
+
var marker_2a7638b5cacc5305e893d4dba45911c5 = L.marker(
|
73 |
+
[21.172654967610296, 72.8284030862854],
|
74 |
+
{}
|
75 |
+
).addTo(map_f07b2c63ab23eb37554ee9e8f9b994bd);
|
76 |
+
|
77 |
+
|
78 |
+
var popup_4d47557e8d279db04bb42be9092829a3 = L.popup({"maxWidth": 300});
|
79 |
+
|
80 |
+
|
81 |
+
|
82 |
+
var html_f9410af04516429d1f1c883381fe8eba = $(`<div id="html_f9410af04516429d1f1c883381fe8eba" style="width: 100.0%; height: 100.0%;"><img src="" alt="Wind Speed Comparison"></div>`)[0];
|
83 |
+
popup_4d47557e8d279db04bb42be9092829a3.setContent(html_f9410af04516429d1f1c883381fe8eba);
|
84 |
+
|
85 |
+
|
86 |
+
|
87 |
+
marker_2a7638b5cacc5305e893d4dba45911c5.bindPopup(popup_4d47557e8d279db04bb42be9092829a3)
|
88 |
+
;
|
89 |
+
|
90 |
+
|
91 |
+
|
92 |
+
|
93 |
+
marker_2a7638b5cacc5305e893d4dba45911c5.bindTooltip(
|
94 |
+
`<div>
|
95 |
+
Utran
|
96 |
+
</div>`,
|
97 |
+
{"sticky": true}
|
98 |
+
);
|
99 |
+
|
100 |
+
</script>
|
101 |
+
</html>
|
surat_AQI_testing_2024.csv
ADDED
The diff for this file is too large to render.
See raw diff
|
|
surat_sample_test.csv
ADDED
@@ -0,0 +1,204 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
PM2.5,PM10,NO,NO2,NOx,NH3,SO2,CO,Ozone,Benzene,Toluene,Xylene,Day,Month,Hour,AQI,Hour sin,Hour cos,Day sin,Day cos,Month sin,Month cos,AQI_step_1,AQI_step_2,AQI_step_3
|
2 |
+
69.95,549.41,64.17,97.97,34.76,0.0,35.78,1.26,0.0,0.0,0.0,0.0,18,11,1,549.2624999999999,0.25881904510252074,0.9659258262890683,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,1111.1625,158.48,639.825
|
3 |
+
67.56,998.93,66.08,101.35,35.02,0.0,36.04,1.23,0.0,0.0,0.0,0.0,18,11,2,1111.1625,0.49999999999999994,0.8660254037844387,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,158.48,639.825,210.0
|
4 |
+
50.98,187.72,66.48,101.38,34.94,0.0,36.09,1.18,0.0,0.0,0.0,0.0,18,11,3,158.48,0.7071067811865476,0.7071067811865476,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,639.825,210.0,150.39999999999998
|
5 |
+
50.59,621.86,66.34,101.0,35.05,0.0,36.14,1.12,0.0,0.0,0.0,0.0,18,11,4,639.825,0.8660254037844386,0.5000000000000001,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,210.0,150.39999999999998,662.3874999999999
|
6 |
+
72.78,260.0,66.92,102.82,35.2,0.0,36.17,1.07,0.0,0.0,0.0,0.0,18,11,5,210.0,0.9659258262890683,0.25881904510252074,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,150.39999999999998,662.3874999999999,111.03333333333335
|
7 |
+
63.9,175.6,62.66,94.63,34.56,0.0,35.95,0.98,0.0,0.0,0.0,0.0,18,11,6,150.39999999999998,1.0,6.123233995736766e-17,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,662.3874999999999,111.03333333333335,327.08750000000003
|
8 |
+
63.0,639.91,57.92,88.44,33.91,0.0,35.72,0.92,0.0,0.0,0.0,0.0,18,11,7,662.3874999999999,0.9659258262890683,-0.25881904510252063,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,111.03333333333335,327.08750000000003,142.98
|
9 |
+
63.31,91.14,56.63,86.67,33.45,0.0,36.41,0.9,0.0,0.0,0.0,0.0,18,11,8,111.03333333333335,0.8660254037844387,-0.4999999999999998,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,327.08750000000003,142.98,202.29999999999998
|
10 |
+
66.03,371.67,51.25,77.13,32.68,0.0,37.09,0.91,0.0,0.0,0.0,0.0,18,11,9,327.08750000000003,0.7071067811865476,-0.7071067811865475,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,142.98,202.29999999999998,302.1625
|
11 |
+
67.62,164.47,45.74,69.24,31.84,0.0,35.9,0.93,0.0,0.0,0.0,0.0,18,11,10,142.98,0.49999999999999994,-0.8660254037844387,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,202.29999999999998,302.1625,91.1
|
12 |
+
90.69,167.21,42.4,64.88,31.07,0.0,36.06,0.95,0.0,0.0,0.0,0.0,18,11,11,202.29999999999998,0.258819045102521,-0.9659258262890682,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,302.1625,91.1,463.2125
|
13 |
+
75.8,351.73,39.57,59.55,30.66,0.0,35.44,0.93,0.0,0.0,0.0,0.0,18,11,12,302.1625,1.2246467991473532e-16,-1.0,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,91.1,463.2125,836.7625
|
14 |
+
54.66,0.0,36.84,56.05,30.16,0.0,35.06,0.83,0.0,0.0,0.0,0.0,18,11,13,91.1,-0.2588190451025208,-0.9659258262890683,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,463.2125,836.7625,104.74666666666667
|
15 |
+
41.84,480.57,36.97,56.66,30.16,0.0,34.74,0.75,0.0,0.0,0.0,0.0,18,11,14,463.2125,-0.4999999999999997,-0.8660254037844388,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,836.7625,104.74666666666667,88.96
|
16 |
+
36.2,779.41,33.93,52.14,29.68,0.0,34.26,0.72,0.0,0.0,0.0,0.0,18,11,15,836.7625,-0.7071067811865471,-0.7071067811865479,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,104.74666666666667,88.96,78.28750000000001
|
17 |
+
42.18,107.12,32.89,49.88,29.61,0.0,33.91,0.69,0.0,0.0,0.0,0.0,18,11,16,104.74666666666667,-0.8660254037844385,-0.5000000000000004,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,88.96,78.28750000000001,98.11666666666666
|
18 |
+
48.77,88.96,35.45,54.23,30.09,0.0,34.17,0.68,0.0,0.0,0.0,0.0,18,11,17,88.96,-0.9659258262890683,-0.25881904510252063,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,78.28750000000001,98.11666666666666,239.33999999999997
|
19 |
+
42.44,53.21,40.4,62.63,30.65,0.0,34.38,0.71,0.0,0.0,0.0,0.0,18,11,18,78.28750000000001,-1.0,-1.8369701987210297e-16,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,98.11666666666666,239.33999999999997,165.43333333333334
|
20 |
+
58.87,60.6,43.38,67.18,31.33,0.0,34.87,0.8,0.0,0.0,0.0,0.0,18,11,19,98.11666666666666,-0.9659258262890684,0.2588190451025203,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,239.33999999999997,165.43333333333334,164.54000000000002
|
21 |
+
58.33,289.34,52.2,80.88,32.49,0.0,35.15,1.0,0.0,0.0,0.0,0.0,18,11,20,239.33999999999997,-0.8660254037844386,0.5000000000000001,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,165.43333333333334,164.54000000000002,143.23333333333335
|
22 |
+
68.13,198.15,59.37,91.22,33.89,0.0,35.14,1.18,0.0,0.0,0.0,0.0,18,11,21,165.43333333333334,-0.7071067811865477,0.7071067811865474,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,164.54000000000002,143.23333333333335,958.75
|
23 |
+
73.0,196.81,62.14,95.41,34.12,0.0,35.31,1.25,0.0,0.0,0.0,0.0,18,11,22,164.54000000000002,-0.5000000000000004,0.8660254037844384,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,143.23333333333335,958.75,221.95
|
24 |
+
72.97,160.86,63.67,97.14,34.43,0.0,35.34,1.28,0.0,0.0,0.0,0.0,18,11,23,143.23333333333335,-0.25881904510252157,0.9659258262890681,-0.4853019625310808,-0.8743466161445822,-0.5000000000000004,0.8660254037844384,958.75,221.95,218.35000000000002
|
25 |
+
72.1,877.0,61.81,94.17,34.42,0.0,35.37,1.27,0.0,0.0,0.0,0.0,19,11,0,958.75,0.0,1.0,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,221.95,218.35000000000002,230.1
|
26 |
+
76.27,271.95,61.12,93.52,34.22,0.0,35.43,1.24,0.0,0.0,0.0,0.0,19,11,1,221.95,0.25881904510252074,0.9659258262890683,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,218.35000000000002,230.1,174.5933333333333
|
27 |
+
84.4,268.35,62.41,95.27,34.28,0.0,35.43,1.18,0.0,0.0,0.0,0.0,19,11,2,218.35000000000002,0.49999999999999994,0.8660254037844387,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,230.1,174.5933333333333,834.6625
|
28 |
+
99.03,216.0,61.08,93.14,34.21,0.0,35.53,1.13,0.0,0.0,0.0,0.0,19,11,3,230.1,0.7071067811865476,0.7071067811865476,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,174.5933333333333,834.6625,121.99999999999999
|
29 |
+
61.17,211.89,62.0,94.81,34.28,0.0,35.52,1.08,0.0,0.0,0.0,0.0,19,11,4,174.5933333333333,0.8660254037844386,0.5000000000000001,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,834.6625,121.99999999999999,1081.5875
|
30 |
+
66.0,777.73,60.93,93.24,34.13,0.0,35.52,1.03,0.0,0.0,0.0,0.0,19,11,5,834.6625,0.9659258262890683,0.25881904510252074,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,121.99999999999999,1081.5875,728.925
|
31 |
+
66.6,0.0,60.47,92.74,34.22,0.0,35.51,1.0,0.0,0.0,0.0,0.0,19,11,6,121.99999999999999,1.0,6.123233995736766e-17,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,1081.5875,728.925,318.65000000000003
|
32 |
+
84.33,975.27,58.39,88.31,33.85,0.0,35.46,0.98,0.0,0.0,0.0,0.0,19,11,7,1081.5875,0.9659258262890683,-0.25881904510252063,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,728.925,318.65000000000003,88.7125
|
33 |
+
94.34,693.14,54.86,84.09,33.22,0.0,35.79,1.0,0.0,0.0,0.0,0.0,19,11,8,728.925,0.8660254037844387,-0.4999999999999998,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,318.65000000000003,88.7125,92.47500000000001
|
34 |
+
106.82,364.92,48.84,73.32,32.34,0.0,36.34,1.02,0.0,0.0,0.0,0.0,19,11,9,318.65000000000003,0.7071067811865476,-0.7071067811865475,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,88.7125,92.47500000000001,87.30000000000001
|
35 |
+
0.0,0.0,46.23,70.97,31.8,0.0,36.44,1.1,0.0,0.0,0.0,0.0,19,11,10,88.7125,0.49999999999999994,-0.8660254037844387,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,92.47500000000001,87.30000000000001,407.8375
|
36 |
+
0.0,0.0,48.37,73.98,31.98,0.0,36.6,1.2,0.0,0.0,0.0,0.0,19,11,11,92.47500000000001,0.258819045102521,-0.9659258262890682,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,87.30000000000001,407.8375,91.5625
|
37 |
+
0.0,0.0,46.02,69.84,31.79,0.0,35.89,1.15,0.0,0.0,0.0,0.0,19,11,12,87.30000000000001,1.2246467991473532e-16,-1.0,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,407.8375,91.5625,139.88666666666666
|
38 |
+
74.23,436.27,44.37,68.98,31.29,0.0,36.83,1.05,0.0,0.0,0.0,0.0,19,11,13,407.8375,-0.2588190451025208,-0.9659258262890683,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,91.5625,139.88666666666666,740.2125
|
39 |
+
52.0,0.0,48.93,73.25,32.02,0.0,39.16,0.98,0.0,0.0,0.0,0.0,19,11,14,91.5625,-0.4999999999999997,-0.8660254037844388,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,139.88666666666666,740.2125,419.85
|
40 |
+
52.52,159.83,42.77,65.19,31.09,0.0,34.67,0.91,0.0,0.0,0.0,0.0,19,11,15,139.88666666666666,-0.7071067811865471,-0.7071067811865479,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,740.2125,419.85,100.39999999999999
|
41 |
+
56.87,702.17,37.11,56.15,30.36,0.0,33.57,0.84,0.0,0.0,0.0,0.0,19,11,16,740.2125,-0.8660254037844385,-0.5000000000000004,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,419.85,100.39999999999999,95.4875
|
42 |
+
51.57,445.88,39.48,60.76,30.29,0.0,34.33,0.81,0.0,0.0,0.0,0.0,19,11,17,419.85,-0.9659258262890683,-0.25881904510252063,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,100.39999999999999,95.4875,195.07999999999998
|
43 |
+
39.21,100.6,43.26,67.11,30.88,0.0,34.35,0.83,0.0,0.0,0.0,0.0,19,11,18,100.39999999999999,-1.0,-1.8369701987210297e-16,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,95.4875,195.07999999999998,1043.5
|
44 |
+
0.0,0.0,49.8,76.39,32.2,0.0,34.83,0.94,0.0,0.0,0.0,0.0,19,11,19,95.4875,-0.9659258262890684,0.2588190451025203,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,195.07999999999998,1043.5,101.34
|
45 |
+
58.48,242.62,50.81,77.88,32.77,0.0,35.01,1.06,0.0,0.0,0.0,0.0,19,11,20,195.07999999999998,-0.8660254037844386,0.5000000000000001,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,1043.5,101.34,152.9
|
46 |
+
79.97,944.8,55.89,85.69,33.42,0.0,34.95,1.16,0.0,0.0,0.0,0.0,19,11,21,1043.5,-0.7071067811865477,0.7071067811865474,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,101.34,152.9,306.45000000000005
|
47 |
+
0.0,0.0,54.82,84.02,33.22,0.0,34.93,1.2,0.0,0.0,0.0,0.0,19,11,22,101.34,-0.5000000000000004,0.8660254037844384,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,152.9,306.45000000000005,200.89999999999998
|
48 |
+
75.87,95.92,56.49,86.79,33.3,0.0,35.02,1.22,0.0,0.0,0.0,0.0,19,11,23,152.9,-0.25881904510252157,0.9659258262890681,-0.651372482722222,-0.7587581226927911,-0.5000000000000004,0.8660254037844384,306.45000000000005,200.89999999999998,820.625
|
49 |
+
100.82,355.16,58.15,88.94,33.62,0.0,35.2,1.23,0.0,0.0,0.0,0.0,20,11,0,306.45000000000005,0.0,1.0,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,200.89999999999998,820.625,157.43333333333334
|
50 |
+
90.27,231.67,57.12,87.2,33.42,0.0,35.2,1.2,0.0,0.0,0.0,0.0,20,11,1,200.89999999999998,0.25881904510252074,0.9659258262890683,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,820.625,157.43333333333334,957.5875000000001
|
51 |
+
65.97,766.5,58.12,89.11,33.58,0.0,35.26,1.16,0.0,0.0,0.0,0.0,20,11,2,820.625,0.49999999999999994,0.8660254037844387,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,157.43333333333334,957.5875000000001,564.7125
|
52 |
+
64.95,186.15,58.82,90.21,33.7,0.0,35.27,1.11,0.0,0.0,0.0,0.0,20,11,3,157.43333333333334,0.7071067811865476,0.7071067811865476,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,957.5875000000001,564.7125,164.98000000000002
|
53 |
+
63.44,876.07,60.04,91.62,33.91,0.0,35.3,1.05,0.0,0.0,0.0,0.0,20,11,4,957.5875000000001,0.8660254037844386,0.5000000000000001,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,564.7125,164.98000000000002,176.93333333333334
|
54 |
+
79.73,561.77,58.7,89.78,33.88,0.0,35.37,1.01,0.0,0.0,0.0,0.0,20,11,5,564.7125,0.9659258262890683,0.25881904510252074,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,164.98000000000002,176.93333333333334,190.66666666666669
|
55 |
+
72.28,197.47,59.51,90.21,33.78,0.0,35.37,0.98,0.0,0.0,0.0,0.0,20,11,6,164.98000000000002,1.0,6.123233995736766e-17,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,176.93333333333334,190.66666666666669,458.425
|
56 |
+
83.08,93.72,57.69,88.41,33.71,0.0,35.26,0.96,0.0,0.0,0.0,0.0,20,11,7,176.93333333333334,0.9659258262890683,-0.25881904510252063,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,190.66666666666669,458.425,814.1875
|
57 |
+
87.2,146.97,54.61,82.63,33.24,0.0,35.42,0.98,0.0,0.0,0.0,0.0,20,11,8,190.66666666666669,0.8660254037844387,-0.4999999999999998,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,458.425,814.1875,59.712500000000006
|
58 |
+
95.65,476.74,49.53,75.45,32.61,0.0,36.28,1.05,0.0,0.0,0.0,0.0,20,11,9,458.425,0.7071067811865476,-0.7071067811865475,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,814.1875,59.712500000000006,352.3076923076923
|
59 |
+
119.95,761.35,45.88,68.69,31.81,0.0,45.23,1.17,0.0,0.0,0.0,0.0,20,11,10,814.1875,0.49999999999999994,-0.8660254037844387,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,59.712500000000006,352.3076923076923,350.46153846153845
|
60 |
+
0.0,0.0,32.17,47.77,29.9,0.0,43.63,1.12,0.0,0.0,0.0,0.0,20,11,11,59.712500000000006,0.258819045102521,-0.9659258262890682,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,352.3076923076923,350.46153846153845,284.59999999999997
|
61 |
+
188.0,0.0,25.49,38.39,28.41,0.0,35.53,0.7,0.0,0.0,0.0,0.0,20,11,12,352.3076923076923,1.2246467991473532e-16,-1.0,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,350.46153846153845,284.59999999999997,204.7
|
62 |
+
185.6,0.0,18.0,27.03,27.31,0.0,37.22,0.69,0.0,0.0,0.0,0.0,20,11,13,350.46153846153845,-0.2588190451025208,-0.9659258262890683,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,284.59999999999997,204.7,258.0
|
63 |
+
115.38,0.0,24.12,37.72,27.76,0.0,33.8,0.58,0.0,0.0,0.0,0.0,20,11,14,284.59999999999997,-0.4999999999999997,-0.8660254037844388,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,204.7,258.0,284.0
|
64 |
+
91.41,216.79,27.57,42.19,28.41,0.0,33.74,0.49,0.0,0.0,0.0,0.0,20,11,15,204.7,-0.7071067811865471,-0.7071067811865479,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,258.0,284.0,314.27500000000003
|
65 |
+
107.4,156.9,25.49,38.95,28.42,0.0,33.28,0.52,0.0,0.0,0.0,0.0,20,11,16,258.0,-0.8660254037844385,-0.5000000000000004,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,284.0,314.27500000000003,891.125
|
66 |
+
109.87,334.0,27.75,42.72,28.6,0.0,33.59,0.66,0.0,0.0,0.0,0.0,20,11,17,284.0,-0.9659258262890683,-0.25881904510252063,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,314.27500000000003,891.125,100.37666666666667
|
67 |
+
105.0,361.42,36.07,56.73,29.88,0.0,33.8,0.76,0.0,0.0,0.0,0.0,20,11,18,314.27500000000003,-1.0,-1.8369701987210297e-16,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,891.125,100.37666666666667,97.6625
|
68 |
+
105.98,822.9,45.57,70.72,31.39,0.0,34.45,0.89,0.0,0.0,0.0,0.0,20,11,19,891.125,-0.9659258262890684,0.2588190451025203,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,100.37666666666667,97.6625,96.13749999999999
|
69 |
+
0.0,0.0,52.85,81.13,32.57,0.0,34.48,1.05,0.0,0.0,0.0,0.0,20,11,20,100.37666666666667,-0.8660254037844386,0.5000000000000001,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,97.6625,96.13749999999999,100.64666666666666
|
70 |
+
0.0,0.0,51.25,78.13,32.62,0.0,34.53,1.12,0.0,0.0,0.0,0.0,20,11,21,97.6625,-0.7071067811865477,0.7071067811865474,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,96.13749999999999,100.64666666666666,100.93333333333334
|
71 |
+
0.0,0.0,50.0,76.91,32.34,0.0,34.66,1.13,0.0,0.0,0.0,0.0,20,11,22,96.13749999999999,-0.5000000000000004,0.8660254037844384,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,100.64666666666666,100.93333333333334,99.925
|
72 |
+
0.0,0.0,53.86,81.94,32.95,0.0,34.78,1.15,0.0,0.0,0.0,0.0,20,11,23,100.64666666666666,-0.25881904510252157,0.9659258262890681,-0.7907757369376986,-0.6121059825476627,-0.5000000000000004,0.8660254037844384,100.93333333333334,99.925,93.1125
|
73 |
+
0.0,0.0,54.16,82.8,33.19,0.0,34.88,1.14,0.0,0.0,0.0,0.0,21,11,0,100.93333333333334,0.0,1.0,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,99.925,93.1125,93.375
|
74 |
+
0.0,0.0,52.22,79.94,32.92,0.0,34.84,1.06,0.0,0.0,0.0,0.0,21,11,1,99.925,0.25881904510252074,0.9659258262890683,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,93.1125,93.375,90.22500000000001
|
75 |
+
0.0,0.0,49.0,74.49,32.36,0.0,34.81,0.97,0.0,0.0,0.0,0.0,21,11,2,93.1125,0.49999999999999994,0.8660254037844387,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,93.375,90.22500000000001,89.4125
|
76 |
+
0.0,0.0,49.03,74.7,32.18,0.0,34.82,0.9,0.0,0.0,0.0,0.0,21,11,3,93.375,0.7071067811865476,0.7071067811865476,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,90.22500000000001,89.4125,90.0375
|
77 |
+
0.0,0.0,47.35,72.18,32.05,0.0,34.8,0.84,0.0,0.0,0.0,0.0,21,11,4,90.22500000000001,0.8660254037844386,0.5000000000000001,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,89.4125,90.0375,94.675
|
78 |
+
0.0,0.0,46.49,71.53,31.92,0.0,34.78,0.81,0.0,0.0,0.0,0.0,21,11,5,89.4125,0.9659258262890683,0.25881904510252074,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,90.0375,94.675,100.22333333333333
|
79 |
+
0.0,0.0,47.38,72.03,31.91,0.0,34.78,0.8,0.0,0.0,0.0,0.0,21,11,6,90.0375,1.0,6.123233995736766e-17,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,94.675,100.22333333333333,97.225
|
80 |
+
0.0,0.0,49.1,75.74,32.26,0.0,34.85,0.82,0.0,0.0,0.0,0.0,21,11,7,94.675,0.9659258262890683,-0.25881904510252063,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,100.22333333333333,97.225,88.66250000000001
|
81 |
+
0.0,0.0,52.78,80.67,32.79,0.0,35.22,0.9,0.0,0.0,0.0,0.0,21,11,8,100.22333333333333,0.8660254037844387,-0.4999999999999998,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,97.225,88.66250000000001,84.71249999999999
|
82 |
+
0.0,0.0,51.19,77.78,32.78,0.0,35.75,0.98,0.0,0.0,0.0,0.0,21,11,9,97.225,0.7071067811865476,-0.7071067811865475,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,88.66250000000001,84.71249999999999,81.1875
|
83 |
+
0.0,0.0,47.1,70.93,32.03,0.0,35.25,1.05,0.0,0.0,0.0,0.0,21,11,10,88.66250000000001,0.49999999999999994,-0.8660254037844387,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,84.71249999999999,81.1875,78.89999999999999
|
84 |
+
0.0,0.0,44.49,67.77,31.56,0.0,35.83,1.14,0.0,0.0,0.0,0.0,21,11,11,84.71249999999999,0.258819045102521,-0.9659258262890682,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,81.1875,78.89999999999999,80.0375
|
85 |
+
0.0,0.0,42.59,64.95,31.29,0.0,35.76,1.12,0.0,0.0,0.0,0.0,21,11,12,81.1875,1.2246467991473532e-16,-1.0,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,78.89999999999999,80.0375,83.5375
|
86 |
+
0.0,0.0,41.34,63.12,31.04,0.0,35.08,1.06,0.0,0.0,0.0,0.0,21,11,13,78.89999999999999,-0.2588190451025208,-0.9659258262890683,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,80.0375,83.5375,82.3625
|
87 |
+
0.0,0.0,41.54,64.03,30.89,0.0,34.61,0.99,0.0,0.0,0.0,0.0,21,11,14,80.0375,-0.4999999999999997,-0.8660254037844388,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,83.5375,82.3625,82.2875
|
88 |
+
0.0,0.0,43.67,66.83,31.18,0.0,34.37,0.95,0.0,0.0,0.0,0.0,21,11,15,83.5375,-0.7071067811865471,-0.7071067811865479,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,82.3625,82.2875,89.225
|
89 |
+
0.0,0.0,43.32,65.89,31.28,0.0,34.09,0.92,0.0,0.0,0.0,0.0,21,11,16,82.3625,-0.8660254037844385,-0.5000000000000004,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,82.2875,89.225,94.825
|
90 |
+
0.0,0.0,42.7,65.83,31.16,0.0,34.03,0.9,0.0,0.0,0.0,0.0,21,11,17,82.2875,-0.9659258262890683,-0.25881904510252063,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,89.225,94.825,101.54
|
91 |
+
0.0,0.0,46.61,71.38,31.66,0.0,34.17,0.92,0.0,0.0,0.0,0.0,21,11,18,89.225,-1.0,-1.8369701987210297e-16,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,94.825,101.54,103.60333333333334
|
92 |
+
0.0,0.0,49.01,75.86,32.1,0.0,34.64,0.97,0.0,0.0,0.0,0.0,21,11,19,94.825,-0.9659258262890684,0.2588190451025203,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,101.54,103.60333333333334,103.41666666666667
|
93 |
+
0.0,0.0,54.66,84.62,32.92,0.0,34.77,1.04,0.0,0.0,0.0,0.0,21,11,20,101.54,-0.8660254037844386,0.5000000000000001,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,103.60333333333334,103.41666666666667,103.96
|
94 |
+
0.0,0.0,59.35,90.81,33.75,0.0,35.09,1.12,0.0,0.0,0.0,0.0,21,11,21,103.60333333333334,-0.7071067811865477,0.7071067811865474,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,103.41666666666667,103.96,105.2
|
95 |
+
0.0,0.0,59.49,90.25,33.92,0.0,35.29,1.14,0.0,0.0,0.0,0.0,21,11,22,103.41666666666667,-0.5000000000000004,0.8660254037844384,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,103.96,105.2,103.45333333333333
|
96 |
+
0.0,0.0,59.97,91.88,34.02,0.0,35.55,1.13,0.0,0.0,0.0,0.0,21,11,23,103.96,-0.25881904510252157,0.9659258262890681,-0.8978045395707417,-0.44039415155763423,-0.5000000000000004,0.8660254037844384,105.2,103.45333333333333,102.02
|
97 |
+
0.0,0.0,62.73,95.6,34.35,0.0,35.81,1.14,0.0,0.0,0.0,0.0,22,11,0,105.2,0.0,1.0,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,103.45333333333333,102.02,100.70666666666666
|
98 |
+
0.0,0.0,59.11,90.36,33.98,0.0,36.11,1.09,0.0,0.0,0.0,0.0,22,11,1,103.45333333333333,0.25881904510252074,0.9659258262890683,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,102.02,100.70666666666666,100.93666666666667
|
99 |
+
0.0,0.0,56.58,86.06,33.5,0.0,35.73,1.02,0.0,0.0,0.0,0.0,22,11,2,102.02,0.49999999999999994,0.8660254037844387,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,100.70666666666666,100.93666666666667,100.89
|
100 |
+
0.0,0.0,53.82,82.12,33.24,0.0,35.77,0.98,0.0,0.0,0.0,0.0,22,11,3,100.70666666666666,0.7071067811865476,0.7071067811865476,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,100.93666666666667,100.89,101.28
|
101 |
+
0.0,0.0,53.66,82.81,32.98,0.0,35.69,0.94,0.0,0.0,0.0,0.0,22,11,4,100.93666666666667,0.8660254037844386,0.5000000000000001,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,100.89,101.28,102.10666666666667
|
102 |
+
0.0,0.0,54.48,82.67,33.05,0.0,35.67,0.94,0.0,0.0,0.0,0.0,22,11,5,100.89,0.9659258262890683,0.25881904510252074,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,101.28,102.10666666666667,102.87333333333333
|
103 |
+
0.0,0.0,54.83,83.84,32.99,0.0,35.69,0.91,0.0,0.0,0.0,0.0,22,11,6,101.28,1.0,6.123233995736766e-17,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,102.10666666666667,102.87333333333333,95.8375
|
104 |
+
0.0,0.0,56.33,86.32,33.23,0.0,35.72,0.92,0.0,0.0,0.0,0.0,22,11,7,102.10666666666667,0.9659258262890683,-0.25881904510252063,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,102.87333333333333,95.8375,90.35
|
105 |
+
0.0,0.0,58.16,88.62,33.66,0.0,36.08,0.95,0.0,0.0,0.0,0.0,22,11,8,102.87333333333333,0.8660254037844387,-0.4999999999999998,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,95.8375,90.35,96.7125
|
106 |
+
0.0,0.0,50.79,76.67,32.95,0.0,36.52,0.99,0.0,0.0,0.0,0.0,22,11,9,95.8375,0.7071067811865476,-0.7071067811865475,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,90.35,96.7125,92.9625
|
107 |
+
0.0,0.0,47.41,72.28,32.08,0.0,36.54,1.08,0.0,0.0,0.0,0.0,22,11,10,90.35,0.49999999999999994,-0.8660254037844387,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,96.7125,92.9625,80.39999999999999
|
108 |
+
0.0,0.0,50.38,77.37,32.43,0.0,37.06,1.21,0.0,0.0,0.0,0.0,22,11,11,96.7125,0.258819045102521,-0.9659258262890682,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,92.9625,80.39999999999999,74.57499999999999
|
109 |
+
0.0,0.0,48.79,74.37,32.14,0.0,36.81,1.21,0.0,0.0,0.0,0.0,22,11,12,92.9625,1.2246467991473532e-16,-1.0,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,80.39999999999999,74.57499999999999,73.675
|
110 |
+
0.0,0.0,42.84,64.32,31.35,0.0,36.1,1.1,0.0,0.0,0.0,0.0,22,11,13,80.39999999999999,-0.2588190451025208,-0.9659258262890683,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,74.57499999999999,73.675,72.7625
|
111 |
+
0.0,0.0,39.15,59.66,30.61,0.0,35.48,0.99,0.0,0.0,0.0,0.0,22,11,14,74.57499999999999,-0.4999999999999997,-0.8660254037844388,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,73.675,72.7625,72.55
|
112 |
+
0.0,0.0,38.86,58.94,30.57,0.0,35.08,0.93,0.0,0.0,0.0,0.0,22,11,15,73.675,-0.7071067811865471,-0.7071067811865479,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,72.7625,72.55,85.9375
|
113 |
+
0.0,0.0,37.92,58.21,30.42,0.0,34.41,0.9,0.0,0.0,0.0,0.0,22,11,16,72.7625,-0.8660254037844385,-0.5000000000000004,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,72.55,85.9375,58.0
|
114 |
+
0.0,0.0,37.54,58.04,30.52,0.0,34.28,0.9,0.0,0.0,0.0,0.0,22,11,17,72.55,-0.9659258262890683,-0.25881904510252063,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,85.9375,58.0,284.20000000000005
|
115 |
+
0.0,0.0,45.18,68.75,31.37,0.0,35.16,0.95,0.0,0.0,0.0,0.0,22,11,18,85.9375,-1.0,-1.8369701987210297e-16,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,58.0,284.20000000000005,303.84615384615387
|
116 |
+
0.0,0.0,27.95,41.84,29.34,0.0,39.62,1.16,0.0,0.0,0.0,0.0,22,11,19,58.0,-0.9659258262890684,0.2588190451025203,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,284.20000000000005,303.84615384615387,306.60769230769233
|
117 |
+
115.26,300.6,38.22,59.84,30.06,0.0,35.62,1.21,0.0,0.0,0.0,0.0,22,11,20,284.20000000000005,-0.8660254037844386,0.5000000000000001,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,303.84615384615387,306.60769230769233,984.8375
|
118 |
+
125.0,0.0,45.61,70.38,31.35,0.0,35.86,1.27,0.0,0.0,0.0,0.0,22,11,21,303.84615384615387,-0.7071067811865477,0.7071067811865474,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,306.60769230769233,984.8375,565.3874999999999
|
119 |
+
128.59,214.31,49.12,75.78,32.06,0.0,35.92,1.32,0.0,0.0,0.0,0.0,22,11,22,306.60769230769233,-0.5000000000000004,0.8660254037844384,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,984.8375,565.3874999999999,100.11666666666666
|
120 |
+
125.03,897.87,53.75,81.87,32.71,0.0,36.02,1.33,0.0,0.0,0.0,0.0,22,11,23,984.8375,-0.25881904510252157,0.9659258262890681,-0.9680771188662041,-0.2506525322587213,-0.5000000000000004,0.8660254037844384,565.3874999999999,100.11666666666666,310.4846153846154
|
121 |
+
125.95,562.31,52.75,80.42,32.78,0.0,36.09,1.3,0.0,0.0,0.0,0.0,23,11,0,565.3874999999999,0.0,1.0,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,100.11666666666666,310.4846153846154,324.46153846153845
|
122 |
+
0.0,0.0,52.23,80.35,32.66,0.0,36.15,1.26,0.0,0.0,0.0,0.0,23,11,1,100.11666666666666,0.25881904510252074,0.9659258262890683,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,310.4846153846154,324.46153846153845,102.88666666666667
|
123 |
+
133.63,307.45,55.58,85.64,33.36,0.0,36.29,0.9,0.0,0.0,0.0,0.0,23,11,2,310.4846153846154,0.49999999999999994,0.8660254037844387,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,324.46153846153845,102.88666666666667,320.3461538461538
|
124 |
+
151.8,221.22,56.48,86.44,33.44,0.0,36.4,1.19,0.0,0.0,0.0,0.0,23,11,3,324.46153846153845,0.7071067811865476,0.7071067811865476,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,102.88666666666667,320.3461538461538,691.4375
|
125 |
+
0.0,0.0,58.45,88.66,33.67,0.0,36.51,1.16,0.0,0.0,0.0,0.0,23,11,4,102.88666666666667,0.8660254037844386,0.5000000000000001,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,320.3461538461538,691.4375,576.225
|
126 |
+
146.45,225.24,57.79,88.81,33.68,0.0,36.53,1.14,0.0,0.0,0.0,0.0,23,11,5,320.3461538461538,0.9659258262890683,0.25881904510252074,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,691.4375,576.225,322.4307692307692
|
127 |
+
133.13,663.15,58.39,89.09,33.65,0.0,36.61,1.11,0.0,0.0,0.0,0.0,23,11,6,691.4375,1.0,6.123233995736766e-17,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,576.225,322.4307692307692,479.8125
|
128 |
+
137.4,570.98,54.36,82.59,33.24,0.0,36.71,1.05,0.0,0.0,0.0,0.0,23,11,7,576.225,0.9659258262890683,-0.25881904510252063,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,322.4307692307692,479.8125,352.0
|
129 |
+
149.16,276.42,52.91,80.58,32.83,0.0,37.28,1.06,0.0,0.0,0.0,0.0,23,11,8,322.4307692307692,0.8660254037844387,-0.4999999999999998,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,479.8125,352.0,977.3375
|
130 |
+
156.1,493.85,58.2,89.68,33.43,0.0,38.24,1.14,0.0,0.0,0.0,0.0,23,11,9,479.8125,0.7071067811865476,-0.7071067811865475,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,352.0,977.3375,376.1538461538462
|
131 |
+
187.6,372.88,59.95,91.19,33.92,0.0,38.26,1.23,0.0,0.0,0.0,0.0,23,11,10,352.0,0.49999999999999994,-0.8660254037844387,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,977.3375,376.1538461538462,583.3125
|
132 |
+
176.0,891.87,48.2,71.9,32.47,0.0,37.66,1.08,0.0,0.0,0.0,0.0,23,11,11,977.3375,0.258819045102521,-0.9659258262890682,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,376.1538461538462,583.3125,55.9375
|
133 |
+
219.0,337.74,37.65,56.69,30.61,0.0,36.4,0.9,0.0,0.0,0.0,0.0,23,11,12,376.1538461538462,1.2246467991473532e-16,-1.0,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,583.3125,55.9375,60.337500000000006
|
134 |
+
216.34,576.65,30.61,45.82,29.38,0.0,35.7,0.82,0.0,0.0,0.0,0.0,23,11,13,583.3125,-0.2588190451025208,-0.9659258262890683,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,55.9375,60.337500000000006,283.5
|
135 |
+
0.0,0.0,29.29,44.75,28.96,0.0,35.25,0.73,0.0,0.0,0.0,0.0,23,11,14,55.9375,-0.4999999999999997,-0.8660254037844388,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,60.337500000000006,283.5,65.4375
|
136 |
+
0.0,0.0,31.01,48.27,29.12,0.0,34.77,0.79,0.0,0.0,0.0,0.0,23,11,15,60.337500000000006,-0.7071067811865471,-0.7071067811865479,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,283.5,65.4375,74.57499999999999
|
137 |
+
115.05,118.07,34.55,52.38,29.58,0.0,34.37,0.81,0.0,0.0,0.0,0.0,23,11,16,283.5,-0.8660254037844385,-0.5000000000000004,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,65.4375,74.57499999999999,308.66153846153844
|
138 |
+
0.0,0.0,34.21,52.35,29.77,0.0,34.66,0.86,0.0,0.0,0.0,0.0,23,11,17,65.4375,-0.9659258262890683,-0.25881904510252063,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,74.57499999999999,308.66153846153844,316.7153846153846
|
139 |
+
0.0,0.0,38.27,59.66,30.44,0.0,35.04,0.95,0.0,0.0,0.0,0.0,23,11,18,74.57499999999999,-1.0,-1.8369701987210297e-16,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,308.66153846153844,316.7153846153846,332.2846153846154
|
140 |
+
131.26,139.7,42.78,66.11,30.97,0.0,35.35,1.02,0.0,0.0,0.0,0.0,23,11,19,308.66153846153844,-0.9659258262890684,0.2588190451025203,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,316.7153846153846,332.2846153846154,325.96153846153845
|
141 |
+
141.73,153.58,51.0,79.18,32.22,0.0,35.46,1.12,0.0,0.0,0.0,0.0,23,11,20,316.7153846153846,-0.8660254037844386,0.5000000000000001,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,332.2846153846154,325.96153846153845,109.59333333333333
|
142 |
+
161.97,163.0,58.85,90.77,33.53,0.0,35.88,1.23,0.0,0.0,0.0,0.0,23,11,21,332.2846153846154,-0.7071067811865477,0.7071067811865474,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,325.96153846153845,109.59333333333333,111.18666666666667
|
143 |
+
153.75,163.16,65.06,100.68,34.83,0.0,36.28,1.39,0.0,0.0,0.0,0.0,23,11,22,325.96153846153845,-0.5000000000000004,0.8660254037844384,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,109.59333333333333,111.18666666666667,112.13666666666667
|
144 |
+
0.0,0.0,71.26,108.78,35.65,0.0,36.54,1.55,0.0,0.0,0.0,0.0,23,11,23,109.59333333333333,-0.25881904510252157,0.9659258262890681,-0.9987165071710528,-0.05064916883871355,-0.5000000000000004,0.8660254037844384,111.18666666666667,112.13666666666667,331.5153846153846
|
145 |
+
0.0,0.0,73.75,113.56,36.27,0.0,36.77,1.59,0.0,0.0,0.0,0.0,24,11,0,111.18666666666667,0.0,1.0,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,112.13666666666667,331.5153846153846,324.3538461538461
|
146 |
+
0.0,0.0,76.07,116.41,36.54,0.0,37.01,1.59,0.0,0.0,0.0,0.0,24,11,1,112.13666666666667,0.25881904510252074,0.9659258262890683,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,331.5153846153846,324.3538461538461,314.7769230769231
|
147 |
+
160.97,162.2,80.42,122.59,37.27,0.0,37.25,1.6,0.0,0.0,0.0,0.0,24,11,2,331.5153846153846,0.49999999999999994,0.8660254037844387,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,324.3538461538461,314.7769230769231,321.2076923076923
|
148 |
+
151.66,163.41,79.01,121.54,37.2,0.0,37.42,1.53,0.0,0.0,0.0,0.0,24,11,3,324.3538461538461,0.7071067811865476,0.7071067811865476,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,314.7769230769231,321.2076923076923,114.25333333333333
|
149 |
+
139.21,170.07,80.13,121.9,37.08,0.0,37.5,1.47,0.0,0.0,0.0,0.0,24,11,4,314.7769230769231,0.8660254037844386,0.5000000000000001,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,321.2076923076923,114.25333333333333,112.11333333333333
|
150 |
+
147.57,152.9,80.44,123.25,37.38,0.0,37.45,1.4,0.0,0.0,0.0,0.0,24,11,5,321.2076923076923,0.9659258262890683,0.25881904510252074,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,114.25333333333333,112.11333333333333,113.59
|
151 |
+
0.0,0.0,80.37,122.76,37.26,0.0,37.42,1.34,0.0,0.0,0.0,0.0,24,11,6,114.25333333333333,1.0,6.123233995736766e-17,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,112.11333333333333,113.59,113.01
|
152 |
+
0.0,0.0,76.6,116.34,36.64,0.0,37.33,1.27,0.0,0.0,0.0,0.0,24,11,7,112.11333333333333,0.9659258262890683,-0.25881904510252063,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,113.59,113.01,110.66666666666667
|
153 |
+
0.0,0.0,78.53,120.77,36.8,0.0,37.81,1.31,0.0,0.0,0.0,0.0,24,11,8,113.59,0.8660254037844387,-0.4999999999999998,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,113.01,110.66666666666667,109.90333333333334
|
154 |
+
0.0,0.0,78.0,119.03,37.2,0.0,38.56,1.36,0.0,0.0,0.0,0.0,24,11,9,113.01,0.7071067811865476,-0.7071067811865475,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,110.66666666666667,109.90333333333334,301.94615384615383
|
155 |
+
0.0,0.0,73.43,112.0,36.34,0.0,38.56,1.37,0.0,0.0,0.0,0.0,24,11,10,110.66666666666667,0.49999999999999994,-0.8660254037844387,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,109.90333333333334,301.94615384615383,102.17
|
156 |
+
0.0,0.0,72.2,109.71,36.06,0.0,38.05,1.44,0.0,0.0,0.0,0.0,24,11,11,109.90333333333334,0.258819045102521,-0.9659258262890682,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,301.94615384615383,102.17,240.7
|
157 |
+
122.53,174.6,65.94,100.0,35.19,0.0,37.8,1.43,0.0,0.0,0.0,0.0,24,11,12,301.94615384615383,1.2246467991473532e-16,-1.0,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,102.17,240.7,266.6666666666667
|
158 |
+
0.0,0.0,58.12,86.51,33.6,0.0,36.79,1.28,0.0,0.0,0.0,0.0,24,11,13,102.17,-0.2588190451025208,-0.9659258262890683,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,240.7,266.6666666666667,88.25
|
159 |
+
102.21,118.79,47.29,71.95,31.97,0.0,35.92,1.15,0.0,0.0,0.0,0.0,24,11,14,240.7,-0.4999999999999997,-0.8660254037844388,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,266.6666666666667,88.25,196.76666666666665
|
160 |
+
110.0,115.0,45.57,70.67,31.81,0.0,35.92,1.09,0.0,0.0,0.0,0.0,24,11,15,266.6666666666667,-0.7071067811865471,-0.7071067811865479,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,88.25,196.76666666666665,101.87
|
161 |
+
0.0,0.0,46.35,70.6,31.66,0.0,35.66,1.03,0.0,0.0,0.0,0.0,24,11,16,88.25,-0.8660254037844385,-0.5000000000000004,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,196.76666666666665,101.87,136.76666666666665
|
162 |
+
89.03,96.31,47.59,73.25,31.91,0.0,35.26,0.99,0.0,0.0,0.0,0.0,24,11,17,196.76666666666665,-0.9659258262890683,-0.25881904510252063,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,101.87,136.76666666666665,143.96666666666667
|
163 |
+
16.45,102.67,55.21,85.61,33.02,0.0,35.56,1.01,0.0,0.0,0.0,0.0,24,11,18,101.87,-1.0,-1.8369701987210297e-16,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,136.76666666666665,143.96666666666667,295.40000000000003
|
164 |
+
71.03,117.02,62.76,95.28,34.31,0.0,36.03,0.99,0.0,0.0,0.0,0.0,24,11,19,136.76666666666665,-0.9659258262890684,0.2588190451025203,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,143.96666666666667,295.40000000000003,112.41666666666667
|
165 |
+
73.19,153.95,59.55,92.24,33.86,0.0,35.9,1.06,0.0,0.0,0.0,0.0,24,11,20,143.96666666666667,-0.8660254037844386,0.5000000000000001,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,295.40000000000003,112.41666666666667,319.2307692307692
|
166 |
+
118.62,152.33,67.72,104.6,34.92,0.0,36.3,1.16,0.0,0.0,0.0,0.0,24,11,21,295.40000000000003,-0.7071067811865477,0.7071067811865474,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,112.41666666666667,319.2307692307692,319.2692307692308
|
167 |
+
0.0,0.0,76.47,117.25,36.6,0.0,36.65,1.27,0.0,0.0,0.0,0.0,24,11,22,112.41666666666667,-0.5000000000000004,0.8660254037844384,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,319.2307692307692,319.2692307692308,320.6923076923077
|
168 |
+
145.0,147.05,80.63,123.09,37.44,0.0,37.0,1.41,0.0,0.0,0.0,0.0,24,11,23,319.2307692307692,-0.25881904510252157,0.9659258262890681,-0.9884683243281114,0.15142777750457667,-0.5000000000000004,0.8660254037844384,319.2692307692308,320.6923076923077,116.83333333333333
|
169 |
+
145.05,148.52,83.77,128.87,37.97,0.0,37.35,1.5,0.0,0.0,0.0,0.0,25,11,0,319.2692307692308,0.0,1.0,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,320.6923076923077,116.83333333333333,117.15666666666667
|
170 |
+
146.9,156.71,86.84,133.74,38.43,0.0,37.49,1.59,0.0,0.0,0.0,0.0,25,11,1,320.6923076923077,0.25881904510252074,0.9659258262890683,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,116.83333333333333,117.15666666666667,302.33076923076925
|
171 |
+
0.0,0.0,85.48,130.5,38.48,0.0,37.4,1.57,0.0,0.0,0.0,0.0,25,11,2,116.83333333333333,0.49999999999999994,0.8660254037844387,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,117.15666666666667,302.33076923076925,111.89666666666666
|
172 |
+
0.0,0.0,86.49,131.47,38.11,0.0,37.38,1.47,0.0,0.0,0.0,0.0,25,11,3,117.15666666666667,0.7071067811865476,0.7071067811865476,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,302.33076923076925,111.89666666666666,247.0
|
173 |
+
123.03,146.07,81.33,123.92,37.72,0.0,37.2,1.37,0.0,0.0,0.0,0.0,25,11,4,302.33076923076925,0.8660254037844386,0.5000000000000001,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,111.89666666666666,247.0,103.81
|
174 |
+
0.0,0.0,76.84,115.69,36.79,0.0,37.03,1.24,0.0,0.0,0.0,0.0,25,11,5,111.89666666666666,0.9659258262890683,0.25881904510252074,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,247.0,103.81,104.08333333333333
|
175 |
+
104.1,118.07,68.84,104.7,35.78,0.0,36.76,1.11,0.0,0.0,0.0,0.0,25,11,6,247.0,1.0,6.123233995736766e-17,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,103.81,104.08333333333333,102.73333333333333
|
176 |
+
0.0,0.0,60.21,91.43,34.15,0.0,36.41,1.02,0.0,0.0,0.0,0.0,25,11,7,103.81,0.9659258262890683,-0.25881904510252063,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,104.08333333333333,102.73333333333333,101.76666666666667
|
177 |
+
0.0,0.0,60.21,92.25,34.06,0.0,36.37,1.05,0.0,0.0,0.0,0.0,25,11,8,104.08333333333333,0.8660254037844387,-0.4999999999999998,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,102.73333333333333,101.76666666666667,101.62666666666667
|
178 |
+
0.0,0.0,57.97,88.2,33.86,0.0,36.66,1.09,0.0,0.0,0.0,0.0,25,11,9,102.73333333333333,0.7071067811865476,-0.7071067811865475,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,101.76666666666667,101.62666666666667,116.14666666666668
|
179 |
+
0.0,0.0,56.02,85.3,33.36,0.0,36.51,1.13,0.0,0.0,0.0,0.0,25,11,10,101.76666666666667,0.49999999999999994,-0.8660254037844387,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,101.62666666666667,116.14666666666668,229.23333333333332
|
180 |
+
0.0,0.0,55.82,84.88,33.28,0.0,36.56,1.17,0.0,0.0,0.0,0.0,25,11,11,101.62666666666667,0.258819045102521,-0.9659258262890682,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,116.14666666666668,229.23333333333332,199.13333333333333
|
181 |
+
49.85,124.22,35.27,51.23,30.56,0.0,38.06,1.07,0.0,0.0,0.0,0.0,25,11,12,116.14666666666668,1.2246467991473532e-16,-1.0,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,229.23333333333332,199.13333333333333,167.7
|
182 |
+
98.77,157.41,29.54,44.57,29.02,0.0,35.37,0.79,0.0,0.0,0.0,0.0,25,11,13,229.23333333333332,-0.2588190451025208,-0.9659258262890683,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,199.13333333333333,167.7,51.8375
|
183 |
+
89.74,108.86,26.87,41.14,28.47,0.0,34.93,0.67,0.0,0.0,0.0,0.0,25,11,14,199.13333333333333,-0.4999999999999997,-0.8660254037844388,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,167.7,51.8375,108.03333333333332
|
184 |
+
80.31,87.53,29.38,45.46,28.92,0.0,34.91,0.6,0.0,0.0,0.0,0.0,25,11,15,167.7,-0.7071067811865471,-0.7071067811865479,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,51.8375,108.03333333333332,160.0
|
185 |
+
0.0,0.0,27.92,41.47,28.7,0.0,35.09,0.55,0.0,0.0,0.0,0.0,25,11,16,51.8375,-0.8660254037844385,-0.5000000000000004,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,108.03333333333332,160.0,102.09333333333333
|
186 |
+
62.41,82.9,30.1,47.35,28.95,0.0,34.76,0.66,0.0,0.0,0.0,0.0,25,11,17,108.03333333333332,-0.9659258262890683,-0.25881904510252063,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,160.0,102.09333333333333,106.89333333333333
|
187 |
+
78.0,81.0,38.42,60.95,30.32,0.0,35.27,0.76,0.0,0.0,0.0,0.0,25,11,18,160.0,-1.0,-1.8369701987210297e-16,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,102.09333333333333,106.89333333333333,207.53333333333336
|
188 |
+
0.0,0.0,55.52,86.28,32.88,0.0,35.66,0.94,0.0,0.0,0.0,0.0,25,11,19,102.09333333333333,-0.9659258262890684,0.2588190451025203,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,106.89333333333333,207.53333333333336,109.64666666666666
|
189 |
+
0.0,0.0,64.88,100.68,34.53,0.0,35.8,1.15,0.0,0.0,0.0,0.0,25,11,20,106.89333333333333,-0.8660254037844386,0.5000000000000001,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,207.53333333333336,109.64666666666666,199.66666666666669
|
190 |
+
92.26,99.45,69.42,106.41,35.35,0.0,36.08,1.29,0.0,0.0,0.0,0.0,25,11,21,207.53333333333336,-0.7071067811865477,0.7071067811865474,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,109.64666666666666,199.66666666666669,185.9666666666667
|
191 |
+
0.0,0.0,71.15,108.94,35.97,0.0,36.3,1.38,0.0,0.0,0.0,0.0,25,11,22,109.64666666666666,-0.5000000000000004,0.8660254037844384,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,199.66666666666669,185.9666666666667,160.0
|
192 |
+
89.9,105.95,73.9,112.93,36.25,0.0,36.49,1.49,0.0,0.0,0.0,0.0,25,11,23,199.66666666666669,-0.25881904510252157,0.9659258262890681,-0.9377521321470804,0.3473052528448203,-0.5000000000000004,0.8660254037844384,185.9666666666667,160.0,109.91666666666667
|
193 |
+
85.79,104.43,75.86,115.85,36.54,0.0,36.54,1.52,0.0,0.0,0.0,0.0,26,11,0,185.9666666666667,0.0,1.0,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,160.0,109.91666666666667,157.23333333333335
|
194 |
+
78.0,92.34,74.79,114.09,36.43,0.0,36.44,1.47,0.0,0.0,0.0,0.0,26,11,1,160.0,0.25881904510252074,0.9659258262890683,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,109.91666666666667,157.23333333333335,172.56666666666666
|
195 |
+
0.0,0.0,72.16,109.75,35.95,0.0,36.43,1.39,0.0,0.0,0.0,0.0,26,11,2,109.91666666666667,0.49999999999999994,0.8660254037844387,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,157.23333333333335,172.56666666666666,149.66666666666669
|
196 |
+
77.17,81.7,69.42,105.79,35.56,0.0,36.39,1.3,0.0,0.0,0.0,0.0,26,11,3,157.23333333333335,0.7071067811865476,0.7071067811865476,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,172.56666666666666,149.66666666666669,101.92
|
197 |
+
81.77,92.01,68.08,103.74,35.42,0.0,36.32,1.2,0.0,0.0,0.0,0.0,26,11,4,172.56666666666666,0.8660254037844386,0.5000000000000001,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,149.66666666666669,101.92,96.25
|
198 |
+
74.9,75.74,64.32,97.55,34.85,0.0,36.19,1.11,0.0,0.0,0.0,0.0,26,11,5,149.66666666666669,0.9659258262890683,0.25881904510252074,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,101.92,96.25,143.16666666666669
|
199 |
+
0.0,0.0,56.51,85.76,33.82,0.0,36.16,1.01,0.0,0.0,0.0,0.0,26,11,6,101.92,1.0,6.123233995736766e-17,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,96.25,143.16666666666669,137.56666666666666
|
200 |
+
0.0,0.0,50.71,77.0,32.5,0.0,36.01,0.94,0.0,0.0,0.0,0.0,26,11,7,96.25,0.9659258262890683,-0.25881904510252063,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,143.16666666666669,137.56666666666666,163.33333333333334
|
201 |
+
72.95,86.28,49.87,76.36,32.15,0.0,36.36,0.94,0.0,0.0,0.0,0.0,26,11,8,143.16666666666669,0.8660254037844387,-0.4999999999999998,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,137.56666666666666,163.33333333333334,86.6375
|
202 |
+
71.27,73.53,50.21,77.14,32.31,0.0,36.9,0.98,0.0,0.0,0.0,0.0,26,11,9,137.56666666666666,0.7071067811865476,-0.7071067811865475,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,163.33333333333334,86.6375,
|
203 |
+
79.0,82.0,49.16,74.06,32.28,0.0,36.56,1.01,0.0,0.0,0.0,0.0,26,11,10,163.33333333333334,0.49999999999999994,-0.8660254037844387,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,86.6375,,
|
204 |
+
0.0,0.0,45.42,69.31,31.8,0.0,36.55,1.05,0.0,0.0,0.0,0.0,26,11,10,86.6375,0.49999999999999994,-0.8660254037844387,-0.8486442574947509,0.5289640103269624,-0.5000000000000004,0.8660254037844384,,,
|
templates/aqi_forecast_with_legend.html
ADDED
@@ -0,0 +1,187 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
|
4 |
+
<head>
|
5 |
+
<meta charset="UTF-8">
|
6 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
7 |
+
<title>AQI Forecast Map</title>
|
8 |
+
<!-- Add Bootstrap 5 CSS -->
|
9 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.0/css/bootstrap.min.css">
|
10 |
+
<style>
|
11 |
+
body {
|
12 |
+
font-family: Arial, sans-serif;
|
13 |
+
background-color: #f8f9fa;
|
14 |
+
margin: 0;
|
15 |
+
padding: 0;
|
16 |
+
}
|
17 |
+
|
18 |
+
/* Thinner Navigation Bar */
|
19 |
+
.navbar {
|
20 |
+
position: fixed;
|
21 |
+
top: 0;
|
22 |
+
left: 0;
|
23 |
+
width: 100%;
|
24 |
+
z-index: 1000;
|
25 |
+
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
26 |
+
padding: 3px 10px;
|
27 |
+
/* Reduced padding for a thinner navbar */
|
28 |
+
font-size: 0.85rem;
|
29 |
+
/* Slightly smaller font size */
|
30 |
+
height: 50px;
|
31 |
+
/* Explicit height for consistent thinness */
|
32 |
+
}
|
33 |
+
|
34 |
+
/* Reduced brand name size */
|
35 |
+
.navbar-brand {
|
36 |
+
font-size: 1rem;
|
37 |
+
/* Smaller font size for brand */
|
38 |
+
padding: 0;
|
39 |
+
/* Remove extra padding */
|
40 |
+
line-height: 50px;
|
41 |
+
/* Center-align text vertically */
|
42 |
+
}
|
43 |
+
|
44 |
+
/* Adjust the navbar toggler button */
|
45 |
+
.navbar-toggler {
|
46 |
+
padding: 2px 8px;
|
47 |
+
/* Smaller padding for toggler */
|
48 |
+
line-height: 1;
|
49 |
+
/* Adjust spacing */
|
50 |
+
}
|
51 |
+
|
52 |
+
/* Adjust form-container inside navbar */
|
53 |
+
.form-container {
|
54 |
+
margin-bottom: 0;
|
55 |
+
display: flex;
|
56 |
+
align-items: center; /* Vertically align form fields and button */
|
57 |
+
}
|
58 |
+
|
59 |
+
/* Reduce size of form inputs and button */
|
60 |
+
.form-control {
|
61 |
+
font-size: 0.85rem;
|
62 |
+
padding: 4px 6px;
|
63 |
+
height: auto;
|
64 |
+
flex-grow: 1; /* Ensure input fields expand proportionally */
|
65 |
+
min-width: 50px; /* Prevent inputs from shrinking too much */
|
66 |
+
margin-right: 5px; /* Spacing between input fields */
|
67 |
+
}
|
68 |
+
|
69 |
+
.btn-outline-success {
|
70 |
+
font-size: 0.85rem;
|
71 |
+
padding: 4px 8px;
|
72 |
+
height: auto;
|
73 |
+
white-space: nowrap; /* Prevent button text from wrapping */
|
74 |
+
}
|
75 |
+
|
76 |
+
/* Content area */
|
77 |
+
.content {
|
78 |
+
display: flex;
|
79 |
+
flex-direction: column;
|
80 |
+
height: 80vh;
|
81 |
+
/* Full height */
|
82 |
+
padding-top: 50px;
|
83 |
+
/* Reduced space for the smaller navbar */
|
84 |
+
}
|
85 |
+
|
86 |
+
/* Map Container to fill the remaining space */
|
87 |
+
.map-container {
|
88 |
+
flex: 1;
|
89 |
+
margin-bottom: 20px;
|
90 |
+
background-color: #e9ecef;
|
91 |
+
/* Optional: set a background color for the map container */
|
92 |
+
}
|
93 |
+
|
94 |
+
/* AQI Legend box */
|
95 |
+
#legend {
|
96 |
+
position: fixed;
|
97 |
+
bottom: 20px;
|
98 |
+
left: 20px;
|
99 |
+
width: 220px;
|
100 |
+
/* Reduced the width */
|
101 |
+
background-color: rgba(255, 255, 255, 0.6);
|
102 |
+
z-index: 9999;
|
103 |
+
font-size: 12px;
|
104 |
+
/* Reduced font size */
|
105 |
+
border-radius: 8px;
|
106 |
+
padding: 10px;
|
107 |
+
/* Reduced padding */
|
108 |
+
box-shadow: 2px 2px 8px rgba(0, 0, 0, 0.3);
|
109 |
+
}
|
110 |
+
|
111 |
+
.form-control {
|
112 |
+
margin-right: 10px;
|
113 |
+
}
|
114 |
+
|
115 |
+
/* List styling for AQI legend */
|
116 |
+
.legend-list li {
|
117 |
+
margin-bottom: 6px;
|
118 |
+
}
|
119 |
+
|
120 |
+
.legend-list i {
|
121 |
+
width: 18px;
|
122 |
+
/* Reduced the size of the color box */
|
123 |
+
height: 18px;
|
124 |
+
display: inline-block;
|
125 |
+
margin-right: 8px;
|
126 |
+
}
|
127 |
+
</style>
|
128 |
+
</head>
|
129 |
+
|
130 |
+
<body>
|
131 |
+
|
132 |
+
<!-- Fixed Navigation Bar -->
|
133 |
+
<nav class="navbar navbar-expand-lg navbar-dark bg-dark">
|
134 |
+
<div class="container-fluid">
|
135 |
+
<!-- Brand -->
|
136 |
+
<a class="navbar-brand" href="#">AQI Forecaster</a>
|
137 |
+
|
138 |
+
<!-- Toggler for mobile view -->
|
139 |
+
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav"
|
140 |
+
aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
|
141 |
+
<span class="navbar-toggler-icon"></span>
|
142 |
+
</button>
|
143 |
+
|
144 |
+
<!-- Navigation content -->
|
145 |
+
<div class="collapse navbar-collapse" id="navbarNav">
|
146 |
+
<div class="w-100 d-flex justify-content-between align-items-center">
|
147 |
+
<!-- Form for latitude and longitude -->
|
148 |
+
<form class="d-flex flex-grow-1 form-container me-2" method="POST" action="{{ url_for('forecast') }}">
|
149 |
+
<input class="form-control me-2" type="text" name="latitude" placeholder="Enter Latitude" required>
|
150 |
+
<input class="form-control me-2" type="text" name="longitude" placeholder="Enter Longitude" required>
|
151 |
+
<button class="btn btn-outline-success" type="submit">Update Map</button>
|
152 |
+
</form>
|
153 |
+
</div>
|
154 |
+
</div>
|
155 |
+
</div>
|
156 |
+
</nav>
|
157 |
+
|
158 |
+
|
159 |
+
|
160 |
+
<div class="content">
|
161 |
+
<div class="map-container">
|
162 |
+
<!-- AQI Forecast Map -->
|
163 |
+
<div>
|
164 |
+
<!-- Render the map -->
|
165 |
+
{{ map_html|safe }}
|
166 |
+
</div>
|
167 |
+
</div>
|
168 |
+
|
169 |
+
<!-- AQI Legend -->
|
170 |
+
<div id="legend">
|
171 |
+
<b>AQI Color Legend</b>
|
172 |
+
<ul class="legend-list" style="list-style: none; padding: 0; margin: 5px;">
|
173 |
+
<li><i style="background:green;"></i> Good (0-50)</li>
|
174 |
+
<li><i style="background:yellow;"></i> Moderate (51-100)</li>
|
175 |
+
<li><i style="background:orange;"></i> Unhealthy for Sensitive Groups (101-150)</li>
|
176 |
+
<li><i style="background:red;"></i> Unhealthy (151-200)</li>
|
177 |
+
<li><i style="background:purple;"></i> Very Unhealthy (201-300)</li>
|
178 |
+
<li><i style="background:maroon;"></i> Hazardous (301+)</li>
|
179 |
+
</ul>
|
180 |
+
</div>
|
181 |
+
</div>
|
182 |
+
|
183 |
+
<!-- Add Bootstrap 5 JS -->
|
184 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.0/js/bootstrap.bundle.min.js"></script>
|
185 |
+
</body>
|
186 |
+
|
187 |
+
</html>
|
templates/index.html
ADDED
@@ -0,0 +1,134 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html lang="en">
|
3 |
+
|
4 |
+
<head>
|
5 |
+
<meta charset="UTF-8">
|
6 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
7 |
+
<title>AQI Forecast Map</title>
|
8 |
+
<!-- Add Bootstrap 5 CSS -->
|
9 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.0/css/bootstrap.min.css">
|
10 |
+
<style>
|
11 |
+
body {
|
12 |
+
font-family: Arial, sans-serif;
|
13 |
+
background-color: #f8f9fa;
|
14 |
+
margin: 0;
|
15 |
+
padding: 0;
|
16 |
+
}
|
17 |
+
|
18 |
+
/* Fixed Navigation Bar at the top */
|
19 |
+
.navbar {
|
20 |
+
position: fixed;
|
21 |
+
top: 0;
|
22 |
+
left: 0;
|
23 |
+
width: 100%;
|
24 |
+
z-index: 1000;
|
25 |
+
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
26 |
+
}
|
27 |
+
|
28 |
+
/* Content area */
|
29 |
+
.content {
|
30 |
+
display: flex;
|
31 |
+
flex-direction: column;
|
32 |
+
height: 100vh; /* Full height */
|
33 |
+
padding-top: 80px; /* Space for fixed navbar */
|
34 |
+
}
|
35 |
+
|
36 |
+
/* Map Container to fill the remaining space */
|
37 |
+
.map-container {
|
38 |
+
flex: 1;
|
39 |
+
margin-bottom: 20px;
|
40 |
+
background-color: #e9ecef; /* Optional: set a background color for the map container */
|
41 |
+
}
|
42 |
+
|
43 |
+
/* AQI Legend box */
|
44 |
+
#legend {
|
45 |
+
position: fixed;
|
46 |
+
bottom: 20px;
|
47 |
+
left: 20px;
|
48 |
+
width: 220px; /* Reduced the width */
|
49 |
+
background-color: rgba(255, 255, 255, 0.8);
|
50 |
+
z-index: 9999;
|
51 |
+
font-size: 12px; /* Reduced font size */
|
52 |
+
border-radius: 8px;
|
53 |
+
padding: 10px; /* Reduced padding */
|
54 |
+
box-shadow: 2px 2px 8px rgba(0, 0, 0, 0.3);
|
55 |
+
}
|
56 |
+
|
57 |
+
/* Navbar style adjustments */
|
58 |
+
.navbar-brand {
|
59 |
+
font-size: 1.5rem;
|
60 |
+
}
|
61 |
+
|
62 |
+
/* Container for the form */
|
63 |
+
.form-container {
|
64 |
+
margin-bottom: 30px;
|
65 |
+
}
|
66 |
+
|
67 |
+
/* Form styling */
|
68 |
+
.form-control {
|
69 |
+
margin-right: 10px;
|
70 |
+
}
|
71 |
+
|
72 |
+
/* List styling for AQI legend */
|
73 |
+
.legend-list li {
|
74 |
+
margin-bottom: 6px;
|
75 |
+
}
|
76 |
+
|
77 |
+
.legend-list i {
|
78 |
+
width: 18px; /* Reduced the size of the color box */
|
79 |
+
height: 18px;
|
80 |
+
display: inline-block;
|
81 |
+
margin-right: 8px;
|
82 |
+
}
|
83 |
+
</style>
|
84 |
+
</head>
|
85 |
+
|
86 |
+
<body>
|
87 |
+
|
88 |
+
<!-- Fixed Navigation Bar -->
|
89 |
+
<nav class="navbar navbar-expand-lg navbar-dark bg-dark">
|
90 |
+
<div class="container-fluid">
|
91 |
+
<a class="navbar-brand" href="#">AQI Forecaster</a>
|
92 |
+
<button class="navbar-toggler" type="button" data-bs-toggle="collapse" data-bs-target="#navbarNav"
|
93 |
+
aria-controls="navbarNav" aria-expanded="false" aria-label="Toggle navigation">
|
94 |
+
<span class="navbar-toggler-icon"></span>
|
95 |
+
</button>
|
96 |
+
<div class="collapse navbar-collapse" id="navbarNav">
|
97 |
+
<form class="d-flex ms-auto form-container" method="POST" action="{{ url_for('forecast') }}">
|
98 |
+
<input class="form-control me-2" type="text" name="latitude" placeholder="Enter Latitude" required>
|
99 |
+
<input class="form-control me-2" type="text" name="longitude" placeholder="Enter Longitude"
|
100 |
+
required>
|
101 |
+
<button class="btn btn-outline-success" type="submit">Update Map</button>
|
102 |
+
</form>
|
103 |
+
</div>
|
104 |
+
</div>
|
105 |
+
</nav>
|
106 |
+
|
107 |
+
<div class="content">
|
108 |
+
<div class="map-container">
|
109 |
+
<!-- AQI Forecast Map -->
|
110 |
+
<div>
|
111 |
+
<!-- Render the map -->
|
112 |
+
{{ map_html|safe }}
|
113 |
+
</div>
|
114 |
+
</div>
|
115 |
+
|
116 |
+
<!-- AQI Legend -->
|
117 |
+
<div id="legend">
|
118 |
+
<b>AQI Color Legend</b>
|
119 |
+
<ul class="legend-list" style="list-style: none; padding: 0; margin: 5px;">
|
120 |
+
<li><i style="background:green;"></i> Good (0-50)</li>
|
121 |
+
<li><i style="background:yellow;"></i> Moderate (51-100)</li>
|
122 |
+
<li><i style="background:orange;"></i> Unhealthy for Sensitive Groups (101-150)</li>
|
123 |
+
<li><i style="background:red;"></i> Unhealthy (151-200)</li>
|
124 |
+
<li><i style="background:purple;"></i> Very Unhealthy (201-300)</li>
|
125 |
+
<li><i style="background:maroon;"></i> Hazardous (301+)</li>
|
126 |
+
</ul>
|
127 |
+
</div>
|
128 |
+
</div>
|
129 |
+
|
130 |
+
<!-- Add Bootstrap 5 JS -->
|
131 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap/5.3.0/js/bootstrap.bundle.min.js"></script>
|
132 |
+
</body>
|
133 |
+
|
134 |
+
</html>
|
templates/multiple_wind_speed_map.html
ADDED
@@ -0,0 +1,101 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!DOCTYPE html>
|
2 |
+
<html>
|
3 |
+
<head>
|
4 |
+
|
5 |
+
<meta http-equiv="content-type" content="text/html; charset=UTF-8" />
|
6 |
+
|
7 |
+
<script>
|
8 |
+
L_NO_TOUCH = false;
|
9 |
+
L_DISABLE_3D = false;
|
10 |
+
</script>
|
11 |
+
|
12 |
+
<style>html, body {width: 100%;height: 100%;margin: 0;padding: 0;}</style>
|
13 |
+
<style>#map {position:absolute;top:0;bottom:0;right:0;left:0;}</style>
|
14 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.js"></script>
|
15 |
+
<script src="https://code.jquery.com/jquery-3.7.1.min.js"></script>
|
16 |
+
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js"></script>
|
17 |
+
<script src="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.js"></script>
|
18 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/leaflet.css"/>
|
19 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css"/>
|
20 |
+
<link rel="stylesheet" href="https://netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap-glyphicons.css"/>
|
21 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/[email protected]/css/all.min.css"/>
|
22 |
+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Leaflet.awesome-markers/2.0.2/leaflet.awesome-markers.css"/>
|
23 |
+
<link rel="stylesheet" href="https://cdn.jsdelivr.net/gh/python-visualization/folium/folium/templates/leaflet.awesome.rotate.min.css"/>
|
24 |
+
|
25 |
+
<meta name="viewport" content="width=device-width,
|
26 |
+
initial-scale=1.0, maximum-scale=1.0, user-scalable=no" />
|
27 |
+
<style>
|
28 |
+
#map_c61b342e70deea09a0734350221e1318 {
|
29 |
+
position: relative;
|
30 |
+
width: 100.0%;
|
31 |
+
height: 100.0%;
|
32 |
+
left: 0.0%;
|
33 |
+
top: 0.0%;
|
34 |
+
}
|
35 |
+
.leaflet-container { font-size: 1rem; }
|
36 |
+
</style>
|
37 |
+
|
38 |
+
</head>
|
39 |
+
<body>
|
40 |
+
|
41 |
+
|
42 |
+
<div class="folium-map" id="map_c61b342e70deea09a0734350221e1318" ></div>
|
43 |
+
|
44 |
+
</body>
|
45 |
+
<script>
|
46 |
+
|
47 |
+
|
48 |
+
var map_c61b342e70deea09a0734350221e1318 = L.map(
|
49 |
+
"map_c61b342e70deea09a0734350221e1318",
|
50 |
+
{
|
51 |
+
center: [21.174364030859888, 73.06735784024302],
|
52 |
+
crs: L.CRS.EPSG3857,
|
53 |
+
zoom: 10,
|
54 |
+
zoomControl: true,
|
55 |
+
preferCanvas: false,
|
56 |
+
}
|
57 |
+
);
|
58 |
+
|
59 |
+
|
60 |
+
|
61 |
+
|
62 |
+
|
63 |
+
var tile_layer_9a692e000cfa98eab120be7ef771ef0c = L.tileLayer(
|
64 |
+
"https://tile.openstreetmap.org/{z}/{x}/{y}.png",
|
65 |
+
{"attribution": "\u0026copy; \u003ca href=\"https://www.openstreetmap.org/copyright\"\u003eOpenStreetMap\u003c/a\u003e contributors", "detectRetina": false, "maxNativeZoom": 19, "maxZoom": 19, "minZoom": 0, "noWrap": false, "opacity": 1, "subdomains": "abc", "tms": false}
|
66 |
+
);
|
67 |
+
|
68 |
+
|
69 |
+
tile_layer_9a692e000cfa98eab120be7ef771ef0c.addTo(map_c61b342e70deea09a0734350221e1318);
|
70 |
+
|
71 |
+
|
72 |
+
var marker_325f7aaaf407ede9431ef3e94b478da8 = L.marker(
|
73 |
+
[21.174364030859888, 73.06735784024302],
|
74 |
+
{}
|
75 |
+
).addTo(map_c61b342e70deea09a0734350221e1318);
|
76 |
+
|
77 |
+
|
78 |
+
var popup_ba5ff7c693e7132352eab20dde701d9f = L.popup({"maxWidth": 300});
|
79 |
+
|
80 |
+
|
81 |
+
|
82 |
+
var html_504edff652c09db11dfbb4099d86edf5 = $(`<div id="html_504edff652c09db11dfbb4099d86edf5" style="width: 100.0%; height: 100.0%;"><img src=""></div>`)[0];
|
83 |
+
popup_ba5ff7c693e7132352eab20dde701d9f.setContent(html_504edff652c09db11dfbb4099d86edf5);
|
84 |
+
|
85 |
+
|
86 |
+
|
87 |
+
marker_325f7aaaf407ede9431ef3e94b478da8.bindPopup(popup_ba5ff7c693e7132352eab20dde701d9f)
|
88 |
+
;
|
89 |
+
|
90 |
+
|
91 |
+
|
92 |
+
|
93 |
+
marker_325f7aaaf407ede9431ef3e94b478da8.bindTooltip(
|
94 |
+
`<div>
|
95 |
+
Bardoli
|
96 |
+
</div>`,
|
97 |
+
{"sticky": true}
|
98 |
+
);
|
99 |
+
|
100 |
+
</script>
|
101 |
+
</html>
|
templates/results.html
ADDED
@@ -0,0 +1,45 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<!-- templates/results.html -->
|
2 |
+
<!DOCTYPE html>
|
3 |
+
<html lang="en">
|
4 |
+
<head>
|
5 |
+
<meta charset="UTF-8">
|
6 |
+
<title>Wind Speed Prediction Results</title>
|
7 |
+
<link href="https://maxcdn.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css" rel="stylesheet">
|
8 |
+
<style>
|
9 |
+
body {
|
10 |
+
display: flex;
|
11 |
+
justify-content: center;
|
12 |
+
align-items: center;
|
13 |
+
min-height: 100vh;
|
14 |
+
background-color: #f8f9fa;
|
15 |
+
}
|
16 |
+
.container {
|
17 |
+
background-color: #ffffff;
|
18 |
+
padding: 30px;
|
19 |
+
border-radius: 8px;
|
20 |
+
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.2);
|
21 |
+
}
|
22 |
+
h1, h2 {
|
23 |
+
text-align: center;
|
24 |
+
color: #343a40;
|
25 |
+
}
|
26 |
+
p {
|
27 |
+
font-size: 1.2em;
|
28 |
+
color: #555;
|
29 |
+
}
|
30 |
+
</style>
|
31 |
+
</head>
|
32 |
+
<body>
|
33 |
+
<div class="container">
|
34 |
+
<h1>Wind Speed Prediction Results</h1>
|
35 |
+
<p><strong>Actual Wind Speed:</strong> {{ actual }} km/h</p>
|
36 |
+
<p><strong>Predicted Wind Speed:</strong> {{ predicted }} km/h</p>
|
37 |
+
<h2>Location Map</h2>
|
38 |
+
<iframe src="{{ url_for('static', filename=map_file) }}" width="100%" height="500px" style="border: none; border-radius: 8px;"></iframe>
|
39 |
+
</div>
|
40 |
+
|
41 |
+
<script src="https://code.jquery.com/jquery-3.5.1.slim.min.js"></script>
|
42 |
+
<script src="https://cdn.jsdelivr.net/npm/@popperjs/[email protected]/dist/umd/popper.min.js"></script>
|
43 |
+
<script src="https://maxcdn.bootstrapcdn.com/bootstrap/4.5.2/js/bootstrap.min.js"></script>
|
44 |
+
</body>
|
45 |
+
</html>
|