#!/usr/bin/env python3
"""
Generate Publication-Ready Figures for New Experiments (E1, E2, E3)

Figures for ICML Paper Improvements:
- Fig E1: Benchmark Sanity Check (CR^* > 1 vs CR^mix < 1)
- Fig E2: Ablation Experiments (component isolation)
- Fig E3: K-Sweep (√K regret scaling)

Style matches existing figures in figures/ directory.

Usage:
    python generate_new_figures.py
"""

import json
import numpy as np
import pandas as pd
import matplotlib.pyplot as plt
from pathlib import Path

# Set publication-quality defaults (matching existing style)
plt.rcParams.update({
    'font.size': 12,
    'axes.labelsize': 14,
    'axes.titlesize': 14,
    'xtick.labelsize': 11,
    'ytick.labelsize': 11,
    'legend.fontsize': 10,
    'figure.dpi': 150,
    'savefig.dpi': 300,
    'savefig.bbox': 'tight',
    'font.family': 'sans-serif',
})

# Color scheme for algorithms
COLORS = {
    'SP-UCB-OLP': '#2ca02c',      # Green (main algorithm)
    'Greedy': '#1f77b4',          # Blue
    'OneHot': '#e377c2',          # Pink
    'Oracle': '#17becf',          # Cyan
    'Random': '#7f7f7f',          # Gray
    # Ablation variants
    'EnvelopeGreedy': '#ff7f0e',   # Orange
    'MixtureLocalPrice': '#d62728', # Red
    'NoSlack': '#9467bd',          # Purple
    'AcceptedOnly': '#8c564b',     # Brown
}

MARKERS = {
    'SP-UCB-OLP': 'o',
    'Greedy': 's',
    'OneHot': '^',
    'Oracle': 'D',
    'Random': 'x',
    'EnvelopeGreedy': 'v',
    'MixtureLocalPrice': '<',
    'NoSlack': '>',
    'AcceptedOnly': 'p',
}


def load_e1_data(results_dir: Path) -> pd.DataFrame:
    """Load E1 benchmark sanity data."""
    json_file = results_dir / "E1_benchmark_sanity.json"
    with open(json_file) as f:
        data = json.load(f)
    return pd.DataFrame(data['results'])


def load_e2_data(results_dir: Path) -> pd.DataFrame:
    """Load E2 ablation data."""
    json_file = results_dir / "E2_ablations.json"
    with open(json_file) as f:
        data = json.load(f)
    return pd.DataFrame(data['results'])


def load_e3_data(results_dir: Path) -> pd.DataFrame:
    """Load E3 K-sweep data."""
    json_file = results_dir / "k_sweep" / "combined_results.json"
    with open(json_file) as f:
        data = json.load(f)
    return pd.DataFrame(data['results'])


# =============================================================================
# FIGURE E1: BENCHMARK SANITY CHECK
# =============================================================================

def generate_fig_e1_benchmark_sanity(df: pd.DataFrame, output_dir: Path):
    """
    Figure E1: Benchmark Sanity Check

    Two-panel figure showing:
    - Left: CR^* (vs V*) - shows values > 1 for switching algorithms
    - Right: CR^mix (vs V^mix) - shows all values ≤ 1

    Key message: V^mix is the correct benchmark, not V*.
    """
    fig, axes = plt.subplots(1, 2, figsize=(12, 5))

    algorithms = ['SP-UCB-OLP', 'Greedy', 'Random', 'OneHot', 'Oracle']
    rho_values = sorted(df['rho'].unique())

    # Left panel: CR^* (can exceed 1)
    ax = axes[0]
    for alg in algorithms:
        alg_df = df[df['algorithm'] == alg]
        grouped = alg_df.groupby('rho')['CR_star'].agg(['mean', 'std'])

        means = grouped['mean'].values
        stds = grouped['std'].values

        ax.errorbar(rho_values, means, yerr=stds,
                   marker=MARKERS.get(alg, 'o'),
                   color=COLORS.get(alg, '#333'),
                   label=alg, linewidth=2, markersize=8, capsize=3)

    ax.axhline(y=1.0, color='red', linestyle='--', linewidth=2, label='CR = 1')
    ax.set_xlabel('Budget Tightness (ρ)')
    ax.set_ylabel('CR* (vs Fixed Oracle V*)')
    ax.set_title('(a) Fixed-Oracle Competitive Ratio', fontweight='bold')
    ax.set_ylim(0.5, 2.2)
    ax.legend(loc='upper right', fontsize=9)
    ax.grid(alpha=0.3)

    # Right panel: CR^mix (all ≤ 1)
    ax = axes[1]
    for alg in algorithms:
        alg_df = df[df['algorithm'] == alg]
        grouped = alg_df.groupby('rho')['CR_mix'].agg(['mean', 'std'])

        means = grouped['mean'].values
        stds = grouped['std'].values

        ax.errorbar(rho_values, means, yerr=stds,
                   marker=MARKERS.get(alg, 'o'),
                   color=COLORS.get(alg, '#333'),
                   label=alg, linewidth=2, markersize=8, capsize=3)

    ax.axhline(y=1.0, color='red', linestyle='--', linewidth=2, label='CR = 1')
    ax.set_xlabel('Budget Tightness (ρ)')
    ax.set_ylabel('CR^mix (vs Switching Oracle V^mix)')
    ax.set_title('(b) Switching-Aware Competitive Ratio', fontweight='bold')
    ax.set_ylim(0.2, 1.1)
    ax.legend(loc='lower right', fontsize=9)
    ax.grid(alpha=0.3)

    # Add complementarity gap annotation
    mean_gap = df.groupby('rho')['complementarity_gap'].mean().mean()
    fig.text(0.5, 0.02, f'Mean Complementarity Gap (V^mix/V*): {mean_gap:.2f}',
             ha='center', fontsize=12, style='italic')

    plt.tight_layout(rect=[0, 0.05, 1, 1])

    # Save
    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_e1_benchmark_sanity.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


def generate_fig_e1_complementarity_bar(df: pd.DataFrame, output_dir: Path):
    """
    Figure E1b: Bar chart showing CR^* vs CR^mix side by side.

    Clear visual that CR^* > 1 while CR^mix < 1.
    """
    # Focus on ρ=0.7 for clarity
    df_rho = df[df['rho'] == 0.7]

    algorithms = ['SP-UCB-OLP', 'Greedy', 'Random', 'OneHot', 'Oracle']

    fig, ax = plt.subplots(figsize=(10, 6))

    x = np.arange(len(algorithms))
    width = 0.35

    cr_star_means = [df_rho[df_rho['algorithm'] == alg]['CR_star'].mean() for alg in algorithms]
    cr_star_stds = [df_rho[df_rho['algorithm'] == alg]['CR_star'].std() for alg in algorithms]
    cr_mix_means = [df_rho[df_rho['algorithm'] == alg]['CR_mix'].mean() for alg in algorithms]
    cr_mix_stds = [df_rho[df_rho['algorithm'] == alg]['CR_mix'].std() for alg in algorithms]

    bars1 = ax.bar(x - width/2, cr_star_means, width, yerr=cr_star_stds,
                   label='CR* (vs V*)', color='#ff7f0e', capsize=4, alpha=0.8)
    bars2 = ax.bar(x + width/2, cr_mix_means, width, yerr=cr_mix_stds,
                   label='CR^mix (vs V^mix)', color='#2ca02c', capsize=4, alpha=0.8)

    ax.axhline(y=1.0, color='red', linestyle='--', linewidth=2, label='CR = 1')

    ax.set_xlabel('Algorithm')
    ax.set_ylabel('Competitive Ratio')
    ax.set_title('S4 Complementarity Scenario (ρ=0.7): CR* Can Exceed 1', fontweight='bold', fontsize=14)
    ax.set_xticks(x)
    ax.set_xticklabels(algorithms)
    ax.legend(loc='upper right')
    ax.set_ylim(0, 2.2)
    ax.grid(axis='y', alpha=0.3)

    # Add value labels on bars
    for bar, val in zip(bars1, cr_star_means):
        ax.text(bar.get_x() + bar.get_width()/2, bar.get_height() + 0.05,
                f'{val:.2f}', ha='center', va='bottom', fontsize=9)
    for bar, val in zip(bars2, cr_mix_means):
        ax.text(bar.get_x() + bar.get_width()/2, bar.get_height() + 0.02,
                f'{val:.2f}', ha='center', va='bottom', fontsize=9)

    plt.tight_layout()

    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_e1_complementarity_bar.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


# =============================================================================
# FIGURE E2: ABLATION EXPERIMENTS
# =============================================================================

def generate_fig_e2_ablations(df: pd.DataFrame, output_dir: Path):
    """
    Figure E2: Ablation Experiments

    Two-panel figure showing ablation results on S3 and S4:
    - Left: S3 (selective admission) - shows AcceptedOnly failure
    - Right: S4 (complementarity) - shows MixtureLocalPrice failure

    Key message: Both mixture sampling and global prices are essential.
    """
    fig, axes = plt.subplots(1, 2, figsize=(12, 5))

    algorithms = ['SP-UCB-OLP', 'EnvelopeGreedy', 'MixtureLocalPrice', 'NoSlack', 'AcceptedOnly']
    rho_values = [0.5, 0.7, 1.0]

    for ax, scenario in zip(axes, ['S3', 'S4']):
        scenario_df = df[df['scenario'] == scenario]

        x = np.arange(len(rho_values))
        width = 0.15

        for i, alg in enumerate(algorithms):
            alg_df = scenario_df[scenario_df['algorithm'] == alg]
            means = [alg_df[alg_df['rho'] == rho]['CR_mix'].mean() for rho in rho_values]
            stds = [alg_df[alg_df['rho'] == rho]['CR_mix'].std() for rho in rho_values]

            offset = (i - len(algorithms)/2 + 0.5) * width
            bars = ax.bar(x + offset, means, width, yerr=stds,
                         label=alg, color=COLORS.get(alg, '#333'),
                         capsize=2, alpha=0.8)

        ax.set_xlabel('Budget Tightness (ρ)')
        ax.set_ylabel('CR^mix' if ax == axes[0] else '')
        ax.set_title(f'({chr(ord("a") + list(axes).index(ax))}) {scenario}', fontweight='bold')
        ax.set_xticks(x)
        ax.set_xticklabels([f'ρ={r}' for r in rho_values])
        ax.set_ylim(0.4, 1.05)
        ax.axhline(y=1.0, color='gray', linestyle='--', alpha=0.5)
        ax.grid(axis='y', alpha=0.3)

        if ax == axes[1]:
            ax.legend(loc='lower right', fontsize=8)

    # Add key findings annotations
    axes[0].annotate('AcceptedOnly fails\n(selection bias)',
                    xy=(1, 0.66), xytext=(1.5, 0.55),
                    fontsize=9, ha='center',
                    arrowprops=dict(arrowstyle='->', color='red', lw=1.5))
    axes[1].annotate('MixtureLocalPrice fails\n(wrong dual price)',
                    xy=(1, 0.50), xytext=(0.5, 0.62),
                    fontsize=9, ha='center',
                    arrowprops=dict(arrowstyle='->', color='red', lw=1.5))

    fig.suptitle('Ablation Experiments: Component Necessity', fontsize=16, fontweight='bold', y=1.02)
    plt.tight_layout()

    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_e2_ablations.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


def generate_fig_e2_ablation_heatmap(df: pd.DataFrame, output_dir: Path):
    """
    Figure E2b: Heatmap showing CR^mix for each (algorithm, scenario, rho) combination.

    Compact visualization of all ablation results.
    """
    # Pivot to create heatmap data
    pivot = df.pivot_table(
        values='CR_mix',
        index='algorithm',
        columns=['scenario', 'rho'],
        aggfunc='mean'
    )

    fig, ax = plt.subplots(figsize=(10, 6))

    im = ax.imshow(pivot.values, cmap='RdYlGn', vmin=0.4, vmax=1.0, aspect='auto')

    # Add colorbar
    cbar = plt.colorbar(im, ax=ax)
    cbar.set_label('CR^mix', fontsize=12)

    # Set ticks
    ax.set_xticks(np.arange(len(pivot.columns)))
    ax.set_yticks(np.arange(len(pivot.index)))
    ax.set_xticklabels([f'{s}\nρ={r}' for s, r in pivot.columns], fontsize=9)
    ax.set_yticklabels(pivot.index, fontsize=10)

    # Add value annotations
    for i in range(len(pivot.index)):
        for j in range(len(pivot.columns)):
            val = pivot.values[i, j]
            color = 'white' if val < 0.6 else 'black'
            ax.text(j, i, f'{val:.2f}', ha='center', va='center', color=color, fontsize=9)

    ax.set_title('Ablation Results: CR^mix by Algorithm, Scenario, and ρ', fontweight='bold', fontsize=14)
    plt.tight_layout()

    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_e2_ablation_heatmap.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


# =============================================================================
# FIGURE E3: K-SWEEP (√K SCALING)
# =============================================================================

def generate_fig_e3_k_sweep(df: pd.DataFrame, output_dir: Path):
    """
    Figure E3: K-Sweep Regret Scaling

    Two-panel figure:
    - Left: Regret/√T vs K (log-log) - should show ~√K slope
    - Right: CR^mix vs K - shows performance degradation with K

    Key message: Regret scales as Õ(√KT) as predicted by theory.
    """
    fig, axes = plt.subplots(1, 2, figsize=(12, 5))

    algorithms = ['SP-UCB-OLP', 'Greedy', 'OneHot', 'Oracle', 'Random']
    K_values = sorted(df['K'].unique())

    # Left panel: Regret/√T vs K (log-log)
    ax = axes[0]
    for alg in algorithms:
        alg_df = df[df['algorithm'] == alg]
        grouped = alg_df.groupby('K')['regret_normalized'].agg(['mean', 'std'])

        means = grouped['mean'].values
        stds = grouped['std'].values

        ax.errorbar(K_values, means, yerr=stds,
                   marker=MARKERS.get(alg, 'o'),
                   color=COLORS.get(alg, '#333'),
                   label=alg, linewidth=2, markersize=8, capsize=3)

    # Add √K reference line
    k_ref = np.array([2, 4, 8, 16])
    sqrt_k_ref = 5 * np.sqrt(k_ref)  # Scaled for visibility
    ax.plot(k_ref, sqrt_k_ref, 'k--', linewidth=2, label='∝ √K', alpha=0.7)

    ax.set_xscale('log', base=2)
    ax.set_yscale('log')
    ax.set_xlabel('Number of Configurations (K)')
    ax.set_ylabel('Regret / √T')
    ax.set_title('(a) Regret Scaling with K', fontweight='bold')
    ax.set_xticks(K_values)
    ax.set_xticklabels([str(k) for k in K_values])
    ax.legend(loc='upper left', fontsize=9)
    ax.grid(alpha=0.3, which='both')

    # Right panel: CR^mix vs K
    ax = axes[1]
    for alg in algorithms:
        alg_df = df[df['algorithm'] == alg]
        grouped = alg_df.groupby('K')['competitive_ratio_mix'].agg(['mean', 'std'])

        means = grouped['mean'].values
        stds = grouped['std'].values

        ax.errorbar(K_values, means, yerr=stds,
                   marker=MARKERS.get(alg, 'o'),
                   color=COLORS.get(alg, '#333'),
                   label=alg, linewidth=2, markersize=8, capsize=3)

    ax.axhline(y=1.0, color='gray', linestyle='--', alpha=0.5)
    ax.set_xscale('log', base=2)
    ax.set_xlabel('Number of Configurations (K)')
    ax.set_ylabel('CR^mix')
    ax.set_title('(b) Competitive Ratio vs K', fontweight='bold')
    ax.set_xticks(K_values)
    ax.set_xticklabels([str(k) for k in K_values])
    ax.set_ylim(0.4, 1.05)
    ax.legend(loc='lower left', fontsize=9)
    ax.grid(alpha=0.3)

    fig.suptitle('K-Sweep: Validating √K Regret Scaling', fontsize=16, fontweight='bold', y=1.02)
    plt.tight_layout()

    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_e3_k_sweep.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


def generate_fig_e3_sqrt_k_ratio(df: pd.DataFrame, output_dir: Path):
    """
    Figure E3b: Regret/(√T · √K) ratio plot.

    If theory is correct, this ratio should be approximately constant.
    """
    fig, ax = plt.subplots(figsize=(8, 5))

    algorithms = ['SP-UCB-OLP', 'Greedy', 'OneHot']
    K_values = sorted(df['K'].unique())

    for alg in algorithms:
        alg_df = df[df['algorithm'] == alg]
        grouped = alg_df.groupby('K')['regret_normalized'].mean()

        # Compute ratio: (Regret/√T) / √K
        ratios = [grouped[k] / np.sqrt(k) for k in K_values]

        ax.plot(K_values, ratios,
               marker=MARKERS.get(alg, 'o'),
               color=COLORS.get(alg, '#333'),
               label=alg, linewidth=2, markersize=10)

    ax.set_xscale('log', base=2)
    ax.set_xlabel('Number of Configurations (K)')
    ax.set_ylabel('Regret / (√T · √K)')
    ax.set_title('Regret Scaling Ratio: Should Be ≈ Constant if Regret ∝ √KT', fontweight='bold')
    ax.set_xticks(K_values)
    ax.set_xticklabels([str(k) for k in K_values])
    ax.legend(loc='upper right')
    ax.grid(alpha=0.3)

    plt.tight_layout()

    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_e3_sqrt_k_ratio.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


# =============================================================================
# COMBINED SUMMARY FIGURE
# =============================================================================

def generate_fig_summary(e1_df: pd.DataFrame, e2_df: pd.DataFrame, e3_df: pd.DataFrame,
                         output_dir: Path):
    """
    Figure Summary: Combined 3-panel figure for main paper.

    Compact summary of all new experimental findings.
    """
    fig, axes = plt.subplots(1, 3, figsize=(15, 4.5))

    # Panel 1: E1 - Benchmark Sanity (bar chart at ρ=0.7)
    ax = axes[0]
    df_rho = e1_df[e1_df['rho'] == 0.7]
    algorithms = ['SP-UCB-OLP', 'Greedy', 'Random']

    x = np.arange(len(algorithms))
    width = 0.35

    cr_star = [df_rho[df_rho['algorithm'] == alg]['CR_star'].mean() for alg in algorithms]
    cr_mix = [df_rho[df_rho['algorithm'] == alg]['CR_mix'].mean() for alg in algorithms]

    ax.bar(x - width/2, cr_star, width, label='CR* (vs V*)', color='#ff7f0e', alpha=0.8)
    ax.bar(x + width/2, cr_mix, width, label='CR^mix (vs V^mix)', color='#2ca02c', alpha=0.8)
    ax.axhline(y=1.0, color='red', linestyle='--', linewidth=2)

    ax.set_ylabel('Competitive Ratio')
    ax.set_title('(a) Benchmark Sanity\n(S4, ρ=0.7)', fontweight='bold')
    ax.set_xticks(x)
    ax.set_xticklabels(algorithms, fontsize=10)
    ax.set_ylim(0, 2.1)
    ax.legend(loc='upper right', fontsize=8)
    ax.grid(axis='y', alpha=0.3)

    # Panel 2: E2 - Ablations (S4, ρ=0.7 only)
    ax = axes[1]
    ablation_df = e2_df[(e2_df['scenario'] == 'S4') & (e2_df['rho'] == 0.7)]
    ablation_algs = ['SP-UCB-OLP', 'EnvelopeGreedy', 'MixtureLocalPrice', 'NoSlack', 'AcceptedOnly']

    means = [ablation_df[ablation_df['algorithm'] == alg]['CR_mix'].mean() for alg in ablation_algs]
    colors = [COLORS.get(alg, '#333') for alg in ablation_algs]

    bars = ax.bar(range(len(ablation_algs)), means, color=colors, alpha=0.8)
    ax.axhline(y=1.0, color='gray', linestyle='--', alpha=0.5)

    ax.set_ylabel('CR^mix')
    ax.set_title('(b) Ablation Study\n(S4, ρ=0.7)', fontweight='bold')
    ax.set_xticks(range(len(ablation_algs)))
    ax.set_xticklabels(['SP-UCB', 'Env.Grdy', 'Lcl.Prc', 'NoSlk', 'Acc.Only'], fontsize=9, rotation=20)
    ax.set_ylim(0, 1.05)
    ax.grid(axis='y', alpha=0.3)

    # Highlight the failure
    bars[2].set_color('#d62728')  # Red for MixtureLocalPrice
    ax.annotate('Global price\nessential', xy=(2, 0.50), xytext=(2, 0.70),
               fontsize=8, ha='center', arrowprops=dict(arrowstyle='->', color='red'))

    # Panel 3: E3 - K-sweep
    ax = axes[2]
    K_values = sorted(e3_df['K'].unique())

    for alg in ['SP-UCB-OLP', 'Greedy', 'Oracle']:
        alg_df = e3_df[e3_df['algorithm'] == alg]
        grouped = alg_df.groupby('K')['regret_normalized'].mean()
        ax.plot(K_values, grouped.values,
               marker=MARKERS.get(alg, 'o'),
               color=COLORS.get(alg, '#333'),
               label=alg, linewidth=2, markersize=8)

    # √K reference
    sqrt_k_ref = 3.5 * np.sqrt(np.array(K_values))
    ax.plot(K_values, sqrt_k_ref, 'k--', linewidth=2, label='∝ √K', alpha=0.7)

    ax.set_xscale('log', base=2)
    ax.set_yscale('log')
    ax.set_xlabel('K')
    ax.set_ylabel('Regret / √T')
    ax.set_title('(c) K-Sweep\n(T=5000, ρ=0.7)', fontweight='bold')
    ax.set_xticks(K_values)
    ax.set_xticklabels([str(k) for k in K_values])
    ax.legend(loc='upper left', fontsize=8)
    ax.grid(alpha=0.3, which='both')

    plt.tight_layout()

    for ext in ['pdf', 'png']:
        output_file = output_dir / f"fig_summary_new_experiments.{ext}"
        plt.savefig(output_file, dpi=300 if ext == 'png' else None, bbox_inches='tight')
        print(f"Saved: {output_file}")

    plt.close()


# =============================================================================
# MAIN
# =============================================================================

def main():
    script_dir = Path(__file__).parent.parent
    results_dir = script_dir / "results" / "comprehensive" / "full_run"
    figures_dir = script_dir / "figures" / "new_experiments"
    figures_dir.mkdir(parents=True, exist_ok=True)

    print("=" * 70)
    print("Generating New Experiment Figures (E1, E2, E3)")
    print("=" * 70)
    print(f"Results directory: {results_dir}")
    print(f"Output directory: {figures_dir}")

    # Load all data
    print("\nLoading data...")
    e1_df = load_e1_data(results_dir)
    e2_df = load_e2_data(results_dir)
    e3_df = load_e3_data(results_dir)

    print(f"  E1: {len(e1_df)} records")
    print(f"  E2: {len(e2_df)} records")
    print(f"  E3: {len(e3_df)} records")

    # Generate figures
    print("\n" + "-" * 50)
    print("Generating E1: Benchmark Sanity Check figures...")
    print("-" * 50)
    generate_fig_e1_benchmark_sanity(e1_df, figures_dir)
    generate_fig_e1_complementarity_bar(e1_df, figures_dir)

    print("\n" + "-" * 50)
    print("Generating E2: Ablation figures...")
    print("-" * 50)
    generate_fig_e2_ablations(e2_df, figures_dir)
    generate_fig_e2_ablation_heatmap(e2_df, figures_dir)

    print("\n" + "-" * 50)
    print("Generating E3: K-Sweep figures...")
    print("-" * 50)
    generate_fig_e3_k_sweep(e3_df, figures_dir)
    generate_fig_e3_sqrt_k_ratio(e3_df, figures_dir)

    print("\n" + "-" * 50)
    print("Generating Summary figure...")
    print("-" * 50)
    generate_fig_summary(e1_df, e2_df, e3_df, figures_dir)

    print("\n" + "=" * 70)
    print(f"All figures saved to: {figures_dir}")
    print("=" * 70)

    # List all generated files
    print("\nGenerated files:")
    for f in sorted(figures_dir.glob("*.pdf")):
        print(f"  {f.name}")


if __name__ == '__main__':
    main()
