"""Submission-grade figures for the SAF paper (one chart per figure).

Produces five single-panel figures referenced from paper/main.tex:
    fig1_matrix.pdf       2x2 signal-coverage matrix, V4-Pro Vanilla × PN#
    fig2_tcsf.pdf         TC% vs Opus-SF% grouped bars, four methods
    fig3_ksat.pdf         SAF K-saturation curve with Lean-Retry/SF markers
    fig4_transition.pdf   Vanilla→Lean-Retry 4×4 transition matrix
    fig5_crossmodel.pdf   3 models × 2 datasets × 3 methods grouped bars

All numbers come from main.tex (§5.1, §5.2, §5.5, §5.6) and the v2 raw run
summaries under v2/runs/. Run from paper/figures/:
    python make_figures.py
"""

from __future__ import annotations

from pathlib import Path

import matplotlib as mpl
mpl.use("Agg")
import matplotlib.pyplot as plt
import matplotlib.patches as patches
import numpy as np

plt.rcParams["font.family"] = "sans-serif"
plt.rcParams["font.sans-serif"] = ["Arial", "Helvetica", "DejaVu Sans"]
plt.rcParams["svg.fonttype"] = "none"
plt.rcParams["pdf.fonttype"] = 42
plt.rcParams["ps.fonttype"] = 42
plt.rcParams["font.size"] = 7.6
plt.rcParams["axes.linewidth"] = 0.7
plt.rcParams["axes.spines.top"] = False
plt.rcParams["axes.spines.right"] = False
plt.rcParams["legend.frameon"] = False
plt.rcParams["xtick.major.width"] = 0.7
plt.rcParams["ytick.major.width"] = 0.7
plt.rcParams["xtick.major.size"] = 2.6
plt.rcParams["ytick.major.size"] = 2.6

METHOD_COLOR = {
    "Vanilla":       "#B4C0E4",
    "Lean-Retry":    "#7884B4",
    "Sample-Filter": "#5C6CA0",
    "SAF":           "#484878",
}
DELTA_UP   = "#2E9E44"
DELTA_DOWN = "#E53935"
NEUTRAL_LIGHT = "#D8D8D8"
NEUTRAL_MID   = "#A8A8A8"
NEUTRAL_DARK  = "#606060"
ROSE_OUTLINE  = "#C2185B"
EDGE_BLUE_DARK = "#3C5169"
EDGE_BLUE_MID  = "#7E8FA5"

OUTDIR = Path(__file__).resolve().parent

SINGLE_W = 3.35
TWO_W    = 7.00

def save(fig, basename):
    OUTDIR.mkdir(parents=True, exist_ok=True)
    for ext in ("pdf", "svg"):
        fig.savefig(OUTDIR / f"{basename}.{ext}", bbox_inches="tight")
    plt.close(fig)

def fig1_matrix():
    fig, ax = plt.subplots(figsize=(SINGLE_W, 3.10))

    matrix = np.array([
        [95, 16],
        [61, 14],
    ], dtype=float)
    cell_label = np.array([
        ["TRUE  SUCCESS", "SEMANTIC-ONLY"],
        ["TYPE-ONLY",     "BOTH-FAIL"],
    ])
    n = int(matrix.sum())
    row_sums = matrix.sum(axis=1).astype(int)
    col_sums = matrix.sum(axis=0).astype(int)

    AGREE = "#4FA773"
    DISAGREE = "#E08E45"
    stripe = np.array([
        [AGREE,    DISAGREE],
        [DISAGREE, AGREE],
    ])

    cmap = plt.cm.Blues
    vmax = matrix.max() * 1.15
    cell_size = 1.0
    gap = 0.06

    def cell_origin(i, j):
        return (j * (cell_size + gap), (1 - i) * (cell_size + gap))

    for (i, j), v in np.ndenumerate(matrix):
        x0, y0 = cell_origin(i, j)
        rgba = cmap(v / vmax)
        rect = patches.Rectangle(
            (x0, y0), cell_size, cell_size,
            facecolor=rgba, edgecolor="none", zorder=2,
        )
        ax.add_patch(rect)
        stripe_rect = patches.Rectangle(
            (x0, y0 + cell_size - 0.06), cell_size, 0.06,
            facecolor=stripe[i, j], edgecolor="none", zorder=3,
        )
        ax.add_patch(stripe_rect)

        text_color = "white" if v / vmax > 0.55 else NEUTRAL_DARK
        cx = x0 + cell_size / 2
        ax.text(cx, y0 + 0.78, cell_label[i, j],
                ha="center", va="center", fontsize=6.6,
                color=text_color, fontweight="bold")
        ax.text(cx, y0 + 0.45, f"{int(v)}",
                ha="center", va="center", fontsize=18,
                color=text_color, fontweight="bold")
        ax.text(cx, y0 + 0.18, f"{v / n * 100:.1f}%",
                ha="center", va="center", fontsize=6.8,
                color=text_color)

    grid_w = 2 * cell_size + gap

    col_titles = ["BT: equivalent", "BT: not equiv."]
    for j, title in enumerate(col_titles):
        x0, _ = cell_origin(0, j)
        ax.text(x0 + cell_size / 2, -0.10, title,
                ha="center", va="top", fontsize=7.4, color=EDGE_BLUE_DARK,
                fontweight="bold")
    for j, cs in enumerate(col_sums):
        x0, _ = cell_origin(0, j)
        ax.text(x0 + cell_size / 2, -0.30,
                f"{cs}  ({cs / n * 100:.1f}%)",
                ha="center", va="top", fontsize=6.4, color=EDGE_BLUE_MID)

    row_titles = ["Elab: pass", "Elab: fail"]
    for i, title in enumerate(row_titles):
        _, y0 = cell_origin(i, 0)
        ax.text(-0.10, y0 + cell_size / 2, title,
                ha="right", va="center", fontsize=7.4, color=EDGE_BLUE_DARK,
                fontweight="bold")
    for i, rs in enumerate(row_sums):
        _, y0 = cell_origin(i, 0)
        ax.text(grid_w + 0.12, y0 + cell_size / 2,
                f"{rs}  ({rs / n * 100:.1f}%)",
                ha="left", va="center", fontsize=6.4, color=EDGE_BLUE_MID)

    leg_y = 2 * cell_size + gap + 0.18
    ax.add_patch(patches.Rectangle((0.05, leg_y), 0.10, 0.05,
                                   facecolor=AGREE, edgecolor="none"))
    ax.text(0.18, leg_y + 0.025, "signals agree",
            ha="left", va="center", fontsize=6.4, color=EDGE_BLUE_DARK)
    ax.add_patch(patches.Rectangle((0.80, leg_y), 0.10, 0.05,
                                   facecolor=DISAGREE, edgecolor="none"))
    ax.text(0.93, leg_y + 0.025, "signals disagree",
            ha="left", va="center", fontsize=6.4, color=EDGE_BLUE_DARK)

    ax.set_xlim(-0.55, grid_w + 0.55)
    ax.set_ylim(-0.42, leg_y + 0.18)
    ax.set_aspect("equal")
    ax.axis("off")

    fig.subplots_adjust(left=0.02, right=0.98, top=0.99, bottom=0.02)
    save(fig, "fig1_matrix")

def fig2_tcsf():
    fig, ax = plt.subplots(figsize=(SINGLE_W, 2.55))

    methods = ["Vanilla", "Lean-Retry", "Sample-Filter", "SAF"]
    tc = np.array([59.7, 75.8, 76.3, 75.3])
    sf = np.array([51.1, 70.4, 70.4, 69.4])

    x = np.arange(len(methods))
    w = 0.36

    tc_color = METHOD_COLOR["SAF"]
    sf_color = METHOD_COLOR["Vanilla"]

    bars_tc = ax.bar(
        x - w / 2, tc, width=w,
        color=tc_color, edgecolor=NEUTRAL_DARK, linewidth=0.5,
        hatch="xxx", zorder=3,
    )
    bars_sf = ax.bar(
        x + w / 2, sf, width=w,
        color=sf_color, edgecolor=NEUTRAL_DARK, linewidth=0.5,
        hatch="////", zorder=3,
    )
    for bar, val in list(zip(bars_tc, tc)) + list(zip(bars_sf, sf)):
        ax.text(
            bar.get_x() + bar.get_width() / 2, val + 1.0,
            f"{val:.1f}",
            ha="center", va="bottom",
            fontsize=6.4, color=NEUTRAL_DARK,
        )

    proxy_tc = patches.Rectangle((0, 0), 1, 1, facecolor=tc_color,
                                 edgecolor=NEUTRAL_DARK, linewidth=0.5,
                                 hatch="xxx")
    proxy_sf = patches.Rectangle((0, 0), 1, 1, facecolor=sf_color,
                                 edgecolor=NEUTRAL_DARK, linewidth=0.5,
                                 hatch="////")
    ax.legend(
        [proxy_tc, proxy_sf], ["TC %", "Opus SF %"],
        loc="upper center", bbox_to_anchor=(0.5, 1.10),
        fontsize=6.8, ncol=2, handlelength=1.2,
        columnspacing=1.4, handletextpad=0.4, frameon=False,
    )

    ax.set_xticks(x)
    ax.set_xticklabels(methods, fontsize=7.0)
    ax.tick_params(axis="x", length=0, pad=3)
    ax.set_ylabel("V4-Pro × ProofNet$^{\\#}$  (%)", fontsize=7.4)
    ax.set_ylim(0, 90)
    ax.set_yticks([0, 20, 40, 60, 80])

    fig.subplots_adjust(left=0.16, right=0.97, top=0.88, bottom=0.13)
    save(fig, "fig2_tcsf")

def fig3_ksat():
    fig, ax = plt.subplots(figsize=(SINGLE_W, 2.65))

    K  = np.array([0, 1, 2, 3])
    tc = np.array([50.5, 72.0, 74.7, 75.3])

    ax.plot(K, tc, color=METHOD_COLOR["SAF"], lw=1.7,
            marker="o", markersize=5.2, markeredgecolor="white",
            markeredgewidth=0.8, zorder=4, label="SAF (this paper)")

    label_offsets = {0: 3.0, 1: 1.5, 2: 1.5, 3: -2.5}
    for k, v in zip(K, tc):
        dy = label_offsets[int(k)]
        va = "top" if dy < 0 else "bottom"
        ax.text(k, v + dy, f"{v:.1f}",
                ha="center", va=va,
                fontsize=6.6, color=METHOD_COLOR["SAF"], fontweight="bold")

    deltas = np.diff(tc)
    for k0, d in zip(K[:-1], deltas):
        ax.text(
            k0 + 0.5, (tc[k0] + tc[k0 + 1]) / 2 - 4.0,
            f"+{d:.1f}",
            ha="center", va="top",
            fontsize=6.0, color=NEUTRAL_MID, style="italic",
        )

    LR_COLOR = "#D08740"
    SF_COLOR = "#2A9D90"
    ax.axhline(75.8, color=LR_COLOR, linestyle="--",
               linewidth=0.9, alpha=0.9, zorder=2)
    ax.axhline(76.3, color=SF_COLOR, linestyle="--",
               linewidth=0.9, alpha=0.9, zorder=2)
    ax.text(-0.15, 85.0, "Sample-Filter 76.3",
            ha="left", va="bottom", fontsize=6.2,
            color=SF_COLOR, fontweight="bold")
    ax.text(-0.15, 81.0, "Lean-Retry 75.8",
            ha="left", va="bottom", fontsize=6.2,
            color=LR_COLOR, fontweight="bold")

    ax.set_xticks(K)
    ax.set_xticklabels([str(k) for k in K], fontsize=7.0)
    ax.set_xlabel("Refinement budget  $K$", fontsize=7.4)
    ax.set_ylabel("SAF TC %  on ProofNet$^{\\#}$ (n=186)", fontsize=7.4)
    ax.set_xlim(-0.30, 3.30)
    ax.set_ylim(45, 90)
    ax.legend(loc="lower right", bbox_to_anchor=(1.0, 0.02),
              fontsize=6.4, handlelength=1.3)

    fig.subplots_adjust(left=0.16, right=0.97, top=0.95, bottom=0.17)
    save(fig, "fig3_ksat")

def fig4_transition():
    fig, ax = plt.subplots(figsize=(SINGLE_W, 3.15))

    M = np.array([
        [92,  3,  0,  0],
        [14,  2,  0,  0],
        [23,  1, 35,  2],
        [ 2,  4,  6,  2],
    ], dtype=float)
    cls = ["TS", "SO", "TO", "BF"]
    n_total = int(M.sum())
    row_sums = M.sum(axis=1).astype(int)
    col_sums = M.sum(axis=0).astype(int)

    cmap = plt.cm.Blues
    vmax = M.max() * 1.10
    cell_size = 1.0
    gap = 0.05

    def cell_origin(i, j):
        return (j * (cell_size + gap), (3 - i) * (cell_size + gap))

    RESCUE = "#4FA773"
    NEW_ERR = "#E08E45"
    col_stripe_color = {0: RESCUE, 1: NEW_ERR}
    col_stripe_label = {0: "rescued (39)", 1: "new SO (8)"}

    for (i, j), v in np.ndenumerate(M):
        x0, y0 = cell_origin(i, j)
        rgba = cmap(v / vmax) if v > 0 else (0.97, 0.97, 0.97, 1.0)
        ax.add_patch(patches.Rectangle(
            (x0, y0), cell_size, cell_size,
            facecolor=rgba, edgecolor="none", zorder=2,
        ))

        if v == 0:
            display, fw, fs, tc = ".", "normal", 9, NEUTRAL_MID
        else:
            display = f"{int(v)}"
            base = v / vmax
            tc = "white" if base > 0.55 else NEUTRAL_DARK
            fw, fs = "bold", 11
        ax.text(x0 + cell_size / 2, y0 + cell_size / 2, display,
                ha="center", va="center", fontsize=fs,
                color=tc, fontweight=fw)

        if i == j:
            ax.add_patch(patches.Rectangle(
                (x0, y0), cell_size, cell_size,
                facecolor="none", edgecolor=NEUTRAL_MID, linewidth=0.7,
                linestyle=":", zorder=4,
            ))

    grid_w = 4 * cell_size + 3 * gap
    grid_h = 4 * cell_size + 3 * gap
    class_label_y = grid_h + 0.06
    stripe_y = class_label_y + 0.22
    stripe_h = 0.08
    stripe_text_y = stripe_y + stripe_h + 0.05

    for j, lab in enumerate(cls):
        x0, _ = cell_origin(0, j)
        ax.text(x0 + cell_size / 2, class_label_y, lab,
                ha="center", va="bottom", fontsize=7.6,
                color=EDGE_BLUE_DARK, fontweight="bold")

    for j, col_color in col_stripe_color.items():
        x0, _ = cell_origin(0, j)
        ax.add_patch(patches.Rectangle(
            (x0, stripe_y), cell_size, stripe_h,
            facecolor=col_color, edgecolor="none", zorder=3,
        ))
        ax.text(x0 + cell_size / 2, stripe_text_y,
                col_stripe_label[j],
                ha="center", va="bottom", fontsize=6.0,
                color=col_color, fontweight="bold")

    for i, lab in enumerate(cls):
        _, y0 = cell_origin(i, 0)
        ax.text(-0.12, y0 + cell_size / 2, lab,
                ha="right", va="center", fontsize=7.6,
                color=EDGE_BLUE_DARK, fontweight="bold")

    for i, rs in enumerate(row_sums):
        _, y0 = cell_origin(i, 0)
        ax.text(grid_w + 0.10, y0 + cell_size / 2,
                f"{int(rs)}",
                ha="left", va="center", fontsize=7.0,
                color=EDGE_BLUE_MID)
    for j, cs in enumerate(col_sums):
        x0, _ = cell_origin(0, j)
        ax.text(x0 + cell_size / 2, -0.18,
                f"{int(cs)}",
                ha="center", va="top", fontsize=7.0,
                color=EDGE_BLUE_MID)
    ax.text(grid_w + 0.10, -0.18, f"n = {n_total}",
            ha="left", va="top", fontsize=6.8, color=EDGE_BLUE_MID,
            style="italic")

    top_super_y = stripe_text_y + 0.30
    ax.text(grid_w / 2, top_super_y, "Lean-Retry cell",
            ha="center", va="bottom", fontsize=7.4,
            color=EDGE_BLUE_DARK, style="italic")
    ax.text(-0.60, grid_h / 2, "Vanilla cell",
            ha="center", va="center", rotation=90,
            fontsize=7.4, color=EDGE_BLUE_DARK, style="italic")

    ax.set_xlim(-0.85, grid_w + 0.65)
    ax.set_ylim(-0.40, top_super_y + 0.20)
    ax.set_aspect("equal")
    ax.axis("off")

    fig.subplots_adjust(left=0.02, right=0.98, top=0.99, bottom=0.02)
    save(fig, "fig4_transition")

def fig5_crossmodel():
    fig, ax = plt.subplots(figsize=(TWO_W, 2.75))

    combos = [
        ("DeepSeek V4-Pro\nProofNet$^{\\#}$", 186, 59.7, 75.8, 75.3),
        ("DeepSeek V4-Pro\nMiniF2F",          244, 87.3, 93.4, 95.5),
        ("Qwen3.5-Plus\nProofNet$^{\\#}$",    186, 71.5, 84.4, 58.6),
        ("Qwen3.5-Plus\nMiniF2F",             244, 93.9, 98.4, 96.3),
        ("MiMo-v2.5-Pro\nProofNet$^{\\#}$",   186, 67.7, 80.1, 75.3),
        ("MiMo-v2.5-Pro\nMiniF2F",            244, 86.5, 95.1, 95.5),
    ]
    labels = [c[0] for c in combos]
    n_vals = [c[1] for c in combos]
    tc_van = np.array([c[2] for c in combos])
    tc_lr  = np.array([c[3] for c in combos])
    tc_saf = np.array([c[4] for c in combos])

    n_cats = len(combos)
    bar_w  = 0.78 / 3
    x      = np.arange(n_cats)

    series = [
        ("Vanilla",    tc_van, METHOD_COLOR["Vanilla"],    ""),
        ("Lean-Retry", tc_lr,  METHOD_COLOR["Lean-Retry"], "////"),
        ("SAF",        tc_saf, METHOD_COLOR["SAF"],        "xxx"),
    ]
    for i, (label, vals, color, hatch) in enumerate(series):
        offset = (i - 1) * bar_w
        bars = ax.bar(
            x + offset, vals, width=bar_w,
            color=color, edgecolor=NEUTRAL_DARK, linewidth=0.5,
            hatch=hatch, label=label, zorder=3,
        )
        for bar, v in zip(bars, vals):
            ax.text(
                bar.get_x() + bar.get_width() / 2, v + 0.8,
                f"{v:.1f}",
                ha="center", va="bottom",
                fontsize=5.6, color=NEUTRAL_DARK,
            )

    xt_labels = [f"{lab}\n(n={n})" for lab, n in zip(labels, n_vals)]
    ax.set_xticks(x)
    ax.set_xticklabels(xt_labels, fontsize=6.6)
    ax.tick_params(axis="x", length=0, pad=4)

    ax.set_ylabel("TC %", fontsize=8)
    ax.set_ylim(0, 115)
    ax.set_yticks([0, 25, 50, 75, 100])

    ax.legend(
        loc="upper center", bbox_to_anchor=(0.5, 1.12),
        ncol=3, fontsize=7.2, handlelength=1.5, columnspacing=1.6,
    )

    fig.subplots_adjust(left=0.07, right=0.99, top=0.86, bottom=0.18)
    save(fig, "fig5_crossmodel")

def fig5_crossmodel_v2():
    fig, ax = plt.subplots(figsize=(SINGLE_W + 0.4, 3.30))

    rows = [
        ("DeepSeek V4-Pro",  "PN$^{\\#}$", 186),
        ("DeepSeek V4-Pro",  "MF",         244),
        ("Qwen3.5-Plus",     "PN$^{\\#}$", 186),
        ("Qwen3.5-Plus",     "MF",         244),
        ("MiMo-v2.5-Pro",    "PN$^{\\#}$", 186),
        ("MiMo-v2.5-Pro",    "MF",         244),
    ]
    methods = ["Vanilla", "Lean-Retry", "SAF"]
    tc = np.array([
        [59.7, 75.8, 75.3],
        [87.3, 93.4, 95.5],
        [71.5, 84.4, 58.6],
        [93.9, 98.4, 96.3],
        [67.7, 80.1, 75.3],
        [86.5, 95.1, 95.5],
    ])

    n_rows = len(rows)
    n_cols = len(methods)
    cell_w, cell_h = 1.0, 0.85
    gap = 0.06

    def cell_xy(i, j):
        return (j * (cell_w + gap), (n_rows - 1 - i) * (cell_h + gap))

    cmap = plt.cm.Blues
    vmin, vmax = 30.0, 100.0

    for i in range(n_rows):
        for j in range(n_cols):
            v = tc[i, j]
            x0, y0 = cell_xy(i, j)
            rgba = cmap((v - vmin) / (vmax - vmin))
            ax.add_patch(patches.Rectangle(
                (x0, y0), cell_w, cell_h,
                facecolor=rgba, edgecolor="none", zorder=2,
            ))
            tc_norm = (v - vmin) / (vmax - vmin)
            txt_color = "white" if tc_norm > 0.55 else NEUTRAL_DARK
            ax.text(x0 + cell_w / 2, y0 + cell_h / 2,
                    f"{v:.1f}",
                    ha="center", va="center", fontsize=8.8,
                    color=txt_color, fontweight="bold")

    i_exc, j_exc = 2, 2
    x0, y0 = cell_xy(i_exc, j_exc)
    ax.add_patch(patches.Circle(
        (x0 + cell_w - 0.13, y0 + cell_h - 0.13),
        0.07, facecolor="white", edgecolor=DELTA_DOWN, linewidth=1.0,
        zorder=4,
    ))
    ax.text(x0 + cell_w - 0.13, y0 + cell_h - 0.13, "!",
            ha="center", va="center", fontsize=6.6,
            color=DELTA_DOWN, fontweight="bold", zorder=5)

    grid_w = n_cols * cell_w + (n_cols - 1) * gap
    grid_h = n_rows * cell_h + (n_rows - 1) * gap

    for j, m in enumerate(methods):
        x0, _ = cell_xy(0, j)
        ax.text(x0 + cell_w / 2, grid_h + 0.06, m,
                ha="center", va="bottom", fontsize=7.6,
                color=NEUTRAL_DARK, fontweight="bold")

    for i, (model, ds, n) in enumerate(rows):
        _, y0 = cell_xy(i, 0)
        ax.text(-0.12, y0 + cell_h / 2 + 0.10, model,
                ha="right", va="center", fontsize=6.8,
                color=NEUTRAL_DARK, fontweight="bold")
        ax.text(-0.12, y0 + cell_h / 2 - 0.13,
                f"{ds}  ($n{{=}}{n}$)",
                ha="right", va="center", fontsize=6.0,
                color=NEUTRAL_MID, style="italic")

    for i in [2, 4]:
        _, y0 = cell_xy(i, 0)
        y_div = y0 + cell_h + gap / 2
        ax.plot([-0.04, grid_w + 0.02], [y_div, y_div],
                color=NEUTRAL_LIGHT, lw=0.6, zorder=1)

    cbar_x = grid_w + 0.20
    cbar_w = 0.10
    cbar_h_total = grid_h
    n_slices = 64
    for k in range(n_slices):
        frac = k / (n_slices - 1)
        rgba = cmap(frac)
        slice_y = (k / n_slices) * cbar_h_total
        slice_h = (1.0 / n_slices) * cbar_h_total + 0.005
        ax.add_patch(patches.Rectangle(
            (cbar_x, slice_y), cbar_w, slice_h,
            facecolor=rgba, edgecolor="none", zorder=2,
        ))
    for tick_val in [30, 50, 70, 90]:
        frac = (tick_val - vmin) / (vmax - vmin)
        ty = frac * cbar_h_total
        ax.text(cbar_x + cbar_w + 0.04, ty, f"{tick_val}",
                ha="left", va="center", fontsize=6.0, color=NEUTRAL_MID)
    ax.text(cbar_x + cbar_w / 2, cbar_h_total + 0.06,
            "TC %", ha="center", va="bottom", fontsize=6.4,
            color=NEUTRAL_DARK, fontweight="bold")

    ax.text(grid_w / 2, -0.25,
            "!  SAF $<$ Vanilla on Qwen3.5-Plus $\\times$ ProofNet$^{\\#}$  "
            "(disclosed IR-extraction issue)",
            ha="center", va="top", fontsize=5.8,
            color=DELTA_DOWN, style="italic")

    ax.set_xlim(-1.55, cbar_x + cbar_w + 0.42)
    ax.set_ylim(-0.45, grid_h + 0.25)
    ax.set_aspect("equal")
    ax.axis("off")

    fig.subplots_adjust(left=0.02, right=0.98, top=0.99, bottom=0.02)
    save(fig, "fig5_crossmodel_v2")

if __name__ == "__main__":
    fig1_matrix()
    fig2_tcsf()
    fig3_ksat()
    fig4_transition()
    fig5_crossmodel()

    print("Final files in", OUTDIR)
    for ext in ("pdf", "svg"):
        for p in sorted(OUTDIR.glob(f"fig*.{ext}")):
            print(" ", p.name)
