from matplotlib import pyplot as plt
from matplotlib.patches import FancyArrowPatch, FancyBboxPatch, Rectangle


plt.rcParams.update({
    "font.family": "DejaVu Sans",
    "font.size": 6.9,
    "pdf.fonttype": 42,
    "ps.fonttype": 42,
})

fig, ax = plt.subplots(figsize=(7.20, 3.55))
ax.set_xlim(0, 1)
ax.set_ylim(0, 1)
ax.axis("off")

INK = "#141414"
DARK = "#333333"
MID = "#6a6a6a"
LIGHT = "#eeeeee"
PALE = "#f8f8f8"
ROW = "#fbfbfb"
HILITE = "#e7e7e7"


def rounded(x, y, w, h, title="", body="", fc="white", ec=INK, lw=0.85, ls="-",
            title_size=6.0, body_size=5.1, weight="bold"):
    patch = FancyBboxPatch(
        (x, y), w, h,
        boxstyle="round,pad=0.007,rounding_size=0.010",
        linewidth=lw,
        edgecolor=ec,
        facecolor=fc,
        linestyle=ls,
    )
    ax.add_patch(patch)
    if title:
        ax.text(x + w / 2, y + h - 0.020, title, ha="center", va="top",
                fontsize=title_size, fontweight=weight, color=INK, linespacing=1.03)
    if body:
        ax.text(x + w / 2, y + h / 2 - 0.010, body, ha="center", va="center",
                fontsize=body_size, color=DARK, linespacing=1.12)
    return patch


def arrow(x1, y1, x2, y2, lw=0.85, ls="-", color=INK, rad=0.0, scale=8):
    arr = FancyArrowPatch(
        (x1, y1), (x2, y2),
        arrowstyle="-|>",
        mutation_scale=scale,
        linewidth=lw,
        linestyle=ls,
        color=color,
        connectionstyle=f"arc3,rad={rad}",
    )
    ax.add_patch(arr)
    return arr


def field_box(x, y, w, h, title, rows, changed=None):
    changed = set(changed or [])
    rounded(x, y, w, h, title, fc=PALE, title_size=6.3, body="")
    top = y + h - 0.052
    row_h = (h - 0.067) / len(rows)
    for i, (label, value) in enumerate(rows):
        yy = top - (i + 1) * row_h
        face = HILITE if i in changed else ROW
        ax.add_patch(Rectangle((x + 0.010, yy + 0.003), w - 0.020, row_h - 0.006,
                               facecolor=face, edgecolor="#d5d5d5", linewidth=0.35))
        ax.text(x + 0.018, yy + row_h / 2, label, ha="left", va="center",
                fontsize=4.9, fontweight="bold", color=INK)
        ax.text(x + 0.076, yy + row_h / 2, value, ha="left", va="center",
                fontsize=4.9, color=DARK)


def small_step(x, y, w, h, title, fc="white"):
    rounded(x, y, w, h, title, fc=fc, title_size=5.0, body="", weight="normal")


ax.text(0.030, 0.974, "A. Reducer-mediated mutation of an explicit reasoning field",
        ha="left", va="top", fontsize=7.5, fontweight="bold", color=INK)

field_box(
    0.035, 0.635, 0.215, 0.270, r"Field $F_t$",
    [
        ("paths", "A active; B suspended"),
        ("constr.", r"$C_t$: c1"),
        ("goals", "g1 open"),
        ("frame", r"$\phi_t$: algebraic"),
        ("audit", "attempt 1 accepted"),
    ],
)

rounded(
    0.300, 0.670, 0.155, 0.190, r"Typed ops $\Delta F_t$",
    "+ constraint c2\n"
    r"reframe to $\phi_b$" "\n"
    "reactivate path B\n"
    "retire missing path C",
    fc="white", title_size=5.9, body_size=4.9,
)

rounded(
    0.505, 0.720, 0.128, 0.112, "Reducer",
    "local field\nvalidity",
    fc=LIGHT, title_size=5.9, body_size=4.8,
)

rounded(0.498, 0.575, 0.142, 0.074, fc="white", ec=MID, ls="--")
ax.text(0.569, 0.625, "Rejected log", ha="center", va="center",
        fontsize=5.2, fontweight="bold", color=INK)
ax.text(0.569, 0.598, "invalid path ref.", ha="center", va="center",
        fontsize=4.6, color=DARK)

field_box(
    0.690, 0.635, 0.225, 0.270, r"Field $F_{t+1}$",
    [
        ("paths", "A active; B reactivated"),
        ("constr.", r"$C_{t+1}$: c1, c2"),
        ("goals", "g1 open"),
        ("frame", r"$\phi_{t+1}$: divisors"),
        ("audit", "accepts + rejects kept"),
    ],
    changed={0, 1, 3, 4},
)

arrow(0.250, 0.770, 0.300, 0.770)
arrow(0.455, 0.770, 0.505, 0.770)
arrow(0.633, 0.770, 0.690, 0.770)
arrow(0.568, 0.720, 0.568, 0.649, color=MID, ls="--", scale=7)
ax.text(0.568, 0.842, "accepted", ha="center", va="bottom", fontsize=4.8, color=DARK)
ax.text(0.478, 0.542,
        "Reducer checks local state validity; it does not certify mathematical truth.",
        ha="center", va="center", fontsize=5.6, color=DARK)

ax.plot([0.035, 0.965], [0.515, 0.515], color="#d0d0d0", linewidth=0.75)
ax.text(0.030, 0.490, "B. One transition record supports inference, audit, and training",
        ha="left", va="top", fontsize=7.35, fontweight="bold", color=INK)

step_y, step_h = 0.385, 0.060
step_w = 0.095
steps = [
    (0.055, "problem +\nrecent attempt", PALE),
    (0.188, "typed\nproposal", "white"),
    (0.321, "reducer", LIGHT),
    (0.454, "field\nupdate", PALE),
    (0.587, "solver\nview", "white"),
    (0.720, "frozen\nsolver", "white"),
    (0.853, "outcome", "white"),
]
for x, title, fc in steps:
    small_step(x, step_y, step_w, step_h, title, fc=fc)
for (x1, _, _), (x2, _, _) in zip(steps[:-1], steps[1:]):
    arrow(x1 + step_w, step_y + step_h / 2, x2, step_y + step_h / 2,
          lw=0.75, scale=7)

record_x, record_y, record_w, record_h = 0.335, 0.255, 0.330, 0.100
rounded(record_x, record_y, record_w, record_h, fc=PALE)
ax.text(record_x + record_w / 2, record_y + record_h - 0.023,
        "transition record", ha="center", va="top",
        fontsize=5.65, fontweight="bold", color=INK)
ax.text(record_x + record_w / 2, record_y + 0.047,
        r"$F_t$, proposal, accept/reject", ha="center", va="center",
        fontsize=4.25, color=DARK)
ax.text(record_x + record_w / 2, record_y + 0.027,
        r"$F_{t+1}$, solver view, outcome", ha="center", va="center",
        fontsize=4.25, color=DARK)
arrow(0.368, step_y, record_x + 0.055, record_y + record_h, lw=0.75, color=MID)
arrow(0.900, step_y, record_x + record_w - 0.055, record_y + record_h,
      lw=0.75, color=MID, rad=-0.18)

uses = [
    (0.090, "Inference", "state projection\nchanges next solve"),
    (0.390, "Audit / ablation", "replay or remove\noperations"),
    (0.690, "Training", "serialize examples\nfor adapters"),
]
for x, title, body in uses:
    rounded(x, 0.080, 0.220, 0.105, title, body, fc="white",
            title_size=5.7, body_size=4.8)

split_y = 0.205
ax.plot([record_x + record_w / 2, record_x + record_w / 2], [record_y, split_y],
        color=MID, linewidth=0.75)
ax.plot([0.200, 0.800], [split_y, split_y], color=MID, linewidth=0.75)
for x, _, _ in uses:
    arrow(x + 0.110, split_y, x + 0.110, 0.185, lw=0.75, color=MID, scale=7)

ax.text(0.500, 0.045,
        "The trace separates discovering an answer, retaining it, and selecting it.",
        ha="center", va="center", fontsize=5.9, color=DARK)

plt.tight_layout(pad=0.18)
fig.savefig("fig1_field_dynamics.pdf", bbox_inches="tight")
