from __future__ import annotations

from dataclasses import dataclass
from typing import Dict, List

import hashlib
import matplotlib as mpl

# Okabe–Ito palette (colorblind-friendly)
COLORBLIND_COLORS: List[str] = [
    "#0072B2",
    "#D55E00",
    "#009E73",
    "#CC79A7",
    "#F0E442",
    "#56B4E9",
    "#000000",
]


def apply_mpl_style() -> None:
    """Apply matplotlib styling for experiment plots."""
    mpl.rcParams.update(
        {
            "figure.dpi": 140,
            "savefig.dpi": 300,
            "axes.spines.top": False,
            "axes.spines.right": False,
            "axes.grid": True,
            "grid.alpha": 0.25,
            "grid.linestyle": "-",
            "axes.axisbelow": True,
            "legend.frameon": True,
            "legend.framealpha": 0.9,
            "legend.fancybox": True,
            "lines.linewidth": 2.0,
            "lines.markersize": 5.0,
            "font.size": 12,
            "axes.titlesize": 13,
            "axes.labelsize": 12,
            "xtick.labelsize": 11,
            "ytick.labelsize": 11,
        }
    )


@dataclass(frozen=True)
class MethodStyle:
    """MethodStyle class."""
    color: str
    linestyle: str
    marker: str


def _stable_u32(key: str) -> int:
    """ stable u32 for the given inputs."""
    digest = hashlib.md5(key.encode("utf-8"), usedforsecurity=False).digest()
    return int.from_bytes(digest[:4], byteorder="little", signed=False)


# Fixed, order-independent styles for known estimators.
# If you add new estimators, consider extending this map to keep plots consistent.
FIXED_METHOD_STYLES: Dict[str, MethodStyle] = {
    "naive_ols": MethodStyle(color="#7A7A7A", linestyle=":", marker="o"),
    "ts_2sls": MethodStyle(color="#0072B2", linestyle="--", marker="s"),
    "ts_iv": MethodStyle(color="#56B4E9", linestyle="--", marker="^"),
    "up_gmm": MethodStyle(color="#D55E00", linestyle="-", marker="D"),
    "up_gmm_hd": MethodStyle(color="#E69F00", linestyle="-", marker="P"),
    "up_gmm_hd_moment": MethodStyle(color="#CC79A7", linestyle="-.", marker="X"),
    "up_gmm_hd_analytic": MethodStyle(color="#E69F00", linestyle="-.", marker="v"),
    "jn": MethodStyle(color="#000000", linestyle="-", marker="*"),
}


def default_method_styles(keys: List[str]) -> Dict[str, MethodStyle]:
    """Return default method styles for the given inputs."""
    markers = ["o", "s", "^", "D", "P", "X", "*", "v"]
    linestyles = ["-", "--", "-.", ":"]
    out: Dict[str, MethodStyle] = {}
    for k in keys:
        fixed = FIXED_METHOD_STYLES.get(k)
        if fixed is not None:
            out[k] = fixed
            continue

        # Deterministic fallback for unknown keys (stable across order and runs).
        h = _stable_u32(k)
        out[k] = MethodStyle(
            color=COLORBLIND_COLORS[h % len(COLORBLIND_COLORS)],
            linestyle=linestyles[(h // 7) % len(linestyles)],
            marker=markers[(h // 49) % len(markers)],
        )
    return out

