未分類

# ==========================================
# 気象GPV 局地分析ツール ビューワー (App B)
# VERSION INFO: 121.0 (GSM 132h以降の全要素6時間間隔・降水差分バグ完全対応版)
# ==========================================
import sys, os, glob, re, gc, time, logging
import numpy as np
import textwrap
from datetime import datetime, timedelta
import scipy.ndimage as ndimage

from PyQt6.QtWidgets import (QApplication, QMainWindow, QWidget, QVBoxLayout, 
                             QHBoxLayout, QSlider, QPushButton, QLabel, 
                             QTabWidget, QListWidget, QTextEdit, QComboBox, QFileDialog, QDialog, 
                             QMessageBox, QLineEdit, QTableWidget, QTableWidgetItem, QHeaderView)
from PyQt6.QtCore import Qt, QTimer, QSettings
from PyQt6.QtGui import QColor, QFont, QCloseEvent

os.environ['QT_API'] = 'pyqt6'
import matplotlib
matplotlib.use('qtagg')
import matplotlib.colors as mcolors
import matplotlib as mpl
import matplotlib.path as mpath
import matplotlib.pyplot as plt

mpl.rcParams['font.family'] = 'MS Gothic'
from matplotlib.backends.backend_qtagg import FigureCanvasQTAgg as FigureCanvas
from matplotlib.figure import Figure
import cartopy.crs as ccrs
import cartopy.feature as cfeature
import cartopy.io.shapereader as shpreader

STATIONS = [
    ("札幌", 43.060, 141.328), ("函館", 41.817, 140.753), ("室蘭", 42.315, 140.975),
    ("帯広", 42.922, 143.212), ("釧路", 42.985, 144.380), ("網走", 44.018, 144.279),
    ("稚内", 45.415, 141.680), ("旭川", 43.760, 142.360)
]

CACHED_SHAPES = {'other_cities': None, 'sapporo_districts': None, 'sapporo_city': None, 'kamikawa': None}
SHAPEFILE_FOUND = False

def search_shapefile():
    paths = glob.glob('**/N03*.shp', recursive=True)
    return paths[0] if paths else None

def get_simplified_geometries():
    global CACHED_SHAPES, SHAPEFILE_FOUND
    if CACHED_SHAPES['other_cities'] is not None: return CACHED_SHAPES['other_cities'], CACHED_SHAPES['sapporo_districts'], CACHED_SHAPES['sapporo_city'], CACHED_SHAPES['kamikawa']
    shp_path = search_shapefile()
    if not shp_path or not os.path.exists(shp_path): return [], [], [], []
    SHAPEFILE_FOUND = True
    other_cities_g = []; sapporo_districts_g = {}; sapporo_city_g = []; kamikawa_g = []
    sapporo_dist_names = ['中央区', '北区', '東区', '白石区', '厚別区', '豊平区', '清田区', '南区', '西区', '手稲区']
    for d_name in sapporo_dist_names: sapporo_districts_g[d_name] = []
    try:
        reader = shpreader.Reader(shp_path, encoding='cp932')
        for r in reader.records():
            geom = r.geometry
            try: geom = geom.simplify(0.002, preserve_topology=False)
            except: pass
            pref_name = str(r.attributes.get('N03_001', '')); subpref_name = str(r.attributes.get('N03_002', ''))
            city_name = str(r.attributes.get('N03_004', '')); cit_name_spec = str(r.attributes.get('N03_003', ''))
            is_sap_dist = False
            if '札幌市' in cit_name_spec:
                for d_name in sapporo_dist_names:
                    if d_name == city_name: is_sap_dist = True; sapporo_districts_g[d_name].append(geom); sapporo_city_g.append(geom); break
            if '上川' in subpref_name: kamikawa_g.append(geom)
            if not is_sap_dist and '北海道' in pref_name:
                try: 
                    if geom.area > 0.0005: other_cities_g.append(geom)
                except: pass
        CACHED_SHAPES['other_cities'] = other_cities_g; CACHED_SHAPES['sapporo_districts'] = sapporo_districts_g; 
        CACHED_SHAPES['sapporo_city'] = sapporo_city_g; CACHED_SHAPES['kamikawa'] = kamikawa_g
        return other_cities_g, sapporo_districts_g, sapporo_city_g, kamikawa_g
    except Exception as e:
        logging.error(f"Shapefile processing error: {e}")
        return [], [], [], []

def calc_t_td(t, rh):
    rh_s = np.clip(rh, 1.0, 100.0)
    e = 6.112 * np.exp((17.67 * t) / (t + 243.5)) * (rh_s / 100.0)
    e = np.clip(e, 0.001, 1000)
    td = (243.5 * np.log(e / 6.112)) / (17.67 - np.log(e / 6.112))
    return t - td

def rh_from_t_td(t, t_td):
    td = t - t_td
    e = 6.112 * np.exp((17.67 * td) / (td + 243.5))
    es = 6.112 * np.exp((17.67 * t) / (t + 243.5))
    rh = (e / es) * 100.0
    return np.clip(rh, 0.0, 100.0)

def calc_ept(t, rh, p):
    t_k = t + 273.15
    rh_s = np.clip(rh, 1.0, 100.0)
    e = 6.112 * np.exp((17.67 * t) / (t + 243.5)) * (rh_s / 100.0)
    e = np.clip(e, 0.001, p - 0.1)
    q = 0.622 * e / (p - e)
    tlcl = 2840.0 / (3.5 * np.log(t_k) - np.log(e) - 4.805) + 55.0
    theta = t_k * (1000.0 / p) ** 0.2854
    ept = theta * np.exp(((3376.0 / tlcl) - 2.54) * q * (1.0 + 0.81 * q))
    return ept

def get_t_from_ept(target_ept, p):
    t_min, t_max = -80.0, 50.0
    for _ in range(15):
        t_mid = (t_min + t_max) / 2.0
        ept_mid = calc_ept(t_mid, 100.0, p)
        if ept_mid < target_ept: t_min = t_mid
        else: t_max = t_mid
    return (t_min + t_max) / 2.0

class SingleImageWindow(QDialog):
    def __init__(self, parent, title, model_name, stat_name, init_z_str, canvas):
        super().__init__(parent)
        self.model_name = model_name; self.stat_name = stat_name; self.init_z_str = init_z_str
        self.setWindowTitle(title)
        self.resize(1100, 950)
        self.setStyleSheet("QDialog { background-color: #0A192F; color: #E0E0E0; }")
        
        layout = QVBoxLayout(self)
        layout.setContentsMargins(10, 10, 10, 10)
        layout.addWidget(canvas, stretch=1)
        
        btn_layout = QHBoxLayout()
        btn_layout.setContentsMargins(10, 10, 10, 10)
        
        save_btn = QPushButton("📷 画像として保存")
        save_btn.setStyleSheet("background-color: #27AE60; color: white; font-size: 14pt; font-weight: bold; padding: 12px; border-radius: 5px;")
        save_btn.clicked.connect(self.save_image)
        
        close_btn = QPushButton("閉じる")
        close_btn.setStyleSheet("background-color: #C0392B; color: white; font-size: 14pt; font-weight: bold; padding: 12px; border-radius: 5px;")
        close_btn.clicked.connect(self.accept)
        
        btn_layout.addStretch()
        btn_layout.addWidget(save_btn)
        btn_layout.addWidget(close_btn)
        btn_layout.addStretch()
        layout.addLayout(btn_layout)

    def save_image(self):
        desktop = os.path.expanduser("~/Desktop")
        filename = f"{self.model_name}_{self.stat_name}_{self.init_z_str}.png"
        path, _ = QFileDialog.getSaveFileName(self, "画像を保存", os.path.join(desktop, filename), "Images (*.png)")
        if path:
            canvas_widget = self.layout().itemAt(0).widget()
            pixmap = canvas_widget.grab() if hasattr(canvas_widget, 'grab') else None
            if pixmap: pixmap.save(path)
            else: self.layout().itemAt(0).widget().figure.savefig(path, dpi=150, bbox_inches='tight', facecolor='white')
            QMessageBox.information(self, "保存完了", f"保存しました:\n{os.path.basename(path)}")

class AdminPanelDialog(QDialog):
    def __init__(self, parent_app):
        super().__init__(parent_app)
        self.setWindowTitle("⚙️ コントロールパネル")
        self.setFixedSize(500, 300)
        self.setStyleSheet("""
            QDialog { background-color: #0A192F; color: #E0E0E0;}
            QLabel { color: #E0E0E0; font-size: 11pt; }
            QPushButton { background-color: #1D3557; color: white; padding: 12px; border-radius: 5px; font-weight: bold; font-size: 12pt; margin: 5px; border: 1px solid #457B9D;}
            QPushButton:hover { background-color: #457B9D; }
        """)
        layout = QVBoxLayout(self)
        self.parent_app = parent_app

        lbl = QLabel("【システム設定メニュー】")
        lbl.setStyleSheet("font-size: 14pt; font-weight: bold; color: #64FFDA;")
        layout.addWidget(lbl)

        self.dir_lbl = QLabel(f"現在の読込先:\n{self.parent_app.cache_dir}")
        self.dir_lbl.setWordWrap(True)
        self.dir_lbl.setStyleSheet("color: #8892B0; font-size: 10pt;")
        layout.addWidget(self.dir_lbl)

        dir_btn = QPushButton("📁 読込先フォルダを変更")
        dir_btn.clicked.connect(self.change_dir)
        layout.addWidget(dir_btn)

        sync_btn = QPushButton("🔄 画面手動強制同期")
        sync_btn.clicked.connect(self.manual_sync)
        layout.addWidget(sync_btn)

        close_btn = QPushButton("閉じる")
        close_btn.setStyleSheet("background-color: #C0392B;")
        close_btn.clicked.connect(self.accept)
        layout.addWidget(close_btn)

    def change_dir(self):
        directory = QFileDialog.getExistingDirectory(self, "GPVデータ(npz)が保存されるフォルダを選択", self.parent_app.cache_dir)
        if directory:
            self.parent_app.cache_dir = directory
            self.parent_app.settings.setValue('cache_dir', directory)
            self.parent_app.setup_logging()
            self.dir_lbl.setText(f"現在の読込先:\n{directory}")
            QMessageBox.information(self, "設定完了", "読込先フォルダを更新しました")

    def manual_sync(self):
        self.parent_app.scan_cache_dir(force_draw=True)
        QMessageBox.information(self, "同期完了", "画面の強制同期を実行しました")

class SplashWindow(QWidget):
    def __init__(self):
        super().__init__()
        self.setWindowFlags(Qt.WindowType.FramelessWindowHint | Qt.WindowType.WindowStaysOnTopHint)
        self.setFixedSize(480, 250)
        self.setStyleSheet("background-color: #0A192F; border: 3px solid #64FFDA; border-radius: 12px;")
        layout = QVBoxLayout(self)
        
        title_lbl = QLabel("気象GPV 局地分析ツール\n(Ver 121.0)")
        title_lbl.setAlignment(Qt.AlignmentFlag.AlignCenter)
        title_lbl.setStyleSheet("font-size: 18pt; font-weight: bold; color: #E0E0E0; border: none; margin-top: 15px;")
        layout.addWidget(title_lbl)
        
        msg_lbl = QLabel("最新の解析データをロードしています...\n(約5秒で自動起動します)")
        msg_lbl.setAlignment(Qt.AlignmentFlag.AlignCenter)
        msg_lbl.setStyleSheet("font-size: 14pt; color: #64FFDA; border: none;")
        layout.addWidget(msg_lbl)

class WeatherApp(QMainWindow):
    def __init__(self):
        super().__init__()
        self.settings = QSettings('WeatherApp', 'Viewer')
        self.cache_dir = self.settings.value('cache_dir', os.path.join(os.getcwd(), "gpv_cache_npz"))
        self.setup_logging()

        self.setWindowTitle("気象GPV 局地分析ツール (Ver 121.0 - GSM132h以降 完全対応版)")
        self.resize(1850, 1050)
        self.model_init_times = {"MSM": None, "GSM_JP": None, "MSM_GUID": None, "GSM_GUID": None, "ANAL": None}
        self.tab_ui = {}
        
        self.monitor_timer = QTimer(self); self.monitor_timer.timeout.connect(self.scan_cache_dir)
        self.anim_timer = QTimer(self); self.anim_timer.timeout.connect(self.anim_step)
        self.current_anim_model = None
        self.setStyleSheet("""
            * { font-size: 13pt; font-family: "MS Gothic", sans-serif; color: #E0E0E0; }
            QMainWindow, QDialog, QMessageBox { background-color: #0A192F; color: #E0E0E0; }
            QMessageBox QLabel { color: #E0E0E0; }
            QMessageBox QPushButton { background-color: #1D3557; color: white; padding: 5px 15px; border-radius: 3px; font-weight: bold; }
            QMessageBox QPushButton:hover { background-color: #457B9D; }
            QTabWidget::pane { border: 1px solid #444; background: #112240; }
            QTabBar::tab { background: #1B2B4A; color: #8892B0; padding: 12px 25px; border: 1px solid #333; border-bottom: none; }
            QTabBar::tab:selected { background: #112240; color: #64FFDA; font-weight: bold; border-top: 3px solid #64FFDA; }
            QSlider::groove:vertical { border: 1px solid #444; background: #0A192F; width: 14px; border-radius: 7px; }
            QSlider::handle:vertical { background: #64FFDA; border: 2px solid #000; height: 32px; margin: 0 -10px; border-radius: 16px; }
            QPushButton { background-color: #1D3557; border: 1px solid #457B9D; padding: 8px; border-radius: 4px; color: #F1FAEE; }
            QPushButton:hover { background-color: #457B9D; }
            QPushButton::checked { background-color: #E63946; color: white; font-weight: bold; }
            QListWidget { background: #112240; border: 1px solid #444; color: #E0E0E0; }
            QListWidget::item { padding: 3px 5px; margin: 1px 5px; background: #1D3557; border: 1px solid #333; border-radius: 6px; }
            QListWidget::item:selected { background: #E63946; color: white; font-weight: bold; }
            QListWidget::item:hover:!selected { background: #457B9D; }
            QComboBox { background: #112240; border: 1px solid #8892B0; color: white; padding: 5px; }
            QComboBox QAbstractItemView { background: #112240; color: white; }
            QLabel { background: transparent; }
            QTextEdit { background-color: #112240; color: #64FFDA; border: 1px solid #444; }
            .NavButton { background-color: #2980B9; color: white; font-weight: bold; padding: 5px; border-radius: 5px; }
            .PlayButton { background-color: #27AE60; color: white; font-weight: bold; padding: 5px 15px; border-radius: 5px; }
            .StopButton { background-color: #C0392B; color: white; font-weight: bold; padding: 5px 15px; border-radius: 5px; }
            .SyncButton { background-color: #F39C12; color: white; font-weight: bold; border: 2px solid #D68910; border-radius: 5px; padding: 5px 15px;}
            .SaveButton { background-color: #8E44AD; color: white; font-weight: bold; border-radius: 5px; padding: 4px 10px; font-size: 10pt;}
            .MetaButton { background-color: #16A085; color: white; font-weight: bold; border: 2px solid #0E6655; border-radius: 5px; padding: 5px 15px; }
            .MetaButton:checked { background-color: #0E6655; border: 2px solid #1ABC9C; }
            .AdminButton { background-color: #34495E; color: white; font-weight: bold; border-radius: 5px; padding: 4px 10px; font-size: 10pt;}
        """)

        main_widget = QWidget(); self.setCentralWidget(main_widget); main_layout = QVBoxLayout(main_widget)
        
        self.tabs = QTabWidget()
        corner_widget = QWidget()
        corner_layout = QHBoxLayout(corner_widget)
        corner_layout.setContentsMargins(0, 0, 5, 0)
        
        self.status_banner = QLabel("待機中...")
        self.status_banner.setStyleSheet("color: white; font-weight: bold; background-color: #E63946; padding: 4px 10px; border-radius: 4px; font-size: 10pt;")

        self.save_img_btn = QPushButton("📷 画面保存")
        self.save_img_btn.setProperty("class", "SaveButton")
        self.save_img_btn.clicked.connect(self.save_current_image)
        
        self.admin_btn = QPushButton("⚙️ 設定")
        self.admin_btn.setProperty("class", "AdminButton")
        self.admin_btn.clicked.connect(self.open_admin_panel)

        self.exit_app_btn = QPushButton("🛑 完全停止")
        self.exit_app_btn.setProperty("class", "StopButton")
        self.exit_app_btn.clicked.connect(self.close)
        
        corner_layout.addWidget(self.status_banner)
        corner_layout.addWidget(self.save_img_btn)
        corner_layout.addWidget(self.admin_btn)
        corner_layout.addWidget(self.exit_app_btn)
        
        self.tabs.setCornerWidget(corner_widget, Qt.Corner.TopRightCorner)
        
        self.setup_tab("MSM", "MSM GPV", is_guidance=False, is_anal=False)
        self.setup_tab("GSM_JP", "GSM 日本域 GPV", is_guidance=False, is_anal=False)
        self.setup_tab("ANAL", "毎時大気解析", is_guidance=False, is_anal=True)
        self.setup_tab("MSM_GUID", "MSM ガイダンス", is_guidance=True, is_anal=False)
        self.setup_tab("GSM_GUID", "GSM ガイダンス", is_guidance=True, is_anal=False)
        self.setup_history_tab()
        
        main_layout.addWidget(self.tabs)
        get_simplified_geometries()
        
        self.splash = SplashWindow()
        self.splash.show()
        QTimer.singleShot(5000, self.finish_startup_loading)

    def closeEvent(self, event: QCloseEvent):
        reply = QMessageBox.question(self, '終了の確認', 'システムを完全に停止してよろしいですか?',
                                     QMessageBox.StandardButton.Yes | QMessageBox.StandardButton.No, QMessageBox.StandardButton.No)
        if reply == QMessageBox.StandardButton.Yes:
            event.accept()
        else:
            event.ignore()

    def setup_logging(self):
        self.log_file = os.path.join(self.cache_dir, "system_log.txt")
        logging.basicConfig(filename=self.log_file, level=logging.INFO, 
                            format='%(asctime)s - %(levelname)s - %(message)s', encoding='utf-8')

    def open_admin_panel(self):
        panel = AdminPanelDialog(self)
        panel.exec()

    def finish_startup_loading(self):
        self.splash.close()
        self.scan_cache_dir(quiet=True)
        self.monitor_timer.start(5000)
        
        for i in range(self.tabs.count()):
            if "MSM" in self.tabs.tabText(i) and "ガイダンス" not in self.tabs.tabText(i):
                self.tabs.setCurrentIndex(i); break
        if "MSM" in self.tab_ui:
            items = self.tab_ui["MSM"]['list'].findItems("地上気圧", Qt.MatchFlag.MatchExactly)
            if items: self.tab_ui["MSM"]['list'].setCurrentItem(items[0])
        self.show()

    def save_current_image(self):
        current_tab = self.tabs.tabText(self.tabs.currentIndex())
        if "MSM ガイダンス" in current_tab: model_name = "MSM_GUID"
        elif "GSM ガイダンス" in current_tab: model_name = "GSM_GUID"
        elif "毎時大気" in current_tab: model_name = "ANAL"
        elif "MSM" in current_tab: model_name = "MSM"
        elif "日本域" in current_tab: model_name = "GSM_JP"
        else: return
        
        if model_name not in self.tab_ui or self.tab_ui[model_name]['ax'] is None: return
        ui = self.tab_ui[model_name]
        val = ui['slider'].value(); elem = ui['list'].currentItem().text()
        it_str = self.model_init_times.get(model_name)
        if not it_str: return
            
        dt_utc = datetime.strptime(it_str, "%Y%m%d%H%M%S")
        init_z = dt_utc.strftime("%m%d%H") + "Z"
        filename = f"{model_name}_{elem}_{init_z}_FT{val:02d}.png"
        desktop = os.path.expanduser("~/Desktop")
        
        path, _ = QFileDialog.getSaveFileName(self, "画像を保存", os.path.join(desktop, filename), "Images (*.png *.jpg)")
        if path:
            ui['fig'].savefig(path, dpi=150, bbox_inches='tight', facecolor='white')
            self.status_banner.setText(f"保存完了: {os.path.basename(path)}")

    def init_map(self, fig, area_idx, model):
        fig.clear()
        ax = fig.add_subplot(1, 1, 1, projection=ccrs.PlateCarree())
        ax.set_extent(self.get_extent(area_idx, model), crs=ccrs.PlateCarree())
        self.draw_base_map(ax, area_idx)
        fig.subplots_adjust(left=0.05, right=0.9, top=0.92, bottom=0.05)
        return ax

    def get_extent(self, area_index, model_name):
        if area_index == 0: return [120, 150, 27.5, 48] if "MSM" in model_name or "ANAL" in model_name else [120, 150, 20, 50]
        elif area_index == 1: return [138.5, 146.5, 40.5, 46.5]  
        elif area_index == 2: return [141.6, 143.6, 42.8, 45.0]  
        else: return [141.15, 141.55, 42.92, 43.20]              

    def crop_data(self, lon, lat, data, extent):
        if lon is None or lat is None or data is None:
            return np.array([]), np.array([]), np.array([])

        buffer = 1.0; lon_min, lon_max = extent[0] - buffer, extent[1] + buffer; lat_min, lat_max = extent[2] - buffer, extent[3] + buffer
        if lon.ndim == 1:
            m_lon = (lon >= lon_min) & (lon <= lon_max); m_lat = (lat >= lat_min) & (lat <= lat_max)
            if not np.any(m_lon) or not np.any(m_lat): return np.array([]), np.array([]), np.array([])
            return lon[m_lon], lat[m_lat], data[np.ix_(m_lat, m_lon)]
        else:
            mask = (lon >= lon_min) & (lon <= lon_max) & (lat >= lat_min) & (lat <= lat_max)
            j, i = np.where(mask)
            if len(j) == 0: return np.array([]), np.array([]), np.array([])
            return lon[j.min():j.max()+1, i.min():i.max()+1], lat[j.min():j.max()+1, i.min():i.max()+1], data[j.min():j.max()+1, i.min():i.max()+1]

    def draw_base_map(self, ax, area_index):
        SEA_COLOR = '#A4C2E4'; LAND_COLOR = '#F5F5F0'; COASTLINE_COLOR = '#333333'
        ax.set_facecolor(SEA_COLOR); ax.add_feature(cfeature.LAND.with_scale('10m'), facecolor=LAND_COLOR, zorder=0)
        ax.add_feature(cfeature.OCEAN.with_scale('10m'), facecolor=SEA_COLOR, zorder=0)
        ax.add_feature(cfeature.LAKES.with_scale('10m'), edgecolor='#4682B4', facecolor=SEA_COLOR, zorder=1)
        ax.add_feature(cfeature.RIVERS.with_scale('10m'), linewidth=0.8, edgecolor='#4682B4', zorder=1)
        ax.add_feature(cfeature.COASTLINE.with_scale('10m'), linewidth=0.6, edgecolor=COASTLINE_COLOR, zorder=2)

        other_cities_g, sapporo_districts_g, sapporo_city_g, kamikawa_g = get_simplified_geometries()
        if area_index >= 1 and other_cities_g: ax.add_geometries(other_cities_g, ccrs.PlateCarree(), edgecolor='#888888', facecolor='none', linewidth=0.6, linestyle=':', zorder=3)
        
        if area_index == 3 and sapporo_districts_g:
            for d_name, geoms in sapporo_districts_g.items():
                if geoms: ax.add_geometries(geoms, ccrs.PlateCarree(), edgecolor='#FF0000', facecolor='none', linewidth=1.5, zorder=10)
        
        if sapporo_city_g and area_index == 3: ax.add_geometries(sapporo_city_g, ccrs.PlateCarree(), edgecolor='#FF0000', facecolor='#FF8C00', alpha=0.3, linewidth=2.0, linestyle='-', zorder=4)
        elif sapporo_city_g and area_index >= 1: ax.add_geometries(sapporo_city_g, ccrs.PlateCarree(), edgecolor='#FF0000', facecolor='#FF8C00', alpha=0.2, linewidth=1.0, linestyle='-', zorder=4)
        
        if area_index >= 1 and kamikawa_g: ax.add_geometries(kamikawa_g, ccrs.PlateCarree(), edgecolor='#FF0000', facecolor='#FF8C00', alpha=0.25 if area_index == 2 else 0.15, linewidth=1.5, linestyle='-', zorder=5)
        
        pref_lines = cfeature.NaturalEarthFeature('cultural', 'admin_1_states_provinces_lines', '10m', edgecolor='#444444', facecolor='none')
        ax.add_feature(pref_lines, linewidth=0.8, linestyle='-.', zorder=7)

        gl = ax.gridlines(crs=ccrs.PlateCarree(), draw_labels=True, linewidth=0.5, color='gray', alpha=0.5, linestyle='--', zorder=6)
        gl.top_labels = False; gl.right_labels = False 
        gl.xlabel_style = {'size': 11, 'color': 'black', 'fontname': 'MS Gothic'}; gl.ylabel_style = {'size': 11, 'color': 'black', 'fontname': 'MS Gothic'}
        
        if area_index == 3: gl.xlocator = mpl.ticker.MultipleLocator(0.05); gl.ylocator = mpl.ticker.MultipleLocator(0.05) 
        elif area_index == 2: gl.xlocator = mpl.ticker.MultipleLocator(0.5); gl.ylocator = mpl.ticker.MultipleLocator(0.5) 
        elif area_index == 1: gl.xlocator = mpl.ticker.MultipleLocator(1.0); gl.ylocator = mpl.ticker.MultipleLocator(1.0) 
        else: gl.xlocator = mpl.ticker.MultipleLocator(5.0) if area_index == 0 else mpl.ticker.MultipleLocator(1.0); gl.ylocator = mpl.ticker.MultipleLocator(5.0) if area_index == 0 else mpl.ticker.MultipleLocator(1.0)

    def get_local_info(self, elem_name, lon, lat, data, u=None, v=None, area_index=0, model_name=""):
        if data is None or lon is None or lat is None or data.size == 0: return ""
        target_lon, target_lat, loc_name = (142.368, 43.760, "旭川") if area_index == 2 else (141.35, 43.06, "札幌")
        if lon.ndim == 1: lon_2d, lat_2d = np.meshgrid(lon, lat)
        else: lon_2d, lat_2d = lon, lat
        dist = (lon_2d - target_lon)**2 + (lat_2d - target_lat)**2
        sy, sx = np.unravel_index(np.argmin(dist), dist.shape)
        if sy >= data.shape[0] or sx >= data.shape[1]: return ""
        s_val = data[sy, sx]
        if np.isnan(s_val): return ""
        
        if "降水量" in elem_name or "降水" in elem_name: info = f"[{loc_name}現況]\n降水量: {s_val:.1f} mm"
        elif "降雪量" in elem_name or "降雪" in elem_name: info = f"[{loc_name}現況]\n降雪量: {s_val:.1f} cm"
        else: info = f"[{loc_name}現況]\n" 
        
        if "降水" not in elem_name and "降雪" not in elem_name:
            if "気圧" in elem_name: info += f"気圧: {s_val:.1f} hPa"
            elif "確率" in elem_name and "発雷" not in elem_name: info += f"{elem_name}: {s_val:.0f} %"
            elif "発雷確率" in elem_name: info += f"発雷確率: {s_val:.0f} %"
            elif "天気" in elem_name: 
                wx_map = {0: "晴", 1: "曇", 2: "雨", 3: "雪", 4: "不明"}
                info += f"卓越天気: {wx_map.get(int(s_val), '不明')}"
            elif "雲量" in elem_name: info += f"{elem_name}: {s_val:.1f} %"
            elif "高度" in elem_name: info += f"{elem_name[:6]}高度: {s_val:.0f} m"
            elif "渦度" in elem_name: info += f"正渦度(10^-5/s): {s_val:.1f}"
            elif "気温" in elem_name: info += f"{elem_name[:6]}気温: {s_val:.1f} ℃"
            elif "鉛直流" in elem_name: info += f"鉛直流: {s_val:.1f} hPa/h"
            elif "湿数" in elem_name: info += f"{elem_name[:6]}湿数: {s_val:.1f} ℃"
            elif "相当温位" in elem_name: info += f"{elem_name[:6]}相当温位: {s_val:.1f} K"
            else: info += f"値: {s_val:.1f}"
        
        if u is not None and v is not None and sy < u.shape[0] and sx < u.shape[1]: info += f"\n風速: {np.hypot(u[sy, sx], v[sy, sx]):.1f} m/s"
        return info

    def on_area_btn_clicked(self, model_name, idx):
        ui = self.tab_ui[model_name]
        for i, btn in enumerate(ui['area_btns']): btn.setChecked(i == idx)
        is_meta_btn = ("メタグラム" in ui['area_btns'][-1].text() and idx == len(ui['area_btns'])-1)
        if is_meta_btn:
            ui['canvas'].hide()
            if 'info_panel' in ui: ui['info_panel'].hide()
            if 'meta_ctrl' in ui: ui['meta_ctrl'].show()
        else:
            if 'meta_ctrl' in ui: ui['meta_ctrl'].hide()
            ui['canvas'].show()
            self.draw_frame(model_name, ui['slider'].value())

    def on_init_changed(self, model_name, idx):
        ui = self.tab_ui[model_name]
        if idx < 0: return
        selected_time_str = ui['init_combo'].itemData(idx)
        if selected_time_str:
            self.model_init_times[model_name] = selected_time_str
            self.update_slider_max(model_name, selected_time_str)
            if len(ui['area_btns']) > 4 and ui['area_btns'][-2].isChecked(): return
            
            elem_name = ui['list'].currentItem().text()
            val = ui['slider'].value()
            if "降水" in elem_name and val == 0 and "確率" not in elem_name and "時間降水" not in elem_name and model_name != "ANAL":
                val = ui['step']
                ui['slider'].blockSignals(True)
                ui['slider'].setValue(val)
                ui['slider'].blockSignals(False)

            self.draw_frame(model_name, val)

    def on_element_changed(self, model_name):
        ui = self.tab_ui[model_name]
        elem_name = ui['list'].currentItem().text()
        
        if "GUID" in model_name:
            min_ft = 0
            if "6時間" in elem_name: min_ft = 6
            elif "24時間" in elem_name: min_ft = 24
            elif "3時間" in elem_name: min_ft = 3
            
            ui['slider'].setMinimum(min_ft)
            if ui['slider'].value() < min_ft:
                ui['slider'].blockSignals(True)
                ui['slider'].setValue(min_ft)
                ui['slider'].blockSignals(False)
        else:
            ui['slider'].setMinimum(0)
            
        self.on_slider_changed(model_name, ui['slider'].value())

    def on_slider_changed(self, model_name, val):
        self.draw_frame(model_name, val)

    def update_slider_max(self, ui_key, init_time_str):
        ui = self.tab_ui.get(ui_key)
        if not ui: return
        files = glob.glob(os.path.join(self.cache_dir, f"{ui_key}_{init_time_str}_FT*.npz"))
        max_ft_found = max([0] + [int(re.search(r'_FT(\d+)\.npz', f).group(1)) for f in files if re.search(r'_FT(\d+)\.npz', f)])
        hh = init_time_str[8:10] if len(init_time_str) >= 10 else '00'
        
        if ui_key == "MSM": default_max = 78 if hh in ['00', '12'] else 39; step = 1; tick_step = 6
        elif ui_key == "GSM_JP": default_max = 264 if hh in ['00', '12'] else 132; step = 3; tick_step = 12
        elif ui_key == "ANAL": default_max = 0; step = 1; tick_step = 1
        elif ui_key == "MSM_GUID": default_max = 84; step = 1; tick_step = 6
        elif ui_key == "GSM_GUID": default_max = 84; step = 3; tick_step = 12
        else: default_max = 132; step = 6; tick_step = 24
            
        final_max = max(default_max, max_ft_found); final_max = (final_max // step) * step
        
        ui['slider'].setMaximum(final_max)
        ui['slider'].setTickInterval(step); ui['slider'].setSingleStep(step); ui['slider'].setPageStep(step)
        
        ticks_layout = ui['ticks_layout']
        for i in reversed(range(ticks_layout.count())):
            w = ticks_layout.itemAt(i).widget()
            if w: w.setParent(None)
        
        if ui_key != "ANAL":
            for t in range(0, final_max + 1, tick_step):
                lbl = QLabel(f"{t}h"); lbl.setStyleSheet("font-size: 8pt; color: #8892B0;"); lbl.setAlignment(Qt.AlignmentFlag.AlignLeft | Qt.AlignmentFlag.AlignVCenter); ticks_layout.addWidget(lbl)

    def create_compressed_left_panel(self, model_name):
        left_panel = QVBoxLayout(); left_panel.setContentsMargins(5, 5, 5, 5)
        step = 1 if model_name in ["MSM", "ANAL", "MSM_GUID"] else 3

        left_panel.addWidget(QLabel(f"📡 {model_name} 初期時刻(UTC):", styleSheet="font-weight: bold; font-size: 11pt; color: #E0E0E0;"))
        init_combo = QComboBox(); init_combo.addItem("(データ待機中...)", None); left_panel.addWidget(init_combo)
        interval_label = QLabel(f"【{step}時間間隔表示】", styleSheet="color: #64FFDA; font-weight: bold; font-size: 11pt;"); left_panel.addWidget(interval_label)

        mid_hbox = QHBoxLayout(); elem_vbox = QVBoxLayout(); elem_vbox.addWidget(QLabel("■ 要素選択"))
        element_list = QListWidget(); element_list.setFixedWidth(200)
        
        if "GUID" in model_name:
            elems = ["卓越天気", "3時間降水量", "6時間降水量", "24時間降水量", "3時間降雪量", "6時間降雪量", "24時間降雪量", "発雷確率"]
        elif model_name == "ANAL":
            elems = ["地上気温", "地上風向風速", "300hpa風向・風速", "500hpa風向・風速", "500hpa気温", "700hpa気温・風", "850hpa気温・風", "925hpa気温・風", "975hpa気温・風"]
        else:
            elems = ["地上気圧","地上風向風速", "地上降水", "全雲量", "上層雲量", "中層雲量", "下層雲量", "300hpa高度", "300hpa風向・風速", "500hpa高度", "500hpa風向・風速", "500hpa渦度", "500hpa気温", "700hpa湿数・風", "700hpa鉛直流", "850hpa気温・風", "850hpa相当温位", "850hpa湿数・風", "925hpa相当温位", "925hpa湿数・風", "975hpa相当温位", "975hpa湿数・風"]
            
        element_list.addItems(elems); element_list.setCurrentRow(0); elem_vbox.addWidget(element_list); mid_hbox.addLayout(elem_vbox)

        slider_vbox = QVBoxLayout()
        time_label = QLabel("FT=0h\n(--/--)"); time_label.setStyleSheet("font-weight: bold; color: #64FFDA; font-size: 11pt; background: #1D3557; padding: 5px; border: 2px solid #457B9D; border-radius: 5px;"); time_label.setAlignment(Qt.AlignmentFlag.AlignCenter); slider_vbox.addWidget(time_label)

        slider_control_hbox = QHBoxLayout()
        btn_prev = QPushButton("▲"); btn_prev.setProperty("class", "NavButton"); btn_prev.setFixedWidth(30)
        btn_next = QPushButton("▼"); btn_next.setProperty("class", "NavButton"); btn_next.setFixedWidth(30)
        slider = QSlider(Qt.Orientation.Vertical); slider.setRange(0, 39); slider.setTickPosition(QSlider.TickPosition.TicksLeft); slider.setTickInterval(step); slider.setSingleStep(step); slider.setPageStep(step); slider.setInvertedAppearance(True)
        
        btn_vbox = QVBoxLayout(); btn_vbox.addWidget(slider, stretch=1); btn_vbox.setAlignment(Qt.AlignmentFlag.AlignHCenter)
        nav_hbox = QHBoxLayout(); nav_hbox.addWidget(btn_prev); nav_hbox.addWidget(btn_next)
        inner_slider_vbox = QVBoxLayout(); inner_slider_vbox.addLayout(nav_hbox); inner_slider_vbox.addLayout(btn_vbox); slider_control_hbox.addLayout(inner_slider_vbox)

        ticks_vbox = QVBoxLayout(); ticks_vbox.setContentsMargins(0, 30, 0, 10)
        slider_control_hbox.addLayout(ticks_vbox); slider_vbox.addLayout(slider_control_hbox, stretch=1)
        
        play_hbox = QHBoxLayout()
        btn_play = QPushButton("▶"); btn_play.setProperty("class", "PlayButton"); btn_play.setFixedWidth(40)
        btn_stop = QPushButton("■"); btn_stop.setProperty("class", "StopButton"); btn_stop.setFixedWidth(40)
        play_hbox.addWidget(btn_play); play_hbox.addWidget(btn_stop); slider_vbox.addLayout(play_hbox)
        
        combo_speed = QComboBox(); combo_speed.addItems(["速1", "速2", "速3", "速4", "速5"]); combo_speed.setCurrentIndex(2); slider_vbox.addWidget(combo_speed)
        mid_hbox.addLayout(slider_vbox); left_panel.addLayout(mid_hbox, stretch=1)

        return left_panel, init_combo, interval_label, element_list, time_label, btn_prev, btn_next, slider, btn_play, btn_stop, combo_speed, ticks_vbox

    def setup_tab(self, model_name, tab_title, is_guidance=False, is_anal=False):
        tab = QWidget(); main_hbox = QHBoxLayout(tab)
        left_panel, init_combo, interval_label, element_list, time_label, btn_prev, btn_next, slider, btn_play, btn_stop, combo_speed, ticks_layout = self.create_compressed_left_panel(model_name)
        main_hbox.addLayout(left_panel, stretch=2)

        right_panel = QVBoxLayout()
        area_layout = QHBoxLayout(); area_layout.addWidget(QLabel("表示領域:"))
        
        area_btns = [QPushButton("日本周辺"), QPushButton("北海道全域"), QPushButton("上川地方"), QPushButton("札幌近郊")]
        for i, btn in enumerate(area_btns):
            btn.setCheckable(True); area_layout.addWidget(btn); btn.clicked.connect(lambda checked, idx=i, m=model_name: self.on_area_btn_clicked(m, idx))
        area_btns[0].setChecked(True)
        
        if not is_anal:
            btn_meta_text = "メタグラム表(A4画像化)" if is_guidance else f"{model_name}メタグラム"
            btn_meta_draw = QPushButton(btn_meta_text)
            btn_meta_draw.setProperty("class", "MetaButton")
            btn_meta_draw.setCheckable(True)
            area_layout.addWidget(btn_meta_draw); area_btns.append(btn_meta_draw)
            btn_meta_draw.clicked.connect(lambda checked, m=model_name: self.on_area_btn_clicked(m, len(area_btns)-1))
        
        area_layout.addStretch()
        right_panel.addLayout(area_layout)

        fig = Figure(figsize=(14, 13)); fig.patch.set_facecolor('white'); canvas = FigureCanvas(fig)

        meta_ctrl_widget = QWidget(); meta_ctrl_layout = QVBoxLayout(meta_ctrl_widget)
        h_ctrl = QHBoxLayout()
        h_ctrl.addWidget(QLabel("地点選択:")); station_combo = QComboBox()
        for name, lat, lon in STATIONS: station_combo.addItem(name, (lat, lon))
        h_ctrl.addWidget(station_combo)
        
        if not is_anal:
            btn_text = "表示 (A4横サイズ テーブル画像生成)" if is_guidance else "表示 (図解メタグラム生成)"
            btn_generate_meta = QPushButton(btn_text)
            btn_generate_meta.setProperty("class", "SyncButton")
            if is_guidance: btn_generate_meta.clicked.connect(lambda: self.draw_guidance_metagram(station_combo, model_name))
            elif model_name == "GSM_JP": btn_generate_meta.clicked.connect(lambda: self.draw_gsm_visual_metagram(station_combo))
            elif model_name == "MSM": btn_generate_meta.clicked.connect(lambda: self.draw_msm_visual_metagram(station_combo))
            h_ctrl.addWidget(btn_generate_meta)
            
        h_ctrl.addStretch()
        meta_ctrl_layout.addLayout(h_ctrl); meta_ctrl_layout.addStretch(); meta_ctrl_widget.hide()

        right_panel.addWidget(canvas, stretch=10)
        right_panel.addWidget(meta_ctrl_widget, stretch=10)
        main_hbox.addLayout(right_panel, stretch=8)

        info_panel = QLabel(canvas)
        info_panel.setStyleSheet("background-color: rgba(17, 34, 64, 0.85); border: 2px solid #64FFDA; border-radius: 5px; padding: 5px; font-weight: bold; font-size: 12pt; color: #E0E0E0;"); info_panel.hide()

        step = 1 if model_name in ["MSM", "ANAL", "MSM_GUID"] else 3

        self.tab_ui[model_name] = {
            'list': element_list, 'area_btns': area_btns, 'slider': slider, 'time_label': time_label, 'init_combo': init_combo, 
            'interval_label': interval_label, 'fig': fig, 'canvas': canvas, 'meta_ctrl': meta_ctrl_widget, 'ax': None, 'current_area_index': -1, 
            'dynamic_artists': [], 'colorbars': [], 'info_panel': info_panel, 'combo_speed': combo_speed, 'ticks_layout': ticks_layout,
            'step': step
        }
        
        if is_anal:
            btn_play.hide(); btn_stop.hide(); combo_speed.hide()
            btn_prev.hide(); btn_next.hide()
            slider.hide()
            for i in range(ticks_layout.count()):
                w = ticks_layout.itemAt(i).widget()
                if w: w.hide()
            interval_label.setText("【現況解析のみ(FT=0)】")
        
        def step_slider(val_change, m=model_name):
            ui = self.tab_ui[m]; s_step = ui['step']
            new_val = ui['slider'].value() + (s_step * val_change)
            if new_val > ui['slider'].maximum(): new_val = ui['slider'].minimum()
            if new_val < ui['slider'].minimum(): new_val = ui['slider'].maximum()
            ui['slider'].setValue(new_val)

        if not is_anal:
            btn_prev.clicked.connect(lambda: step_slider(-1)); btn_next.clicked.connect(lambda: step_slider(1))
            slider.valueChanged.connect(lambda val, m=model_name: self.on_slider_changed(m, val))
            btn_play.clicked.connect(lambda: self.start_animation(model_name)); btn_stop.clicked.connect(self.stop_animation)
        element_list.itemSelectionChanged.connect(lambda m=model_name: self.on_element_changed(m))
        self.tabs.addTab(tab, tab_title)

    def get_exact_key(self, data_obj, exact_candidates):
        for c in exact_candidates:
            if c in data_obj.files: return c
        for k in data_obj.files:
            if k.lower() in [c.lower() for c in exact_candidates]: return k
        return None

    def get_precip_key(self, data):
        return self.get_exact_key(data, ['precip', 'pr', 'tp', 'apcp', 'prate', 'rain', 'rn', 'pop', 'var_0_1_8', 'var_0_1_52'])

    def get_w_key(self, data):
        return self.get_exact_key(data, ['w', 'w700', 'vvel', 'vvel700', 'dzdt', 'dzdt700'])

    def get_coords_for_data(self, data_obj, data_array):
        if data_array is None: return None, None
        ny = data_array.shape[0]; nx = data_array.shape[1] if data_array.ndim > 1 else len(data_array)
        for k in data_obj.files:
            if 'lon' in k.lower():
                if (data_obj[k].ndim == 1 and data_obj[k].shape[0] == nx) or (data_obj[k].ndim == 2 and data_obj[k].shape == data_array.shape):
                    lat_k = k.replace('lon', 'lat').replace('Lon', 'Lat')
                    if lat_k in data_obj.files: return data_obj[k], data_obj[lat_k]
                    for lk in data_obj.files:
                        if 'lat' in lk.lower() and ((data_obj[lk].ndim == 1 and data_obj[lk].shape[0] == ny) or data_obj[lk].shape == data_array.shape):
                            return data_obj[k], data_obj[lk]
        return None, None

    def extract_val_with_key(self, data, key, stat_lon, stat_lat):
        if key not in data.files: return np.nan
        arr = data[key]
        lon_arr, lat_arr = self.get_coords_for_data(data, arr)
        
        if lon_arr is None or lat_arr is None:
            lon_k = next((k for k in data.files if 'lon' in k.lower()), None)
            lat_k = next((k for k in data.files if 'lat' in k.lower()), None)
            if not lon_k or not lat_k: return np.nan
            lon_arr, lat_arr = data[lon_k], data[lat_k]

        if lon_arr.ndim == 1 and arr.shape == (len(lon_arr), len(lat_arr)):
            arr = arr.T
            
        if lon_arr.ndim == 1 and arr.shape != (len(lat_arr), len(lon_arr)):
            if arr.shape[1] > arr.shape[0] and len(lat_arr) > len(lon_arr):
                arr = arr.T
            lon_arr = np.linspace(lon_arr.min(), lon_arr.max(), arr.shape[1])
            lat_arr = np.linspace(lat_arr.min(), lat_arr.max(), arr.shape[0])
            
        if lon_arr.ndim == 1: lon_2d, lat_2d = np.meshgrid(lon_arr, lat_arr)
        else: lon_2d, lat_2d = lon_arr, lat_arr
            
        dist = (lon_2d - stat_lon)**2 + (lat_2d - stat_lat)**2
        sy, sx = np.unravel_index(np.argmin(dist), dist.shape)
        
        if sy < arr.shape[0] and sx < arr.shape[1]:
            v = arr[sy, sx]
            if np.isnan(v) or v > 200000 or v < -10000: return np.nan
            return float(v)
        return np.nan
    
    def extract_val_robust(self, data, keys, stat_lon, stat_lat):
        k = self.get_exact_key(data, keys)
        if not k:
            for dk in data.files:
                if any(x in dk.lower() for x in keys):
                    k = dk; break
        if not k: return np.nan
        return self.extract_val_with_key(data, k, stat_lon, stat_lat)

    # --- MSM用メタグラム ---
    def draw_msm_visual_metagram(self, station_combo):
        it_str = self.model_init_times.get("MSM")
        if not it_str:
            QMessageBox.warning(self, "エラー", "データが存在しません。"); return
            
        stat_name = station_combo.currentText(); stat_lat, stat_lon = station_combo.currentData()
        dt_utc = datetime.strptime(it_str, "%Y%m%d%H%M%S") # UTC
        dt_jst = dt_utc + timedelta(hours=9)
        init_z_str = f"{dt_utc.strftime('%m%d%H')}Z" 
        title_time_str = f"{dt_utc.strftime('%m月%d日%H')}Z / {dt_jst.strftime('%H')}JST" 
        
        max_ft = self.tab_ui['MSM']['slider'].maximum()
        fts_surf = list(range(0, max_ft + 1))
        fts_prof = [ft for ft in fts_surf if ft % 3 == 0]
        
        plevs = [1000, 975, 950, 925, 850, 700, 600, 500, 300]; y_coords = [0.0, 0.05, 0.1, 0.15, 0.25, 0.5, 0.625, 0.75, 1.0]
        T_prof = np.full((len(plevs), len(fts_prof)), np.nan); RH_prof = np.full((len(plevs), len(fts_prof)), np.nan)
        U_prof = np.full((len(plevs), len(fts_prof)), np.nan); V_prof = np.full((len(plevs), len(fts_prof)), np.nan)
        SSI_850_series = np.full(len(fts_prof), np.nan); SSI_700_series = np.full(len(fts_prof), np.nan); SSI_925_series = np.full(len(fts_prof), np.nan)
        P_surf_series = np.full(len(fts_surf), np.nan); T_surf_series = np.full(len(fts_surf), np.nan); Pr_surf_series = np.full(len(fts_surf), np.nan)

        for c_idx, ft in enumerate(fts_surf):
            gpv_file = os.path.join(self.cache_dir, f"MSM_{it_str}_FT{ft:02d}.npz")
            if not os.path.exists(gpv_file): continue
            try:
                d = np.load(gpv_file)
                P_surf_series[c_idx] = self.extract_val_robust(d, ['prmsl', 'mslet', 'msl', 'slp'], stat_lon, stat_lat)
                if not np.isnan(P_surf_series[c_idx]) and P_surf_series[c_idx] > 2000: P_surf_series[c_idx] /= 100.0
                
                T_surf_series[c_idx] = self.extract_val_robust(d, ['t2m', 'tmp2m', 'temp2m'], stat_lon, stat_lat)
                if not np.isnan(T_surf_series[c_idx]) and T_surf_series[c_idx] > 100: T_surf_series[c_idx] -= 273.15
                
                p_key = self.get_precip_key(d)
                if p_key:
                    pr = self.extract_val_with_key(d, p_key, stat_lon, stat_lat)
                    Pr_surf_series[c_idx] = pr if not np.isnan(pr) else 0.0
                else:
                    Pr_surf_series[c_idx] = 0.0
                d.close()
            except Exception as e: logging.error(f"Error reading {gpv_file}: {e}")

        for c_idx, ft in enumerate(fts_prof):
            gpv_file = os.path.join(self.cache_dir, f"MSM_{it_str}_FT{ft:02d}.npz")
            if not os.path.exists(gpv_file): continue
            try:
                d = np.load(gpv_file)
                t2m = self.extract_val_robust(d, ['t2m', 'tmp2m', 'temp2m'], stat_lon, stat_lat)
                if not np.isnan(t2m) and t2m > 100: t2m -= 273.15
                T_prof[0, c_idx] = t2m
                
                RH_prof[0, c_idx] = self.extract_val_robust(d, ['rh2m', 'r2m'], stat_lon, stat_lat)
                U_prof[0, c_idx] = self.extract_val_robust(d, ['u10', '10u', 'u_10m'], stat_lon, stat_lat)
                V_prof[0, c_idx] = self.extract_val_robust(d, ['v10', '10v', 'v_10m'], stat_lon, stat_lat)
                
                h_map = {975:1, 950:2, 925:3, 850:4, 700:5, 600:6, 500:7, 300:8}
                for p_lvl, idx in h_map.items():
                    t_val = self.extract_val_robust(d, [f't{p_lvl}', f'tmp{p_lvl}', f't_{p_lvl}', f'tmp_{p_lvl}'], stat_lon, stat_lat)
                    if not np.isnan(t_val) and t_val > 100: t_val -= 273.15
                    
                    rh_val = self.extract_val_robust(d, [f'rh{p_lvl}', f'r{p_lvl}', f'rh_{p_lvl}', f'r_{p_lvl}'], stat_lon, stat_lat)
                    if np.isnan(rh_val) and not np.isnan(t_val):
                        t_td = self.extract_val_robust(d, [f't_td{p_lvl}', f'dew{p_lvl}'], stat_lon, stat_lat)
                        if not np.isnan(t_td): rh_val = rh_from_t_td(t_val, t_td)
                    
                    T_prof[idx, c_idx] = rh_val if not np.isnan(rh_val) else 0.0
                    T_prof[idx, c_idx] = t_val
                    RH_prof[idx, c_idx] = rh_val if not np.isnan(rh_val) else 0.0
                    U_prof[idx, c_idx] = self.extract_val_robust(d, [f'u{p_lvl}', f'u_{p_lvl}'], stat_lon, stat_lat)
                    V_prof[idx, c_idx] = self.extract_val_robust(d, [f'v{p_lvl}', f'v_{p_lvl}'], stat_lon, stat_lat)

                t925 = T_prof[3, c_idx]; rh925 = RH_prof[3, c_idx]
                t850 = T_prof[4, c_idx]; rh850 = RH_prof[4, c_idx]
                t700 = T_prof[5, c_idx]; rh700 = RH_prof[5, c_idx]
                t500 = T_prof[7, c_idx]
                
                if not np.isnan(t500):
                    if not np.isnan(t850) and not np.isnan(rh850):
                        SSI_850_series[c_idx] = t500 - get_t_from_ept(calc_ept(t850, rh850, 850), 500)
                    if not np.isnan(t700) and not np.isnan(rh700):
                        SSI_700_series[c_idx] = t500 - get_t_from_ept(calc_ept(t700, rh700, 700), 500)
                    if not np.isnan(t700) and not np.isnan(t925) and not np.isnan(rh925):
                        SSI_925_series[c_idx] = t700 - get_t_from_ept(calc_ept(t925, rh925, 925), 700)
                d.close()
            except: pass

        fig = Figure(figsize=(10, 16)); fig.patch.set_facecolor('white')
        fig.subplots_adjust(left=0.08, right=0.98, top=0.92, bottom=0.08, hspace=0.65) 
        fig.suptitle(f"MSM 図解メタグラム - {stat_name} (初期時: {title_time_str})", fontsize=14, weight='bold', y=0.97)

        gs = fig.add_gridspec(14, 1)
        ax_cross = fig.add_subplot(gs[0:5, 0]); ax_slp = fig.add_subplot(gs[5:7, 0], sharex=ax_cross)
        ax_t2m = fig.add_subplot(gs[7:9, 0], sharex=ax_cross); ax_pr = fig.add_subplot(gs[9:11, 0], sharex=ax_cross)
        ax_ssi = fig.add_subplot(gs[11:14, 0], sharex=ax_cross)
        
        X_prof, Y_prof = np.meshgrid(fts_prof, y_coords)
        cmap = mcolors.ListedColormap(['#E8F8F5', '#A3E4D7', '#48C9B0', '#117A65'])
        bounds = [70, 80, 90, 95, 100]; norm = mcolors.BoundaryNorm(bounds, cmap.N)
        cf = ax_cross.contourf(X_prof, Y_prof, RH_prof, levels=bounds, cmap=cmap, norm=norm, extend='max', alpha=0.6)
        
        cs = ax_cross.contour(X_prof, Y_prof, T_prof, levels=np.arange(-60, 40, 6), colors='red', linewidths=1.0, alpha=0.7)
        ax_cross.clabel(cs, inline=True, fontsize=9, fmt='%1.0f')
        cs_zero = ax_cross.contour(X_prof, Y_prof, T_prof, levels=[0], colors='blue', linewidths=2.0)
        ax_cross.clabel(cs_zero, inline=True, fontsize=10, fmt='0℃')
        ax_cross.barbs(X_prof[:, ::1], Y_prof[:, ::1], U_prof[:, ::1], V_prof[:, ::1], length=4.5, color='black', linewidth=0.6)
        
        ax_cross.set_yticks(y_coords); ax_cross.set_yticklabels([str(p) for p in plevs])
        ax_cross.set_ylabel("気\n圧\n\n[hPa]", weight='bold', rotation=0, labelpad=10, va='center')
        ax_cross.set_title("湿域・気温・風 断面図 (3時間毎)", loc='left', weight='bold')

        ax_slp.plot(fts_surf, P_surf_series, color='#2980B9', marker='s', markersize=3, linestyle='-', linewidth=2, label="海面気圧")
        ax_slp.set_ylabel("海\n面\n気\n圧\n\n[hPa]", color='#2980B9', weight='bold', rotation=0, labelpad=10, va='center')
        ax_slp.legend(loc='upper left')

        ax_top_jst = ax_slp.twiny()
        ax_top_jst.set_xlim(ax_slp.get_xlim())
        ax_top_jst.set_xticks(range(0, fts_surf[-1]+1, 6))
        jsts = [(dt_jst + timedelta(hours=x)).strftime("%d日%H時") for x in range(0, fts_surf[-1]+1, 6)]
        ax_top_jst.set_xticklabels(jsts, fontsize=9, color='#333333', weight='bold')

        ax_t2m.plot(fts_surf, T_surf_series, color='#E74C3C', marker='o', markersize=3, linestyle='-', linewidth=2, label="地上気温")
        ax_t2m.set_ylabel("地\n上\n気\n温\n\n[℃]", color='#E74C3C', weight='bold', rotation=0, labelpad=10, va='center')
        ax_t2m.legend(loc='upper left')

        ax_pr.bar(fts_surf, Pr_surf_series, width=0.6, color='#3498DB', alpha=0.9, label="期間降水量 (1時間積算)")
        ax_pr.set_ylabel("降\n水\n量\n\n[mm]", color='#3498DB', weight='bold', rotation=0, labelpad=10, va='center')
        ax_pr.legend(loc='upper left')
        ax_pr.axhline(0, color='gray', linewidth=1.0)

        ax_ssi.plot(fts_prof, SSI_850_series, color='#C0392B', marker='o', markersize=4, linestyle='-', linewidth=2, label="SSI (850-500)")
        ax_ssi.plot(fts_prof, SSI_700_series, color='#E67E22', marker='^', markersize=4, linestyle='-', linewidth=2, label="SSI (700-500)")
        ax_ssi.plot(fts_prof, SSI_925_series, color='#8E44AD', marker='s', markersize=4, linestyle='-', linewidth=2, label="SSI (925-700)")
        ax_ssi.axhline(0, color='gray', linestyle=':', linewidth=1)
        ax_ssi.set_ylabel("S\nS\nI\n\n[℃]", weight='bold', rotation=0, labelpad=10, va='center')
        ax_ssi.set_xlabel("予報時間 (FT)", weight='bold', fontsize=12)
        ax_ssi.legend(loc='lower left')

        for ax in [ax_cross, ax_slp, ax_t2m, ax_pr, ax_ssi]:
            ax.set_xlim(fts_surf[0]-1, fts_surf[-1]+1)
            ax.set_xticks(range(0, fts_surf[-1]+1, 6))
            ax.grid(True, axis='x', color='gray', linestyle='--', alpha=0.5)
            ax.grid(True, axis='y', color='gray', linestyle='--', alpha=0.3)
            if ax != ax_ssi:
                plt.setp(ax.get_xticklabels(), visible=False)
        
        ax_top = ax_cross.twiny()
        ax_top.set_xlim(ax_cross.get_xlim())
        ax_top.set_xticks(range(0, fts_surf[-1]+1, 6))
        ax_top.set_xticklabels([f"FT{x}" for x in range(0, fts_surf[-1]+1, 6)], fontsize=10, weight='bold')

        plt.setp(ax_ssi.get_xticklabels(), rotation=45, ha='right', fontsize=11)
            
        fig.align_ylabels([ax_cross, ax_slp, ax_t2m, ax_pr, ax_ssi]) 
        canvas = FigureCanvas(fig)
        win = SingleImageWindow(self, f"MSM 図解メタグラム - {stat_name}", "MSM", stat_name, init_z_str, canvas)
        win.exec()

    # --- GSM用メタグラム (修正済み) ---
    def draw_gsm_visual_metagram(self, station_combo):
        it_str = self.model_init_times.get("GSM_JP")
        if not it_str:
            QMessageBox.warning(self, "エラー", "データが存在しません。"); return
            
        stat_name = station_combo.currentText(); stat_lat, stat_lon = station_combo.currentData()
        dt_utc = datetime.strptime(it_str, "%Y%m%d%H%M%S") # UTC
        dt_jst = dt_utc + timedelta(hours=9)
        init_z_str = f"{dt_utc.strftime('%m%d%H')}Z" 
        title_time_str = f"{dt_utc.strftime('%m月%d日%H')}Z / {dt_jst.strftime('%H')}JST" 
        
        max_ft = self.tab_ui['GSM_JP']['slider'].maximum()
        fts = list(range(0, max_ft + 1, 3))
        
        plevs = [1000, 975, 950, 925, 850, 700, 600, 500, 300]; y_coords = [0.0, 0.05, 0.1, 0.15, 0.25, 0.5, 0.625, 0.75, 1.0]
        T_prof = np.full((len(plevs), len(fts)), np.nan); RH_prof = np.full((len(plevs), len(fts)), np.nan)
        U_prof = np.full((len(plevs), len(fts)), np.nan); V_prof = np.full((len(plevs), len(fts)), np.nan)
        SSI_850_series = np.full(len(fts), np.nan); SSI_700_series = np.full(len(fts), np.nan); SSI_925_series = np.full(len(fts), np.nan)
        P_surf_series = np.full(len(fts), np.nan); T_surf_series = np.full(len(fts), np.nan); Pr_surf_series = np.full(len(fts), np.nan)

        # 修正: 132h以降は地上・上空とも6時間間隔になるため、全て disp_ft にフォールバックして処理
        for c_idx, ft in enumerate(fts):
            disp_ft = ft if ft <= 132 or ft % 6 == 0 else ft - (ft % 6)
            gpv_file = os.path.join(self.cache_dir, f"GSM_JP_{it_str}_FT{disp_ft:02d}.npz")
            
            if os.path.exists(gpv_file):
                try:
                    d = np.load(gpv_file)
                    P_surf_series[c_idx] = self.extract_val_robust(d, ['prmsl', 'mslet', 'msl', 'slp'], stat_lon, stat_lat)
                    if not np.isnan(P_surf_series[c_idx]) and P_surf_series[c_idx] > 2000: P_surf_series[c_idx] /= 100.0
                    
                    T_surf_series[c_idx] = self.extract_val_robust(d, ['t2m', 'tmp2m', 'temp2m'], stat_lon, stat_lat)
                    if not np.isnan(T_surf_series[c_idx]) and T_surf_series[c_idx] > 100: T_surf_series[c_idx] -= 273.15
                    
                    # 修正: 132h以降の降水差分計算エラー防止
                    p_key = self.get_precip_key(d)
                    if p_key:
                        pr = self.extract_val_with_key(d, p_key, stat_lon, stat_lat)
                        if disp_ft >= 3:
                            offset = 6 if disp_ft > 132 else 3
                            prev_file = os.path.join(self.cache_dir, f"GSM_JP_{it_str}_FT{disp_ft-offset:02d}.npz")
                            if os.path.exists(prev_file):
                                try:
                                    with np.load(prev_file) as d_prev:
                                        prev_p_key = self.get_precip_key(d_prev)
                                        if prev_p_key:
                                            prev_pr = self.extract_val_with_key(d_prev, prev_p_key, stat_lon, stat_lat)
                                            if not np.isnan(prev_pr): pr = max(0, pr - prev_pr)
                                except: pass
                        Pr_surf_series[c_idx] = pr if not np.isnan(pr) else 0.0
                    else:
                        Pr_surf_series[c_idx] = 0.0

                    # 上空データ
                    T_prof[0, c_idx] = T_surf_series[c_idx] if not np.isnan(T_surf_series[c_idx]) else self.extract_val_robust(d, ['t2m', 'tmp2m', 'temp2m'], stat_lon, stat_lat)
                    RH_prof[0, c_idx] = self.extract_val_robust(d, ['rh2m', 'r2m'], stat_lon, stat_lat)
                    U_prof[0, c_idx] = self.extract_val_robust(d, ['u10', '10u', 'u_10m'], stat_lon, stat_lat)
                    V_prof[0, c_idx] = self.extract_val_robust(d, ['v10', '10v', 'v_10m'], stat_lon, stat_lat)
                    
                    h_map = {975:1, 950:2, 925:3, 850:4, 700:5, 600:6, 500:7, 300:8}
                    for p_lvl, idx in h_map.items():
                        t_val = self.extract_val_robust(d, [f't{p_lvl}', f'tmp{p_lvl}', f't_{p_lvl}'], stat_lon, stat_lat)
                        if not np.isnan(t_val) and t_val > 100: t_val -= 273.15
                        
                        rh_val = self.extract_val_robust(d, [f'rh{p_lvl}', f'r{p_lvl}', f'rh_{p_lvl}'], stat_lon, stat_lat)
                        if np.isnan(rh_val) and not np.isnan(t_val):
                            t_td = self.extract_val_robust(d, [f't_td{p_lvl}', f'dew{p_lvl}'], stat_lon, stat_lat)
                            if not np.isnan(t_td): rh_val = rh_from_t_td(t_val, t_td)
                        
                        T_prof[idx, c_idx] = t_val
                        RH_prof[idx, c_idx] = rh_val if not np.isnan(rh_val) else 0.0
                        U_prof[idx, c_idx] = self.extract_val_robust(d, [f'u{p_lvl}', f'u_{p_lvl}'], stat_lon, stat_lat)
                        V_prof[idx, c_idx] = self.extract_val_robust(d, [f'v{p_lvl}', f'v_{p_lvl}'], stat_lon, stat_lat)

                    t925 = T_prof[3, c_idx]; rh925 = RH_prof[3, c_idx]
                    t850 = T_prof[4, c_idx]; rh850 = RH_prof[4, c_idx]
                    t700 = T_prof[5, c_idx]; rh700 = RH_prof[5, c_idx]
                    t500 = T_prof[7, c_idx]
                    
                    if not np.isnan(t500):
                        if not np.isnan(t850) and not np.isnan(rh850):
                            SSI_850_series[c_idx] = t500 - get_t_from_ept(calc_ept(t850, rh850, 850), 500)
                        if not np.isnan(t700) and not np.isnan(rh700):
                            SSI_700_series[c_idx] = t500 - get_t_from_ept(calc_ept(t700, rh700, 700), 500)
                        if not np.isnan(t700) and not np.isnan(t925) and not np.isnan(rh925):
                            SSI_925_series[c_idx] = t700 - get_t_from_ept(calc_ept(t925, rh925, 925), 700)
                    d.close()
                except: pass

        fig = Figure(figsize=(15, 12)); fig.patch.set_facecolor('white')
        fig.subplots_adjust(left=0.08, right=0.98, top=0.91, bottom=0.08, hspace=0.65) 
        fig.suptitle(f"GSM_JP 図解メタグラム - {stat_name} (初期時: {title_time_str})", fontsize=14, weight='bold', y=0.97)

        gs = fig.add_gridspec(12, 1); ax_cross = fig.add_subplot(gs[0:4, 0]); ax_slp = fig.add_subplot(gs[4:6, 0], sharex=ax_cross)
        ax_t2m = fig.add_subplot(gs[6:8, 0], sharex=ax_cross); ax_pr = fig.add_subplot(gs[8:10, 0], sharex=ax_cross); ax_ssi = fig.add_subplot(gs[10:12, 0], sharex=ax_cross)
        
        X, Y = np.meshgrid(fts, y_coords)
        cmap = mcolors.ListedColormap(['#E8F8F5', '#A3E4D7', '#48C9B0', '#117A65'])
        bounds = [70, 80, 90, 95, 100]; norm = mcolors.BoundaryNorm(bounds, cmap.N)
        cf = ax_cross.contourf(X, Y, RH_prof, levels=bounds, cmap=cmap, norm=norm, extend='max', alpha=0.6)
        
        cs = ax_cross.contour(X, Y, T_prof, levels=np.arange(-60, 40, 6), colors='red', linewidths=1.0, alpha=0.7)
        ax_cross.clabel(cs, inline=True, fontsize=9, fmt='%1.0f')
        cs_zero = ax_cross.contour(X, Y, T_prof, levels=[0], colors='blue', linewidths=2.0)
        ax_cross.clabel(cs_zero, inline=True, fontsize=10, fmt='0℃')
        ax_cross.barbs(X[:, ::1], Y[:, ::1], U_prof[:, ::1], V_prof[:, ::1], length=4.5, color='black', linewidth=0.6)
        
        ax_cross.set_yticks(y_coords); ax_cross.set_yticklabels([str(p) for p in plevs])
        ax_cross.set_ylabel("気\n圧\n\n[hPa]", weight='bold', rotation=0, labelpad=10, va='center'); ax_cross.set_title("湿域・気温・風 断面図 (※132h以降は全要素6時間間隔で補間展開)", loc='left', weight='bold')

        ax_slp.plot(fts, P_surf_series, color='#2980B9', marker='s', markersize=4, linestyle='-', linewidth=2, label="海面気圧")
        ax_slp.set_ylabel("海\n面\n気\n圧\n\n[hPa]", color='#2980B9', weight='bold', rotation=0, labelpad=10, va='center')
        ax_slp.legend(loc='upper left')

        ax_top_jst = ax_slp.twiny()
        ax_top_jst.set_xlim(ax_slp.get_xlim())
        ax_top_jst.set_xticks(range(0, fts[-1]+1, 12))
        jsts = [(dt_jst + timedelta(hours=x)).strftime("%d日%H時") for x in range(0, fts[-1]+1, 12)]
        ax_top_jst.set_xticklabels(jsts, fontsize=9, color='#333333', weight='bold')

        ax_t2m.plot(fts, T_surf_series, color='#E74C3C', marker='o', markersize=4, linestyle='-', linewidth=2, label="地上気温")
        ax_t2m.set_ylabel("地\n上\n気\n温\n\n[℃]", color='#E74C3C', weight='bold', rotation=0, labelpad=10, va='center')
        ax_t2m.legend(loc='upper left')

        ax_pr.bar(fts, Pr_surf_series, width=1.8, color='#3498DB', alpha=0.9, label="期間降水量 (差分)")
        ax_pr.set_ylabel("降\n水\n量\n\n[mm]", color='#3498DB', weight='bold', rotation=0, labelpad=10, va='center')
        ax_pr.legend(loc='upper left')
        ax_pr.axhline(0, color='gray', linewidth=1.0)

        ax_ssi.plot(fts, SSI_850_series, color='#C0392B', marker='o', markersize=4, linestyle='-', linewidth=2, label="SSI (850-500)")
        ax_ssi.plot(fts, SSI_700_series, color='#E67E22', marker='^', markersize=4, linestyle='-', linewidth=2, label="SSI (700-500)")
        ax_ssi.plot(fts, SSI_925_series, color='#8E44AD', marker='s', markersize=4, linestyle='-', linewidth=2, label="SSI (925-700)")
        ax_ssi.axhline(0, color='gray', linestyle=':', linewidth=1)
        ax_ssi.set_ylabel("S\nS\nI\n\n[℃]", weight='bold', rotation=0, labelpad=10, va='center')
        ax_ssi.set_xlabel("予報時間 (FT)", weight='bold', fontsize=12)
        ax_ssi.legend(loc='lower left')

        for ax in [ax_cross, ax_slp, ax_t2m, ax_pr, ax_ssi]:
            ax.set_xlim(fts[0]-1, fts[-1]+1)
            ax.set_xticks(range(0, fts[-1]+1, 12))
            ax.grid(True, axis='x', color='gray', linestyle='--', alpha=0.5)
            ax.grid(True, axis='y', color='gray', linestyle='--', alpha=0.3)
            if ax != ax_ssi:
                plt.setp(ax.get_xticklabels(), visible=False)
        
        ax_top = ax_cross.twiny()
        ax_top.set_xlim(ax_cross.get_xlim())
        ax_top.set_xticks(range(0, fts[-1]+1, 12))
        ax_top.set_xticklabels([f"FT{x}" for x in range(0, fts[-1]+1, 12)], fontsize=10, weight='bold')

        plt.setp(ax_ssi.get_xticklabels(), rotation=45, ha='right', fontsize=11)
            
        fig.align_ylabels([ax_cross, ax_slp, ax_t2m, ax_pr, ax_ssi]) 
        canvas = FigureCanvas(fig)
        win = SingleImageWindow(self, f"GSM_JP 図解メタグラム - {stat_name}", "GSM_JP", stat_name, init_z_str, canvas)
        win.exec()

    def draw_guidance_metagram(self, station_combo, model_name):
        it_str = self.model_init_times.get(model_name)
        if not it_str:
            QMessageBox.warning(self, "エラー", "データが存在しません。"); return
            
        stat_name = station_combo.currentText(); stat_lat, stat_lon = station_combo.currentData()
        dt_utc = datetime.strptime(it_str, "%Y%m%d%H%M%S") # UTC
        
        fig = Figure(figsize=(11.69, 8.27)); fig.patch.set_facecolor('white')
        ax = fig.add_subplot(111); ax.axis('off')
        
        table_data = []
        headers = ["FT", "日時(JST)", "卓越天気", "降水量", "降雪量", "発雷確率", "気温(℃)", "風向", "風速(m/s)"]
        
        max_ft = self.tab_ui[model_name]['slider'].maximum()
        fts = list(range(0, max_ft + 1, 3))
        
        gpv_model = "MSM" if model_name == "MSM_GUID" else "GSM_JP"
        gpv_it_str = self.model_init_times.get(gpv_model)
        
        for ft in fts:
            target_jst = dt_utc + timedelta(hours=9+ft)
            row = [f"FT={ft:02d}", target_jst.strftime("%m/%d %H:00"), "-", "-", "-", "-", "-", "-", "-"]
            
            guid_file = os.path.join(self.cache_dir, f"{model_name}_{it_str}_FT{ft:02d}.npz")
            if os.path.exists(guid_file):
                try:
                    d = np.load(guid_file)
                    pop = self.extract_val_robust(d, ['precip', 'tp', 'apcp', 'pop', 'var_0_1_8', 'var_0_1_52'], stat_lon, stat_lat)
                    pos = self.extract_val_robust(d, ['snow', 'weasd', 'snod', 'var_0_1_11', 'var_0_1_13', 'var_0_1_29'], stat_lon, stat_lat)
                    pol = self.extract_val_robust(d, ['thund', 'pol', 'lig', 'thunder', 'tstm', 'var_0_19_193'], stat_lon, stat_lat)
                    wx = self.extract_val_robust(d, ['wea', 'weather', 'wx', 'var_-1_-1_-1', 'var_0_19_192'], stat_lon, stat_lat)                   
                    
                    if not np.isnan(wx): row[2] = {0:"晴", 1:"曇", 2:"雨", 3:"雪"}.get(int(wx), "不明")
                    if not np.isnan(pop): row[3] = f"{pop:.1f} mm"
                    if not np.isnan(pos): row[4] = f"{pos:.1f} cm"
                    if not np.isnan(pol): row[5] = f"{pol:.0f} %"
                    d.close()
                except: pass
                
            if gpv_it_str:
                gpv_ft = ft
                if gpv_model == "GSM_JP" and ft > 132:
                    gpv_ft = ft - (ft % 6) if ft % 6 != 0 else ft
                gpv_file = os.path.join(self.cache_dir, f"{gpv_model}_{gpv_it_str}_FT{gpv_ft:02d}.npz")
                if os.path.exists(gpv_file):
                    try:
                        d = np.load(gpv_file)
                        t = self.extract_val_robust(d, ['t2m', 'tmp2m', 'temp2m'], stat_lon, stat_lat)
                        if not np.isnan(t): row[6] = f"{(t - 273.15 if t > 100 else t):.1f}"
                        u = self.extract_val_robust(d, ['u10', '10u', 'u_10m'], stat_lon, stat_lat)
                        v = self.extract_val_robust(d, ['v10', '10v', 'v_10m'], stat_lon, stat_lat)
                        if not np.isnan(u) and not np.isnan(v):
                            row[8] = f"{np.hypot(u, v):.1f}"
                            deg = (np.degrees(np.arctan2(u, v)) + 180) % 360
                            dirs = ["北", "北北東", "北東", "東北東", "東", "東南東", "南東", "南南東", "南", "南南西", "南西", "西南西", "西", "西北西", "北西", "北北西", "北"]
                            row[7] = dirs[int(round(deg / 22.5)) % 16]
                        d.close()
                    except: pass
                    
            table_data.append(row)
            
        table = ax.table(cellText=table_data, colLabels=headers, loc='center', cellLoc='center')
        table.auto_set_font_size(False)
        table.set_fontsize(10)
        table.scale(1, 1.4)
        
        for j in range(len(headers)):
            cell = table[0, j]
            cell.set_facecolor('#1D3557')
            cell.get_text().set_color('white')
            cell.get_text().set_weight('bold')
            
        fig.suptitle(f"{stat_name} 統合ガイダンス時系列表 (初期時: {dt_utc.strftime('%m/%d %H:00Z')})", fontsize=15, weight='bold', y=0.96)
        
        canvas = FigureCanvas(fig)
        win = SingleImageWindow(self, f"GUID_{stat_name}", model_name, stat_name, dt_utc.strftime("%m%d%H") + "Z", canvas)
        win.exec()

    def plot_contour(self, ax, lon, lat, data, ui, levels, color, extent, linewidths=1.5):
        lon_c, lat_c, data_c = self.crop_data(lon, lat, data, extent)
        if lon_c.size == 0 or lat_c.size == 0 or data_c.size == 0: return None
                
        cs = ax.contour(lon_c, lat_c, data_c, levels=levels, colors=color, linewidths=linewidths, transform=ccrs.PlateCarree(), zorder=7)
        ui['dynamic_artists'].append(cs); ax.clabel(cs, inline=True, fontsize=11, fmt='%1.0f'); return cs

    def plot_contourf(self, ax, lon, lat, data, ui, levels, cmap_or_colors, extent, extend='both', alpha=0.6, norm=None):
        lon_c, lat_c, data_c = self.crop_data(lon, lat, data, extent)
        if lon_c.size == 0 or lat_c.size == 0 or data_c.size == 0: return None
        
        kwargs = {'levels': levels, 'extend': extend, 'transform': ccrs.PlateCarree(), 'alpha': alpha, 'zorder': 6}
        if isinstance(cmap_or_colors, str) or isinstance(cmap_or_colors, mcolors.Colormap): kwargs['cmap'] = cmap_or_colors
        else: kwargs['colors'] = cmap_or_colors
        if norm is not None: kwargs['norm'] = norm
            
        cf = ax.contourf(lon_c, lat_c, data_c, **kwargs)
        ui['dynamic_artists'].append(cf)
        return cf

    def plot_blocky(self, ax, lon, lat, data, ui, cmap_or_colors, extent, norm=None, alpha=0.85):
        lon_c, lat_c, data_c = self.crop_data(lon, lat, data, extent)
        if lon_c.size == 0 or lat_c.size == 0 or data_c.size == 0: return None
        
        if lon_c.ndim == 1:
            lon_c, lat_c = np.meshgrid(lon_c, lat_c)
            
        kwargs = {'transform': ccrs.PlateCarree(), 'alpha': alpha, 'zorder': 6, 'shading': 'nearest'}
        if isinstance(cmap_or_colors, str) or isinstance(cmap_or_colors, mcolors.Colormap): kwargs['cmap'] = cmap_or_colors
        else: kwargs['cmap'] = mcolors.ListedColormap(cmap_or_colors)
        if norm is not None: kwargs['norm'] = norm
        
        pm = ax.pcolormesh(lon_c, lat_c, data_c, **kwargs)
        ui['dynamic_artists'].append(pm)
        return pm

    def plot_barbs(self, ax, lon, lat, u, v, ui, area_index, extent, model_name=""):
        lon_c, lat_c, u_c = self.crop_data(lon, lat, u, extent)
        _, _, v_c = self.crop_data(lon, lat, v, extent)
        if lon_c.size == 0 or lat_c.size == 0 or u_c.size == 0 or v_c.size == 0: return None
        u_c = np.nan_to_num(u_c, nan=0.0); v_c = np.nan_to_num(v_c, nan=0.0)
        
        if lon_c.ndim == 1: lon_2d, lat_2d = np.meshgrid(lon_c, lat_c)
        else: lon_2d, lat_2d = lon_c, lat_c
            
        if lon_2d.shape != u_c.shape: return None
        
        if area_index == 0: skip = 20
        elif area_index == 3: skip = 1
        elif area_index == 2: skip = 1
        elif "MSM" in model_name or "ANAL" in model_name: skip = 4
        else: skip = 3
            
        q = ax.barbs(lon_2d[::skip,::skip], lat_2d[::skip,::skip], u_c[::skip,::skip], v_c[::skip,::skip], length=5.5, color='black', transform=ccrs.PlateCarree(), zorder=10)
        ui['dynamic_artists'].append(q)

    def show_error_msg(self, ax, ui, msg):
        wrapped_msg = "\n".join(textwrap.wrap(msg, width=50))
        txt = ax.text(0.5, 0.5, wrapped_msg, transform=ax.transAxes, color='red', fontsize=12, ha='center', va='center', weight='bold', bbox=dict(facecolor='white', alpha=0.9, edgecolor='red'))
        ui['dynamic_artists'].append(txt)

    def draw_guidance(self, ax, ui, data, elem_name, val, extent, area_index, model_name, it_str):
        drawn_flag = False; info_text = ""
        def _cl(arr, minv, maxv, fill=0.0):
            if arr is None: return None
            a = np.array(arr, dtype=float)
            a = np.nan_to_num(a, nan=fill, posinf=fill, neginf=fill)
            return np.clip(a, minv, maxv)
        
        def add_vertical_colorbar(mappable, label_text):
            cax = ui['fig'].add_axes([0.89, 0.15, 0.02, 0.7]) 
            cb = ui['fig'].colorbar(mappable, cax=cax, orientation='vertical')
            cb.set_label(label_text, fontsize=12, weight='bold'); ui['colorbars'].append(cb)

        if "降水量" in elem_name or "降雪量" in elem_name:
            accum_hours = 3
            if "6時間" in elem_name: accum_hours = 6
            elif "24時間" in elem_name: accum_hours = 24
            
            accum_val = None; valid_data = False; lon_c = None; lat_c = None
            
            for hr_offset in range(0, accum_hours, 3):
                target_ft = val - hr_offset
                if target_ft <= 0: continue
                
                d_curr = None
                if target_ft == val: d_curr = data
                else:
                    target_file = os.path.join(self.cache_dir, f"{model_name}_{it_str}_FT{target_ft:02d}.npz")
                    if os.path.exists(target_file):
                        try: d_curr = np.load(target_file)
                        except: pass
                
                if d_curr is not None:
                    if "降水量" in elem_name: key = self.get_exact_key(d_curr, ['precip', 'tp', 'apcp', 'pop', 'var_0_1_8', 'var_0_1_52'])
                    else: key = self.get_exact_key(d_curr, ['snow', 'weasd', 'snod', 'var_0_1_11', 'var_0_1_13', 'var_0_1_29', 'asnow'])   
                    if key:
                        v = _cl(d_curr[key], 0, 1000, 0.0)
                        if accum_val is None:
                            accum_val = v.copy() 
                            lon_c, lat_c = self.get_coords_for_data(d_curr, v)
                            valid_data = True
                        else:
                            accum_val += v
                            
                    if target_ft != val: d_curr.close()
                
            if valid_data and accum_val is not None:
                accum_val_masked = np.ma.masked_less(accum_val, 0.1)
                levels = [0.1, 1, 5, 10, 20, 30, 50, 80]
                
                if "降水量" in elem_name:
                    cmap = mcolors.ListedColormap(['#A0E8A0', '#D4F05A', '#FFFF00', '#FFA500', '#FF5500', '#FF0000', '#FF00FF'])
                    cmap.set_under('none'); cmap.set_over('#800080')
                    label_text = f'降水量 ({accum_hours}時間積算) [mm]'
                else:
                    cmap = mcolors.ListedColormap(['#E8EAF6', '#C5CAE9', '#9FA8DA', '#7986CB', '#5C6BC0', '#3F51B5', '#303F9F'])
                    cmap.set_under('none'); cmap.set_over('#1A237E')
                    label_text = f'降雪量 ({accum_hours}時間積算) [cm等]'
                
                norm = mcolors.BoundaryNorm(levels, cmap.N)
                pm = self.plot_blocky(ax, lon_c, lat_c, accum_val_masked, ui, cmap, extent, norm=norm, alpha=0.45)
                
                if pm:
                    add_vertical_colorbar(pm, label_text)
                    stat_lon = 141.35; stat_lat = 43.06
                    
                    if lon_c.ndim == 1: lon_2d, lat_2d = np.meshgrid(lon_c, lat_c)
                    else: lon_2d, lat_2d = lon_c, lat_c
                    dist = (lon_2d - stat_lon)**2 + (lat_2d - stat_lat)**2
                    sy, sx = np.unravel_index(np.argmin(dist), dist.shape)
                    local_val = accum_val[sy, sx] if (sy < accum_val.shape[0] and sx < accum_val.shape[1]) else np.nan
                    
                    info_text = f"[札幌現況]\n降水量({accum_hours}h): {local_val:.1f} mm" if "降水" in elem_name else f"[札幌現況]\n降雪量({accum_hours}h): {local_val:.1f} cm"
                    drawn_flag = True
                    
                    if area_index == 3:
                        lc, lac, dc = self.crop_data(lon_c, lat_c, accum_val, extent)
                        if lc.size > 0:
                            l2, la2 = np.meshgrid(lc, lac) if lc.ndim == 1 else (lc, lac)
                            min_lon, max_lon, min_lat, max_lat = extent 
                            
                            try:
                                import scipy.ndimage as ndimage
                                import matplotlib.patheffects as pe
                                
                                zoom_factor = 8 if "GSM" in model_name else 2
                                
                                dc_z = ndimage.zoom(dc, zoom_factor, order=1)
                                if lc.ndim == 1:
                                    lc_z = ndimage.zoom(lc, zoom_factor, order=1)
                                    lac_z = ndimage.zoom(lac, zoom_factor, order=1)
                                    l2_z, la2_z = np.meshgrid(lc_z, lac_z)
                                else:
                                    l2_z = ndimage.zoom(l2, zoom_factor, order=1)
                                    la2_z = ndimage.zoom(la2, zoom_factor, order=1)
                                
                                for r in range(dc_z.shape[0]):
                                    for c in range(dc_z.shape[1]):
                                        lon_v, lat_v, val_p = l2_z[r, c], la2_z[r, c], dc_z[r, c]
                                        if r % zoom_factor == 0 and c % zoom_factor == 0: continue
                                        
                                        if val_p >= 0.1 and min_lon <= lon_v <= max_lon and min_lat <= lat_v <= max_lat:
                                            txt = ax.text(lon_v, lat_v, f"{val_p:.1f}", color='#1f77b4', fontsize=10, ha='center', va='center', transform=ccrs.PlateCarree(), zorder=12, weight='bold', clip_on=True, path_effects=[pe.withStroke(linewidth=2, foreground='white')])
                                            ui['dynamic_artists'].append(txt)
                                            
                                for r in range(dc.shape[0]):
                                    for c in range(dc.shape[1]):
                                        lon_v, lat_v, val_p = l2[r, c], la2[r, c], dc[r, c]
                                        if val_p >= 0.1 and min_lon <= lon_v <= max_lon and min_lat <= lat_v <= max_lat:
                                            txt = ax.text(lon_v, lat_v, f"{val_p:.1f}", color='black', fontsize=11, ha='center', va='center', transform=ccrs.PlateCarree(), zorder=13, weight='bold', bbox=dict(facecolor='white', edgecolor='black', boxstyle='square,pad=0.2', alpha=0.9), clip_on=True)
                                            ui['dynamic_artists'].append(txt)
                            except Exception as e:
                                print(f"内挿描画エラー: {e}")
        elif "発雷" in elem_name:
            key = None
            for k in data.files:
                if any(x in k.lower() for x in ['thund', 'pol', 'lig', 'ts', 'tstm', 'var_0_19_193']): key = k; break

            if key:
                try:
                    val_data = _cl(data[key], 0, 100, 0.0)
                    lon_c, lat_c = self.get_coords_for_data(data, val_data)
                    
                    if lon_c is None or lat_c is None:
                        lon_k = next((k for k in data.files if 'lon' in k.lower()), None)
                        lat_k = next((k for k in data.files if 'lat' in k.lower()), None)
                        if lon_k and lat_k: lon_c, lat_c = data[lon_k], data[lat_k]

                    if lon_c is not None and lat_c is not None:
                        if lon_c.ndim == 1 and val_data.shape == (len(lon_c), len(lat_c)):
                            val_data = val_data.T
                            
                        if lon_c.ndim == 1 and val_data.shape != (len(lat_c), len(lon_c)):
                            if val_data.shape[1] > val_data.shape[0] and len(lat_c) > len(lon_c):
                                val_data = val_data.T
                            lon_c = np.linspace(lon_c.min(), lon_c.max(), val_data.shape[1])
                            lat_c = np.linspace(lat_c.min(), lat_c.max(), val_data.shape[0])

                    val_masked = np.ma.masked_less(val_data, 1.0)
                    levels = [10, 20, 30, 40, 50, 60, 70, 80, 90, 100]
                    
                    cmap = mcolors.ListedColormap(['#FFF3E0', '#FFE0B2', '#FFCC80', '#FFB74D', '#FFA726', '#FF9800', '#FB8C00', '#F57C00', '#EF6C00'])
                    cmap.set_under('none')
                    norm = mcolors.BoundaryNorm(levels, cmap.N)
                    
                    pm = self.plot_blocky(ax, lon_c, lat_c, val_masked, ui, cmap, extent, norm=norm, alpha=0.6)
                    
                    if pm: 
                        add_vertical_colorbar(pm, '発雷確率 [%]')
                        info_text = self.get_local_info(elem_name, lon_c, lat_c, val_data, area_index=area_index, model_name=model_name)
                        drawn_flag = True
                except Exception as e:
                    print(f"平面地図発雷描画エラー: {e}")
                    
        elif "天気" in elem_name:
            key = None
            for k in data.files:
                if any(x in k.lower() for x in ['wea', 'wx', 'var_-1_-1_-1', 'var_0_19_192']): key = k; break
                
            if key:
                val_data = _cl(data[key], 0, 4, 0.0)
                lon_c, lat_c = self.get_coords_for_data(data, val_data)
                
                if lon_c is None or lat_c is None:
                    lon_k = next((k for k in data.files if 'lon' in k.lower()), None)
                    lat_k = next((k for k in data.files if 'lat' in k.lower()), None)
                    if lon_k and lat_k: lon_c, lat_c = data[lon_k], data[lat_k]

                if lon_c is not None and lat_c is not None:
                    if lon_c.ndim == 1 and val_data.shape == (len(lon_c), len(lat_c)):
                        val_data = val_data.T

                levels = [-0.5, 0.5, 1.5, 2.5, 3.5] 
                colors = ['#FFD700', '#B0C4DE', '#1E90FF', '#FFFFFF']
                cmap_wx = mcolors.ListedColormap(colors); norm_wx = mcolors.BoundaryNorm(levels, cmap_wx.N)
                pm = self.plot_blocky(ax, lon_c, lat_c, val_data, ui, cmap_wx, extent, norm=norm_wx, alpha=0.6)
                if pm: 
                    add_vertical_colorbar(pm, '卓越天気 (0:晴 1:曇 2:雨 3:雪)')
                    info_text = self.get_local_info(elem_name, lon_c, lat_c, val_data, area_index=area_index, model_name=model_name)
                    drawn_flag = True
        
        return drawn_flag, info_text
    
    def extract_2d_array(self, data, exact_keys, target_level=None, fuzzy_keys=None):
        for k in exact_keys:
            if k in data.files:
                arr = np.squeeze(data[k])
                if arr.ndim == 2: return arr
        
        if fuzzy_keys:
            for k in data.files:
                kl = k.lower()
                if any(fk in kl for fk in fuzzy_keys) and not any(ex in kl for ex in ['lon', 'lat', 'time', 'valid']):
                    arr = np.squeeze(data[k])
                    if arr.ndim == 2: return arr
                    if arr.ndim == 3 and target_level is not None:
                        levels_16 = [1000, 925, 850, 700, 500, 400, 300, 250, 200, 150, 100, 70, 50, 30, 20, 10]
                        levels_msm = [1000, 975, 950, 925, 900, 850, 800, 700, 600, 500, 400, 300]
                        axis = 0 if arr.shape[0] < arr.shape[1] else 2
                        sz = arr.shape[axis]
                        if sz == len(levels_16) and target_level in levels_16:
                            return arr[levels_16.index(target_level),:,:] if axis == 0 else arr[:,:,levels_16.index(target_level)]
                        elif sz == len(levels_msm) and target_level in levels_msm:
                            return arr[levels_msm.index(target_level),:,:] if axis == 0 else arr[:,:,levels_msm.index(target_level)]
        return None

    def draw_frame(self, model_name, val):
        ui = self.tab_ui[model_name]; it_str = self.model_init_times.get(model_name)
        if not it_str: 
            self.show_error_msg(ui['ax'] if ui['ax'] else self.init_map(ui['fig'], 0, model_name), ui, f"{model_name} のデータがありません。")
            ui['canvas'].draw_idle(); return

        elem_name = ui['list'].currentItem().text()
        display_val = val
        fallback_msg = ""
        
        # 修正: GSM 132h以降のフォールバックロジックを【全要素(地上・上空とも)】に適用
        if model_name == "GSM_JP" and val > 132 and val % 6 != 0:
            display_val = val - (val % 6)
            fallback_msg = f" (※132h以降は全要素6時間間隔のため FT={display_val}h を代替表示)"
                
        elif model_name == "MSM" and "hpa" in elem_name.lower() and val % 3 != 0:
            display_val = (val // 3) * 3
            fallback_msg = f" (※上空データ3時間間隔のため FT={display_val}h を代替表示)"

        if "GUID" in model_name and "気温" not in elem_name and "風" not in elem_name:
            if val % 3 != 0:
                display_val = (val // 3) * 3
                fallback_msg += f" (※3時間間隔データのため FT={display_val}h を代替表示)"

        dt_utc = datetime.strptime(it_str, "%Y%m%d%H%M%S")
        dt_jst = dt_utc + timedelta(hours=9)
        target_time = dt_jst + timedelta(hours=val)
        ui['time_label'].setText(f"FT={val:02d}h\n({target_time.strftime('%m/%d %H:00')})")
        
        area_index = next(i for i, btn in enumerate(ui['area_btns']) if btn.isChecked())
        
        if ui['ax'] is None or ui['current_area_index'] != area_index:
            ui['ax'] = self.init_map(ui['fig'], area_index, model_name)
            ui['current_area_index'] = area_index
        
        ax = ui['ax']
        extent = self.get_extent(area_index, model_name)
        
        while ui['dynamic_artists']:
            art = ui['dynamic_artists'].pop(0)
            try:
                if hasattr(art, 'collections'):
                    for c in art.collections:
                        try: c.remove()
                        except: pass
                else: art.remove()
            except: pass
            
        for cb in ui['colorbars']:
            try: cb.remove()
            except: pass
        ui['colorbars'].clear(); ui['info_panel'].hide()

        # The code `ax.title.set` appears to be incomplete and does not perform any specific action.
        # It seems like there might be a typo or missing part of the code. If you provide more context
        # or complete the code snippet, I can help you understand its purpose.
        ax.title.set_text(f"[{model_name}] {elem_name} / 初期時刻(UTC): {dt_utc.strftime('%m/%d %H:00Z')} / FT={val}h ({target_time.strftime('%m/%d %H:00')} JST){fallback_msg}")
        
        target_file = os.path.join(self.cache_dir, f"{model_name}_{it_str}_FT{display_val:02d}.npz")
        print(f"DEBUG: 読み込み試行 -> FT={display_val} / {target_file}")
        if not os.path.exists(target_file):
            if "降水" in elem_name and val == 0 and "確率" not in elem_name and "時間降水" not in elem_name and model_name != "ANAL": self.show_error_msg(ax, ui, "【初期時のためデータなし】")
            else: self.show_error_msg(ax, ui, f"データが存在しません (FT={display_val}h)")
            ui['canvas'].draw_idle(); return

        try:
            data = np.load(target_file)
            lon_key = self.get_exact_key(data, ['lon_surf', 'lon', 'lon_p', 'lon_pall'])
            lat_key = self.get_exact_key(data, ['lat_surf', 'lat', 'lat_p', 'lat_pall'])
            
            if not lon_key or not lat_key:
                self.show_error_msg(ax, ui, f"[!] {elem_name} の座標データが見つかりません。(キー: {data.files})")
                data.close(); ui['canvas'].draw_idle(); return

            def add_vertical_colorbar(mappable, label_text):
                cax = ui['fig'].add_axes([0.89, 0.15, 0.02, 0.7]) 
                cb = ui['fig'].colorbar(mappable, cax=cax, orientation='vertical')
                cb.set_label(label_text, fontsize=12, weight='bold'); ui['colorbars'].append(cb)

            def _cl(arr, minv, maxv, fill=0.0):
                if arr is None: return None
                a = np.array(arr, dtype=float)
                a = np.nan_to_num(a, nan=fill, posinf=fill, neginf=fill)
                return np.clip(a, minv, maxv)

            drawn_flag = False; info_text = ""

            if "GUID" in model_name:
                drawn_flag, info_text = self.draw_guidance(ax, ui, data, elem_name, val, extent, area_index, model_name, it_str)

            elif elem_name == "地上気温":
                t_val = self.extract_2d_array(data, ['t2m', 'tmp2m', 'temp2m'], fuzzy_keys=['tmp', 'temp', 't'])
                if t_val is not None:
                    if np.nanmax(t_val) > 100: t_val = t_val - 273.15
                    t_val = _cl(t_val, -50, 40, 0.0)
                    lon_c, lat_c = self.get_coords_for_data(data, t_val)
                    cf = self.plot_contourf(ax, lon_c, lat_c, t_val, ui, np.arange(-30, 40, 3), 'jet', extent, extend='both', alpha=0.4)
                    self.plot_contour(ax, lon_c, lat_c, t_val, ui, np.arange(-30, 40, 3), 'black', extent, linewidths=0.5)
                    if cf:
                        add_vertical_colorbar(cf, '地上気温 [℃]')
                        info_text = self.get_local_info(elem_name, lon_c, lat_c, t_val, area_index=area_index, model_name=model_name); drawn_flag = True

            elif elem_name == "地上気圧":
                slp = self.extract_2d_array(data, ['prmsl', 'mslet', 'msl', 'slp'], fuzzy_keys=['msl', 'prmsl'])
                if slp is not None:
                    if np.nanmax(slp) > 2000: slp = slp / 100.0
                    slp_hpa = _cl(slp, 800, 1150, 1013.0)
                    lon_c, lat_c = self.get_coords_for_data(data, slp_hpa)
                    import scipy.ndimage as ndimage
                    slp_smooth = ndimage.gaussian_filter(slp_hpa, sigma=1.0)
                    levels = np.arange(940, 1060, 4)
                    cs = self.plot_contour(ax, lon_c, lat_c, slp_smooth, ui, levels, '#2980B9', extent)
                    if cs: info_text += self.get_local_info("気圧", lon_c, lat_c, slp, area_index=area_index, model_name=model_name); drawn_flag = True

            elif elem_name == "地上風向風速":
                u_val = self.extract_2d_array(data, ['u10', '10u', 'u_10m'], fuzzy_keys=['u10', 'u_wind'])
                v_val = self.extract_2d_array(data, ['v10', '10v', 'v_10m'], fuzzy_keys=['v10', 'v_wind'])
                if u_val is not None and v_val is not None:
                    u_val = _cl(u_val, -200, 200, 0.0); v_val = _cl(v_val, -200, 200, 0.0)
                    spd = np.hypot(u_val, v_val); lon_c, lat_c = self.get_coords_for_data(data, spd)
                    levels = [2, 4, 6, 8, 10, 15, 20, 25]; colors = ['#E0F7FA', '#B2EBF2', '#4DD0E1', '#00BCD4', '#00838F', '#9C27B0', '#4A148C']
                    cf = self.plot_contourf(ax, lon_c, lat_c, spd, ui, levels, colors, extent, extend='max', alpha=0.5)
                    if cf:
                        add_vertical_colorbar(cf, '地上風速 [m/s]')
                        self.plot_barbs(ax, lon_c, lat_c, u_val, v_val, ui, area_index, extent, model_name)
                        info_text = self.get_local_info("風向風速", lon_c, lat_c, spd, u_val, v_val, area_index=area_index, model_name=model_name); drawn_flag = True
                
            elif "地上降水" in elem_name:
                p_key = self.get_precip_key(data)
                if p_key:
                    pval = _cl(data[p_key], 0, 1000, 0.0)
                    # 修正: GSM 132h以降の降水量差分計算 (引くファイルを6時間前へ変更し、valではなくdisplay_valを使用)
                    if model_name == "GSM_JP" and display_val >= 3:
                        offset = 6 if display_val > 132 else 3
                        prev_file = os.path.join(self.cache_dir, f"{model_name}_{it_str}_FT{display_val-offset:02d}.npz")
                        if os.path.exists(prev_file):
                            try:
                                with np.load(prev_file) as d_prev:
                                    prev_p_key = self.get_precip_key(d_prev)
                                    if prev_p_key:
                                        prev_pval = _cl(d_prev[prev_p_key], 0, 1000, 0.0)
                                        if prev_pval.shape == pval.shape: pval = np.clip(pval - prev_pval, 0, None)
                            except: pass

                    lon_c, lat_c = self.get_coords_for_data(data, pval)
                    if pval.ndim == 1 and lon_c.ndim == 1 and lat_c.ndim == 1 and pval.size == len(lat_c) * len(lon_c):
                        pval = pval.reshape((len(lat_c), len(lon_c)))
                        
                    import scipy.ndimage as ndimage
                    pval_smooth = ndimage.gaussian_filter(pval, sigma=0.2)
                    pval_smooth = np.ma.masked_less(pval_smooth, 0.1) 
                    
                    levels = [0.1, 1, 5, 10, 20, 30, 50, 80]
                    cmap_precip = mcolors.ListedColormap(['#A0E8A0', '#D4F05A', '#FFFF00', '#FFA500', '#FF5500', '#FF0000', '#FF00FF'])
                    cmap_precip.set_under('none'); cmap_precip.set_over('#800080')
                    norm_precip = mcolors.BoundaryNorm(levels, cmap_precip.N)
                    
                    cf = self.plot_contourf(ax, lon_c, lat_c, pval_smooth, ui, levels, cmap_precip, extent, extend='max', alpha=0.85, norm=norm_precip)
                    if cf:
                        self.plot_contour(ax, lon_c, lat_c, pval_smooth, ui, levels, '#444444', extent, linewidths=0.5)
                        add_vertical_colorbar(cf, f'降水量 [mm]')
                        info_text = self.get_local_info("降水", lon_c, lat_c, pval, area_index=area_index, model_name=model_name); drawn_flag = True
                else:
                    if display_val != 0 or model_name == "ANAL":
                        self.show_error_msg(ax, ui, f"[!] 降水キー不明。\nデータが未収録の可能性があります。")
                        drawn_flag = True

            elif "雲量" in elem_name:
                c_val = None
                if elem_name == "全雲量": c_val = self.extract_2d_array(data, ['tcc', 'tcdc', 'lcdc', 'cloud', 'hcc', 'mcc', 'lcc'], fuzzy_keys=['tcc'])
                elif elem_name == "上層雲量": c_val = self.extract_2d_array(data, ['hcc', 'hcdc', 'high', 'tcc'], fuzzy_keys=['hcc', 'high'])
                elif elem_name == "中層雲量": c_val = self.extract_2d_array(data, ['mcc', 'mcdc', 'mid', 'tcc'], fuzzy_keys=['mcc', 'mid'])
                elif elem_name == "下層雲量": c_val = self.extract_2d_array(data, ['lcc', 'lcdc', 'low', 'tcc'], fuzzy_keys=['lcc', 'low'])

                if c_val is not None:
                    c_val = _cl(c_val, 0, 150, 0.0)
                    lon_c, lat_c = self.get_coords_for_data(data, c_val)
                    if np.nanmax(c_val) <= 1.0 and np.nanmax(c_val) > 0: c_val = c_val * 100.0
                    
                    levels = [20, 30, 40, 50, 60, 70, 80, 90, 100]
                    cmap_cloud = mcolors.ListedColormap(['#D9D9D9', '#BDBDBD', '#969696', '#737373', '#525252', '#252525', '#0F0F0F', '#000000'])
                    cmap_cloud.set_under('none'); cmap_cloud.set_over('#000000'); norm_cloud = mcolors.BoundaryNorm(levels, cmap_cloud.N)
                    
                    cf = self.plot_contourf(ax, lon_c, lat_c, c_val, ui, levels, cmap_cloud, extent, extend='max', alpha=0.7, norm=norm_cloud)
                    if cf:
                        add_vertical_colorbar(cf, f'{elem_name} [%]'); info_text = self.get_local_info(elem_name, lon_c, lat_c, c_val, area_index=area_index, model_name=model_name); drawn_flag = True
            
            elif "300hpa高度" in elem_name:
                h_val = self.extract_2d_array(data, ['gh300', 'hgt300', 'hgt_300'], 300, ['hgt', 'gh', 'z'])
                if h_val is not None:
                    h_val = _cl(h_val, -500, 20000, 0.0); lon_p, lat_p = self.get_coords_for_data(data, h_val)
                    levels = np.arange(8400, 10800, 60); cf = self.plot_contourf(ax, lon_p, lat_p, h_val, ui, levels, 'PuBu', extent, alpha=0.6)
                    self.plot_contour(ax, lon_p, lat_p, h_val, ui, levels, 'darkblue', extent)
                    if cf: add_vertical_colorbar(cf, '300hPa 高度 [m]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, h_val, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "300hpa風向・風速" in elem_name:
                u_val = self.extract_2d_array(data, ['u300', 'u_300'], 300, ['u', 'ugrd'])
                v_val = self.extract_2d_array(data, ['v300', 'v_300'], 300, ['v', 'vgrd'])
                if u_val is not None and v_val is not None:
                    u_val = _cl(u_val, -300, 300, 0.0); v_val = _cl(v_val, -300, 300, 0.0)
                    spd = np.hypot(u_val, v_val); lon_p, lat_p = self.get_coords_for_data(data, spd)
                    levels = [20, 30, 40, 50, 60, 80, 100]
                    cf = self.plot_contourf(ax, lon_p, lat_p, spd, ui, levels, 'YlOrRd', extent, extend='max', alpha=0.35)
                    self.plot_barbs(ax, lon_p, lat_p, u_val, v_val, ui, area_index, extent, model_name)
                    if cf: add_vertical_colorbar(cf, '300hPa 風速 [m/s]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, spd, u_val, v_val, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "500hpa高度" in elem_name or "500hPa高度" in elem_name:
                h_val = self.extract_2d_array(data, ['gh500', 'hgt500', 'hgt_500'], 500, ['hgt', 'gh', 'z'])
                if h_val is not None:
                    h_val = _cl(h_val, -500, 20000, 0.0); lon_p, lat_p = self.get_coords_for_data(data, h_val)
                    levels = np.arange(4800, 6000, 60); cf = self.plot_contourf(ax, lon_p, lat_p, h_val, ui, levels, 'Greens', extent, alpha=0.6)
                    self.plot_contour(ax, lon_p, lat_p, h_val, ui, levels, 'darkgreen', extent)
                    if cf: add_vertical_colorbar(cf, '500hPa 高度 [m]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, h_val, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "500hpa風向・風速" in elem_name:
                u_val = self.extract_2d_array(data, ['u500', 'u_500'], 500, ['u', 'ugrd'])
                v_val = self.extract_2d_array(data, ['v500', 'v_500'], 500, ['v', 'vgrd'])
                if u_val is not None and v_val is not None: 
                    u_val = _cl(u_val, -300, 300, 0.0); v_val = _cl(v_val, -300, 300, 0.0)
                    spd = np.hypot(u_val, v_val); lon_p, lat_p = self.get_coords_for_data(data, spd)
                    levels = [10, 20, 30, 40, 50, 60, 80]
                    cf = self.plot_contourf(ax, lon_p, lat_p, spd, ui, levels, 'YlOrRd', extent, extend='max', alpha=0.35)
                    self.plot_barbs(ax, lon_p, lat_p, u_val, v_val, ui, area_index, extent, model_name)
                    if cf: add_vertical_colorbar(cf, '500hPa 風速 [m/s]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, spd, u_val, v_val, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "500hpa渦度" in elem_name:
                vort = self.extract_2d_array(data, ['vort500', 'vort', 'abs_vort'], 500, ['vort', 'abs_vort'])
                if vort is not None:
                    vort = _cl(vort, -1000, 1000, 0.0); lon_p, lat_p = self.get_coords_for_data(data, vort)
                    levels = [-50, -20, 0, 20, 40, 60, 80, 100, 150, 200]
                    colors = ['#ffffff', '#fdf0e6', '#fad7a1', '#f8c471', '#f39c12', '#d35400', '#c0392b', '#922b21', '#641e16']
                    cf = self.plot_contourf(ax, lon_p, lat_p, vort, ui, levels, colors, extent, extend='both', alpha=0.5)
                    self.plot_contour(ax, lon_p, lat_p, vort, ui, levels, '#4A235A', extent)
                    if cf: add_vertical_colorbar(cf, '500hPa 正渦度 [10^-5/s]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, vort, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "hpa気温" in elem_name.lower():
                w = re.search(r'(\d+)hpa', elem_name.lower())
                if w:
                    lvl = int(w.group(1))
                    t_val = self.extract_2d_array(data, [f't{lvl}', f'tmp{lvl}', f't_{lvl}', f'tmp_{lvl}'], lvl, ['tmp', 'temp', 't'])
                    if t_val is not None:
                        if np.nanmax(t_val) > 100: t_val = t_val - 273.15
                        t_val = _cl(t_val, -100, 60, 0.0); lon_p, lat_p = self.get_coords_for_data(data, t_val)
                        cf = self.plot_contourf(ax, lon_p, lat_p, t_val, ui, np.arange(-54, 40, 3), 'jet', extent, extend='both', alpha=0.3)
                        self.plot_contour(ax, lon_p, lat_p, t_val, ui, np.arange(-54, 40, 3), 'blue', extent, linewidths=0.5)
                        if cf:
                            add_vertical_colorbar(cf, f'{lvl}hPa 気温 [℃]')
                            lon_c, lat_c, t_c = self.crop_data(lon_p, lat_p, t_val, extent)
                            if lon_c.size > 0:
                                cs = ax.contour(lon_c if lon_c.ndim==1 else lon_c, 
                                                lat_c if lat_c.ndim==1 else lat_c, 
                                                t_c, levels=[-36, -6, 0, 15, 30], colors='red', linewidths=2.0, transform=ccrs.PlateCarree(), zorder=8)
                                ui['dynamic_artists'].append(cs); ax.clabel(cs, inline=True, fontsize=12, fmt='%1.0f')
                            
                        u_val = self.extract_2d_array(data, [f'u{lvl}', f'u_{lvl}'], lvl, ['u', 'ugrd'])
                        v_val = self.extract_2d_array(data, [f'v{lvl}', f'v_{lvl}'], lvl, ['v', 'vgrd'])
                        if u_val is not None: u_val = _cl(u_val, -200, 200, 0.0)
                        if v_val is not None: v_val = _cl(v_val, -200, 200, 0.0)
                        info_text = self.get_local_info(elem_name, lon_p, lat_p, t_val, u_val, v_val, area_index=area_index, model_name=model_name)
                        drawn_flag = True
                        if u_val is not None and v_val is not None: self.plot_barbs(ax, lon_p, lat_p, u_val, v_val, ui, area_index, extent, model_name)

            elif "700hpa湿数" in elem_name:
                t = self.extract_2d_array(data, ['t700', 'tmp700', 't_700'], 700, ['tmp', 'temp', 't'])
                rh_raw = self.extract_2d_array(data, ['rh700', 'r700', 'rh_700'], 700, ['rh', 'hum', 'r'])
                if t is not None and rh_raw is not None:
                    if np.nanmax(t) > 100: t = t - 273.15
                    rh = _cl(rh_raw, 0, 150, 0.0)
                    if np.nanmax(rh) <= 1.0 and np.nanmax(rh) > 0: rh = rh * 100.0
                    lon_p, lat_p = self.get_coords_for_data(data, t); t_td = calc_t_td(t, rh)
                    levels = [0, 3, 6, 9, 12, 15]; colors = ['#27AE60', '#82E0AA', '#D5F5E3', '#F9E79F', '#F5CBA7', '#E74C3C']
                    cf = self.plot_contourf(ax, lon_p, lat_p, t_td, ui, levels, colors, extent, extend='max', alpha=0.35) 
                    u_val = self.extract_2d_array(data, ['u700', 'u_700'], 700, ['u', 'ugrd'])
                    v_val = self.extract_2d_array(data, ['v700', 'v_700'], 700, ['v', 'vgrd'])
                    if u_val is not None: u_val = _cl(u_val, -200, 200, 0.0)
                    if v_val is not None: v_val = _cl(v_val, -200, 200, 0.0)
                    if cf:
                        add_vertical_colorbar(cf, '700hPa 湿数 [℃]')
                        info_text = self.get_local_info("湿数", lon_p, lat_p, t_td, u_val, v_val, area_index=area_index, model_name=model_name); drawn_flag = True
                        if u_val is not None and v_val is not None: self.plot_barbs(ax, lon_p, lat_p, u_val, v_val, ui, area_index, extent, model_name)

            elif "700hpa鉛直流" in elem_name:
                w_val = self.extract_2d_array(data, ['w', 'w700', 'vvel', 'vvel700', 'dzdt', 'dzdt700'], 700, ['w', 'vvel', 'dzdt'])
                if w_val is not None:
                    if np.nanmax(np.abs(w_val)) < 15.0: w_val = w_val * 36.0 
                    w_val = _cl(w_val, -500, 500, 0.0); lon_p, lat_p = self.get_coords_for_data(data, w_val)
                    import scipy.ndimage as ndimage
                    w_val = ndimage.gaussian_filter(w_val, sigma=0.8)
                    w_val_masked = np.ma.masked_inside(w_val, -4.9, 4.9)
                    levels = [-50, -30, -20, -10, -5, 5, 10, 20, 30, 50]
                    cf = self.plot_contourf(ax, lon_p, lat_p, w_val_masked, ui, levels, 'PiYG', extent, extend='both', alpha=0.8)
                    if cf: add_vertical_colorbar(cf, '700hPa 鉛直流 [hPa/h]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, w_val, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "850hpa相当温位" in elem_name:
                t = self.extract_2d_array(data, ['t850', 'tmp850', 't_850'], 850, ['tmp', 'temp', 't'])
                rh_raw = self.extract_2d_array(data, ['rh850', 'r850', 'rh_850'], 850, ['rh', 'hum', 'r'])
                if t is not None and rh_raw is not None:
                    if np.nanmax(t) > 100: t = t - 273.15
                    rh = _cl(rh_raw, 0, 150, 0.0)
                    if np.nanmax(rh) <= 1.0 and np.nanmax(rh) > 0: rh = rh * 100.0
                    lon_p, lat_p = self.get_coords_for_data(data, t); ept = calc_ept(t, rh, 850); levels = np.arange(270, 360, 6)
                    cf = self.plot_contourf(ax, lon_p, lat_p, ept, ui, levels, 'rainbow', extent, extend='both', alpha=0.6) 
                    self.plot_contour(ax, lon_p, lat_p, ept, ui, levels, 'black', extent)
                    if cf: add_vertical_colorbar(cf, '850hPa 相当温位 [K]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, ept, area_index=area_index, model_name=model_name); drawn_flag = True

            elif "850hpa湿数" in elem_name:
                t = self.extract_2d_array(data, ['t850', 'tmp850', 't_850'], 850, ['tmp', 'temp', 't'])
                rh_raw = self.extract_2d_array(data, ['rh850', 'r850', 'rh_850'], 850, ['rh', 'hum', 'r'])
                if t is not None and rh_raw is not None:
                    if np.nanmax(t) > 100: t = t - 273.15
                    rh = _cl(rh_raw, 0, 150, 0.0)
                    if np.nanmax(rh) <= 1.0 and np.nanmax(rh) > 0: rh = rh * 100.0
                    lon_p, lat_p = self.get_coords_for_data(data, t); t_td = calc_t_td(t, rh)
                    levels = [0, 3, 6, 9, 12, 15]; colors = ['#27AE60', '#82E0AA', '#D5F5E3', '#F9E79F', '#F5CBA7', '#E74C3C']
                    cf = self.plot_contourf(ax, lon_p, lat_p, t_td, ui, levels, colors, extent, extend='max', alpha=0.35)
                    u_val = self.extract_2d_array(data, ['u850', 'u_850'], 850, ['u', 'ugrd'])
                    v_val = self.extract_2d_array(data, ['v850', 'v_850'], 850, ['v', 'vgrd'])
                    if u_val is not None: u_val = _cl(u_val, -200, 200, 0.0)
                    if v_val is not None: v_val = _cl(v_val, -200, 200, 0.0)
                    if cf:
                        add_vertical_colorbar(cf, '850hPa 湿数 [℃]')
                        info_text = self.get_local_info("湿数", lon_p, lat_p, t_td, u_val, v_val, area_index=area_index, model_name=model_name); drawn_flag = True
                        if u_val is not None and v_val is not None: self.plot_barbs(ax, lon_p, lat_p, u_val, v_val, ui, area_index, extent, model_name)

            elif "925hpa" in elem_name or "975hpa" in elem_name:
                lvl = 925 if "925" in elem_name else 975
                t = self.extract_2d_array(data, [f't{lvl}', f'tmp{lvl}', f't_{lvl}'], lvl, ['tmp', 'temp', 't'])
                rh_raw = self.extract_2d_array(data, [f'rh{lvl}', f'r{lvl}', f'rh_{lvl}'], lvl, ['rh', 'hum', 'r'])
                if t is not None and rh_raw is not None:
                    if np.nanmax(t) > 100: t = t - 273.15
                    rh = _cl(rh_raw, 0, 150, 0.0)
                    if np.nanmax(rh) <= 1.0 and np.nanmax(rh) > 0: rh = rh * 100.0
                    lon_p, lat_p = self.get_coords_for_data(data, t)
                    
                    if "相当温位" in elem_name:
                        ept = calc_ept(t, rh, lvl); levels = np.arange(270, 360, 6)
                        cf = self.plot_contourf(ax, lon_p, lat_p, ept, ui, levels, 'rainbow', extent, extend='both', alpha=0.6)
                        self.plot_contour(ax, lon_p, lat_p, ept, ui, levels, 'black', extent)
                        if cf: add_vertical_colorbar(cf, f'{lvl}hPa 相当温位 [K]'); info_text = self.get_local_info(elem_name, lon_p, lat_p, ept, area_index=area_index, model_name=model_name); drawn_flag = True
                    else:
                        t_td = calc_t_td(t, rh)
                        levels = [0, 3, 6, 9, 12, 15]; colors = ['#27AE60', '#82E0AA', '#D5F5E3', '#F9E79F', '#F5CBA7', '#E74C3C']
                        cf = self.plot_contourf(ax, lon_p, lat_p, t_td, ui, levels, colors, extent, extend='max', alpha=0.35)
                        u_val = self.extract_2d_array(data, [f'u{lvl}', f'u_{lvl}'], lvl, ['u', 'ugrd'])
                        v_val = self.extract_2d_array(data, [f'v{lvl}', f'v_{lvl}'], lvl, ['v', 'vgrd'])
                        if u_val is not None: u_val = _cl(u_val, -200, 200, 0.0)
                        if v_val is not None: v_val = _cl(v_val, -200, 200, 0.0)
                        if cf:
                            add_vertical_colorbar(cf, f'{lvl}hPa 湿数 [℃]')
                            info_text = self.get_local_info("湿数", lon_p, lat_p, t_td, u_val, v_val, area_index=area_index, model_name=model_name); drawn_flag = True
                            if u_val is not None and v_val is not None: self.plot_barbs(ax, lon_p, lat_p, u_val, v_val, ui, area_index, extent, model_name)

            data.close(); del data 
            
            if info_text:
                ui['info_panel'].setText(info_text); ui['info_panel'].adjustSize(); ui['info_panel'].move(10, 10); ui['info_panel'].show()
            if not drawn_flag:
                if "降水" in elem_name and display_val == 0 and "確率" not in elem_name and "時間降水" not in elem_name and model_name != "ANAL": self.show_error_msg(ax, ui, "【初期時のためデータなし。次画像から表示】")
                elif not ("降水" in elem_name and display_val != 0 and "確率" not in elem_name and "時間降水" not in elem_name): 
                    self.show_error_msg(ax, ui, f"[!] {elem_name} のデータがありません。\n保存時の変数名が異なっている可能性があります。")
            
        except Exception as e: logging.error(f"Draw Error for {elem_name} at FT={display_val}: {e}")

        ui['canvas'].draw_idle(); gc.collect()

    def setup_history_tab(self):
        tab = QWidget(); layout = QVBoxLayout(tab); history_text = QTextEdit(); history_text.setReadOnly(True)
        history_text.setStyleSheet("background-color: #112240; color: #E0E0E0; font-family: 'MS Gothic'; font-size: 12pt; padding: 10px; border: none;")
        html_content = """
        <h2 style='color: #64FFDA;'>気象GPV 局地分析ツール 更新履歴</h2><hr style='border-color: #457B9D;'><ul>
        <li><b>Ver 121.0 (GSM 132h以降 完全対応版)</b><br>
        1. <b>132時間以降のGSMフォールバック処理を修正</b>: 上空要素だけでなく、地上のすべての要素に対して、GSMの6時間間隔データへの自動フォールバック(間引き読み込み)を適用。<br>
        2. <b>降水量差分計算のバグ修正</b>: GSMの132時間以降の降水量(積算)の差分を求める際、常に3時間前を引いていたため発生していたデータ消失・スパイクエラーを、動的に6時間前を参照するよう修正。<br>
        3. <b>メタグラムの互換性向上</b>: 132時間以降のGSMメタグラムにおいて、ファイルの読み込み参照を自動で6時間ごとに補正するよう改善。
        </ul>"""
        history_text.setHtml(html_content); layout.addWidget(history_text); self.tabs.addTab(tab, "更新履歴")

    def scan_cache_dir(self, quiet=False, force_draw=False):
        if not os.path.exists(self.cache_dir): return
        files = glob.glob(os.path.join(self.cache_dir, "*.npz"))
        updated = False
        
        for m in ["MSM", "GSM_JP", "ANAL", "MSM_GUID", "GSM_GUID"]:
            m_files = [f for f in files if os.path.basename(f).startswith(f"{m}_")]
            times = sorted(list(set(re.search(r'_(\d{14})_', f).group(1) for f in m_files if re.search(r'_(\d{14})_', f))), reverse=True)
            if times:
                combo = self.tab_ui.get(m, {}).get('init_combo')
                if combo:
                    current_items = [combo.itemData(i) for i in range(combo.count())]
                    if current_items != times:
                        combo.blockSignals(True); combo.clear()
                        for t in times:
                            dt_utc = datetime.strptime(t, "%Y%m%d%H%M%S") # UTC完全準拠
                            dt_jst = dt_utc + timedelta(hours=9)
                            combo.addItem(dt_utc.strftime('%m/%d %H:%M Z') + (" (最新)", "")[t != times[0]], t)
                        combo.blockSignals(False)
                        if self.model_init_times.get(m) not in times:
                            self.model_init_times[m] = times[0]; 
                            self.update_slider_max(m, times[0])
                            self.tab_ui[m]['slider'].setValue(0); combo.setCurrentIndex(0); updated = True
            
        if (updated or force_draw) and not quiet:
            self.status_banner.setText("最新データを画面に同期しました")
            current_tab_name = self.tabs.tabText(self.tabs.currentIndex())
            if "MSM ガイダンス" in current_tab_name: current_m = "MSM_GUID"
            elif "GSM ガイダンス" in current_tab_name: current_m = "GSM_GUID"
            elif "毎時大気" in current_tab_name: current_m = "ANAL"
            elif "MSM" in current_tab_name: current_m = "MSM"
            elif "日本域" in current_tab_name: current_m = "GSM_JP"
            else: current_m = None
            
            if current_m in self.tab_ui: 
                if len(self.tab_ui[current_m]['area_btns']) > 4 and self.tab_ui[current_m]['area_btns'][-2].isChecked(): pass
                else: self.draw_frame(current_m, self.tab_ui[current_m]['slider'].value())
        
    def start_animation(self, model_name):
        self.current_anim_model = model_name
        speeds = [2000, 1000, 500, 200, 50]
        self.anim_timer.start(speeds[self.tab_ui[model_name]['combo_speed'].currentIndex()])

    def stop_animation(self):
        self.anim_timer.stop()

    def anim_step(self):
        if not self.current_anim_model: return
        ui = self.tab_ui[self.current_anim_model]
        new_val = ui['slider'].value() + ui['step']
        ui['slider'].setValue(new_val if new_val <= ui['slider'].maximum() else ui['slider'].minimum())

if __name__ == '__main__':
    import traceback
    try:
        app = QApplication(sys.argv); window = WeatherApp(); sys.exit(app.exec())
    except Exception as e:
        print("起動時にエラーが発生しました:")
        traceback.print_exc()
        input("Enterキーを押すと終了します...")

コメント