""" Tufts Jumbo Weather Forecast — Deep Learning Demo Usage: cd demo && python app.py """ import logging import math from datetime import timedelta import gradio as gr from hrrr_fetch import fetch_hrrr_input from model_utils import run_forecast, load_model, AVAILABLE_MODELS from var_mapping import JUMBO_ROW, JUMBO_COL from visualization import ( get_static_maps, plot_temperature, plot_precipitation, plot_wind_speed, plot_humidity, ) logging.basicConfig(level=logging.INFO) logger = logging.getLogger(__name__) # ── CSS ─────────────────────────────────────────────────────────────── CUSTOM_CSS = """ @import url('https://fonts.googleapis.com/css2?family=Inter:wght@300;400;500;600;700&display=swap'); :root { --font: -apple-system, BlinkMacSystemFont, "SF Pro Display", "SF Pro Text", Inter, "Helvetica Neue", Arial, sans-serif; --bg: #F2F2F7; --card: #FFFFFF; --border: #E5E5EA; --text: #1D1D1F; --muted: #86868B; --accent: #0A84FF; --dark: #1C1C1E; } * { font-family: var(--font) !important; } .gradio-container { max-width: 1320px !important; margin: 0 auto !important; background: var(--bg) !important; padding-bottom: 24px !important; } /* ── Top bar ── */ .top-bar { background: linear-gradient(135deg, #1C1C1E 0%, #2C2C2E 100%); border-radius: 16px; padding: 28px 36px; margin-bottom: 16px; display: flex; justify-content: space-between; align-items: center; } .top-bar .title { font-size: 24px; font-weight: 700; color: #F5F5F7; letter-spacing: -0.3px; } .top-bar .subtitle { font-size: 13px; color: #98989D; margin-top: 2px; } .top-bar .location { text-align: right; font-size: 13px; color: #98989D; line-height: 1.6; } .top-bar .location b { color: #F5F5F7; font-weight: 600; } /* ── Hero card ── */ .hero-card { background: var(--card); border-radius: 16px; border: 1px solid var(--border); box-shadow: 0 2px 8px rgba(0,0,0,0.04); padding: 32px 36px 28px; margin-bottom: 16px; } .hero-main { display: flex; align-items: baseline; gap: 20px; margin-bottom: 4px; } .hero-temp { font-size: 64px; font-weight: 300; color: var(--text); letter-spacing: -2px; line-height: 1; } .hero-temp-unit { font-size: 28px; font-weight: 400; color: var(--muted); margin-left: 2px; } .hero-status { font-size: 20px; font-weight: 500; color: var(--text); padding-left: 8px; border-left: 3px solid var(--accent); } .hero-metrics { display: flex; gap: 12px; margin: 20px 0 18px; } .metric-tile { flex: 1; background: var(--bg); border-radius: 12px; padding: 14px 16px; text-align: center; } .metric-value { font-size: 22px; font-weight: 600; color: var(--text); line-height: 1.2; } .metric-label { font-size: 12px; font-weight: 500; color: var(--muted); text-transform: uppercase; letter-spacing: 0.5px; margin-top: 4px; } .hero-meta { font-size: 13px; color: var(--muted); line-height: 1.6; } .hero-meta code { background: var(--bg); padding: 2px 6px; border-radius: 4px; font-size: 12px; } .hero-placeholder { text-align: center; padding: 36px 0; color: var(--muted); font-size: 16px; font-weight: 500; } /* ── Map section ── */ .maps-heading { font-size: 11px; font-weight: 600; text-transform: uppercase; letter-spacing: 0.8px; color: var(--muted); margin: 8px 0 8px 4px; } .map-cell { background: var(--card) !important; border-radius: 14px !important; border: 1px solid var(--border) !important; box-shadow: 0 1px 4px rgba(0,0,0,0.04) !important; overflow: hidden !important; min-height: 380px !important; } /* ── Controls inside hero ── */ .controls-row { display: flex; align-items: end; gap: 10px; margin-top: 18px; padding-top: 16px; border-top: 1px solid var(--border); } /* ── Status ── */ .status-text p, .status-text em { font-size: 12px !important; color: var(--muted) !important; } /* ── About ── */ .about-section { font-size: 13px !important; color: #6E6E73 !important; line-height: 1.65 !important; } /* ── Button ── */ button.primary { background: var(--accent) !important; border: none !important; border-radius: 10px !important; font-weight: 600 !important; font-size: 15px !important; padding: 10px 28px !important; } button.primary:hover { background: #0A74E0 !important; } /* ── Current / delta annotations ── */ .metric-current { font-size: 11px; font-weight: 500; color: var(--muted); margin-top: 6px; line-height: 1.4; } .delta-up { color: #FF3B30; font-weight: 600; } .delta-down { color: #0A84FF; font-weight: 600; } .delta-neutral { color: #86868B; font-weight: 600; } .hero-current-note { font-size: 13px; color: var(--muted); margin-top: 2px; margin-bottom: 0; } """ # ── Helpers ──────────────────────────────────────────────────────────── model_choices = [ f"{v['display_name']} ({v['params']})" for v in AVAILABLE_MODELS.values() ] model_keys = list(AVAILABLE_MODELS.keys()) def _resolve_model(display: str) -> str: return model_keys[model_choices.index(display)] def _extract_current(input_array) -> dict: """Extract current observed values at Jumbo from the input array.""" tmp_k = float(input_array[JUMBO_ROW, JUMBO_COL, 0]) rh = float(input_array[JUMBO_ROW, JUMBO_COL, 1]) u = float(input_array[JUMBO_ROW, JUMBO_COL, 2]) v = float(input_array[JUMBO_ROW, JUMBO_COL, 3]) gust = float(input_array[JUMBO_ROW, JUMBO_COL, 4]) apcp = float(input_array[JUMBO_ROW, JUMBO_COL, 6]) tmp_c = tmp_k - 273.15 return { "temperature_c": tmp_c, "temperature_f": tmp_c * 9 / 5 + 32, "humidity_pct": max(0.0, min(100.0, rh)), "wind_speed_ms": math.sqrt(u**2 + v**2), "gust_ms": max(gust, 0.0), "precipitation_mm": max(apcp, 0.0), } def _delta_html(now_val: float, fcst_val: float, fmt: str = ".1f", unit: str = "") -> str: """Render 'Now X · +/-delta' with colored arrow.""" diff = fcst_val - now_val if abs(diff) < 0.05: arrow, cls = "", "delta-neutral" elif diff > 0: arrow, cls = " ↑", "delta-up" else: arrow, cls = " ↓", "delta-down" sign = "+" if diff >= 0 else "" return ( f'Now {now_val:{fmt}}{unit} · ' f'{sign}{diff:{fmt}}{unit}{arrow}' ) def _hero_placeholder() -> str: return ( '
' f'Now {cur["temperature_c"]:.1f}°C · ' f'{sign}{temp_diff:.1f}°C in 24h' f'
' ) return ( '