import numpy as np
from typing import Optional, List, Tuple


def box_kernel(k: int) -> np.ndarray:
    return np.ones((k, k), dtype=np.float64) / (k * k)


def gaussian_kernel(k: int, sigma: Optional[float] = None) -> np.ndarray:
    if sigma is None:
        sigma = 0.3 * k
    c = (k - 1) / 2.0
    x = np.arange(k) - c
    g1 = np.exp(-0.5 * (x / sigma) ** 2)
    g1 /= g1.sum()
    g2 = g1[:, None] @ g1[None, :]
    g2 /= g2.sum()
    return g2


def freq_radius_grid(N: int, norm: str = "l2"):
    coords = np.fft.fftfreq(N) * 2 * np.pi
    X, Y = np.meshgrid(np.fft.fftshift(coords), np.fft.fftshift(coords), indexing="xy")
    if norm == "linf":
        R = np.maximum(np.abs(X), np.abs(Y))
        r_max = np.pi
    else:
        R = np.sqrt(X**2 + Y**2)
        r_max = np.pi * np.sqrt(2.0)
    return R, r_max


def fft_mag_kernel(kernel: np.ndarray, N: int = 512) -> np.ndarray:
    pad = np.zeros((N, N), dtype=np.float64)
    kH, kW = kernel.shape
    r0 = (N - kH) // 2
    c0 = (N - kW) // 2
    pad[r0 : r0 + kH, c0 : c0 + kW] = kernel
    pad = np.fft.ifftshift(pad)
    K = np.fft.fft2(pad)
    K = np.fft.fftshift(K)
    mag = np.abs(K)
    dc = mag[N // 2, N // 2]
    mag = mag / max(dc, 1e-12)
    return mag


def radial_profile(
    mag2d: np.ndarray, nbins: int = 600, metric: str = "mean", norm: str = "l2"
):
    N = mag2d.shape[0]
    R, r_max = freq_radius_grid(N, norm=norm)
    Rf = R.ravel()
    Mf = mag2d.ravel()
    r_edges = np.linspace(0.0, r_max, nbins + 1)
    r_centers = 0.5 * (r_edges[:-1] + r_edges[1:])
    bin_idx = np.clip(np.digitize(Rf, r_edges) - 1, 0, nbins - 1)
    if metric == "mean":
        counts = np.bincount(bin_idx, minlength=nbins)
        sums = np.bincount(bin_idx, weights=Mf, minlength=nbins)
        avg = sums / np.maximum(counts, 1e-12)
        mask = counts > 10
        return r_centers[mask], avg[mask]
    elif metric == "p95":
        vals = [Mf[bin_idx == i] for i in range(nbins)]
        q = np.array([np.quantile(v, 0.95) if v.size else np.nan for v in vals])
        mask = ~np.isnan(q)
        return r_centers[mask], q[mask]
    elif metric == "max":
        m = np.full(nbins, -np.inf, dtype=np.float64)
        np.maximum.at(m, bin_idx, Mf)
        mask = np.isfinite(m)
        return r_centers[mask], m[mask]
    else:
        raise ValueError("metric must be one of {'mean','p95','max'}")


def fit_beta_delta_weighted(r, m, k, p=0.0, r_cut=0.0, beta_grid=None, delta_grid=None):
    eps = 1e-12
    if beta_grid is None:
        beta_grid = np.logspace(-2, 1.0, 40)
    if delta_grid is None:
        delta_grid = np.linspace(0.1, 3.0, 40)
    low_cut = max(1e-06, r_cut)
    mask = r > low_cut
    r_fit = r[mask]
    m_fit = m[mask]
    logm = np.log(np.maximum(m_fit, eps))
    w = r_fit**p
    w = w / np.maximum(w.mean(), 1e-12)
    best = None
    best_err = np.inf
    for beta in beta_grid:
        kr = beta * k * r_fit
        for delta in delta_grid:
            phi = (1.0 + kr) ** -(1.0 + delta)
            logp = np.log(np.maximum(phi, eps))
            err = np.mean(w * (logm - logp) ** 2)
            if err < best_err:
                best_err = err
                best = beta, delta
    beta, delta = best
    phi_full = (1.0 + beta * k * r) ** -(1.0 + delta)
    return beta, delta, best_err, phi_full


def strict_upper_bound_multiplier_2d(
    mag2d: np.ndarray, k: int, beta: float, delta: float, norm: str = "l2"
):
    eps = 1e-12
    N = mag2d.shape[0]
    R, _ = freq_radius_grid(N, norm=norm)
    phi2d = (1.0 + beta * k * R) ** -(1.0 + delta)
    ratio2d = (mag2d + eps) / (phi2d + eps)
    c_star = float(np.max(ratio2d))
    c_star_db = 2e1 * np.log10(max(c_star, 1e-12))
    return c_star, c_star_db


def _nearest_idx(arr: np.ndarray, value: float) -> int:
    return int(np.argmin(np.abs(arr - value)))


def _pick_idx_with_ratio_leq_one(
    r: np.ndarray, ratio: np.ndarray, target_r: float, max_span_bins: int, taken: set
) -> Optional[int]:
    j0 = _nearest_idx(r, target_r)
    if j0 not in taken and ratio[j0] <= 1.0:
        return j0
    for d in range(1, max_span_bins + 1):
        left = j0 - d
        right = j0 + d
        cand: List[Tuple[int, float]] = []
        if left >= 0 and left not in taken:
            cand.append((left, abs(r[left] - target_r)))
        if right < len(r) and right not in taken:
            cand.append((right, abs(r[right] - target_r)))
        for idx, _ in sorted(cand, key=lambda x: x[1]):
            if ratio[idx] <= 1.0:
                return idx


def print_pointwise_filtered(
    r: np.ndarray,
    m: np.ndarray,
    phi: np.ndarray,
    targets: List[float],
    max_span_bins: int = 50,
):
    eps = 1e-12
    ratio_full = (m + eps) / (phi + eps)
    print("  Pointwise (selected radii; only rows with ratio ≤ 1 kept)")
    hr("-", 86)
    print(
        "     r (rad) |       Empirical (|K̂|) |      Theoretical (φ_k) |    ratio |  diff_dB"
    )
    hr("-", 86)
    used = set()
    for t in targets:
        idx = _pick_idx_with_ratio_leq_one(r, ratio_full, t, max_span_bins, used)
        if idx is None:
            continue
        used.add(idx)
        ri, mi, phii = float(r[idx]), float(m[idx]), float(phi[idx])
        ratio = (mi + eps) / (phii + eps)
        diff_db = 2e1 * np.log10(max(ratio, 1e-12))
        print(f"{ri:11.4f} | {mi:22.6g} | {phii:22.6g} | {ratio:8.5g} | {diff_db:8.4g}")
    print()


def hr(ch="-", n=90):
    print(ch * n)


def main():
    kernels = {"box": box_kernel, "gaussian": gaussian_kernel}
    k = 31
    N = 512
    NORM = "l2"
    METRIC = "mean"
    TARGETS = [np.pi / 8, np.pi / 4, np.pi / 2]
    ALT_SEARCH_BINS = 60
    cases = [
        ("box", k, "unweighted", {"p": 0.0, "r_cut": 0.0}),
        ("gaussian", k, "unweighted", {"p": 0.0, "r_cut": 0.0}),
    ]
    for name, ksize, fit_label, spec in cases:
        K = kernels[name](ksize)
        mag = fft_mag_kernel(K, N=N)
        r, m = radial_profile(mag, nbins=600, metric=METRIC, norm=NORM)
        beta, delta, err, phi = fit_beta_delta_weighted(
            r, m, ksize, p=spec["p"], r_cut=spec["r_cut"]
        )
        hr("=", 90)
        print(
            f"Empirical vs Theoretical Envelope — kernel={name}, k={ksize}, fit={fit_label}"
        )
        hr("-", 90)
        print_pointwise_filtered(
            r, m, phi, targets=TARGETS, max_span_bins=ALT_SEARCH_BINS
        )
        c_star, c_star_db = strict_upper_bound_multiplier_2d(
            mag, ksize, beta, delta, norm=NORM
        )
        print(
            f"  Strict upper-bound multiplier c* (2D): {c_star:.4g}  ( = {c_star_db:.4g} dB )"
        )
        print(
            f"  Fitted parameters: beta_hat = {beta:.5g},  delta_hat = {delta:.5g},  logMSE_fit(log) = {err:.5g}"
        )
        print()


if __name__ == "__main__":
    main()
