"""Enumeration types for Expected GradCAM configuration."""

from __future__ import annotations

from enum import Enum


class SolverMethod(str, Enum):
    """Methods for solving the linear system M_I @ α = b.

    The choice of solver depends on the rank of M_I:
    - Full rank (M >= K): Any method works, 'pinv' is fastest
    - Rank deficient (M < K or data-aware): Use 'pinv' or 'adaptive_reg'
    """

    PINV = "pinv"
    """Moore-Penrose pseudo-inverse (recommended).

    α* = pinv(M_I) @ b

    Works for both full-rank and rank-deficient cases.
    Fastest and most numerically stable for data-aware perturbations.
    """

    ADAPTIVE_REG = "adaptive_reg"
    """Adaptive Tikhonov regularization.

    α* = (M_I + ε*I)^{-1} @ b

    Automatically selects ε based on eigenvalue analysis.
    Good for ill-conditioned problems.
    """

    SUBSPACE = "subspace"
    """Subspace projection method.

    Projects to the significant eigenspace of M_I before solving.
    Uses eigenvalue threshold to determine effective rank.
    """

    REGULARIZED = "regularized"
    """Fixed Tikhonov regularization.

    α* = (M_I + ε*I)^{-1} @ b

    Uses fixed regularization_eps from config.
    """

    @classmethod
    def default(cls) -> "SolverMethod":
        """Return the default solver method (pinv)."""
        return cls.PINV


class WeightTransform(str, Enum):
    """Weight transformation methods applied after solving.

    These transforms improve the visual quality and reduce infidelity
    by modifying the weight distribution.
    """

    NONE = "none"
    """No transformation - use raw optimal weights."""

    DOUBLE_POWER = "double_power"
    """Double power transform (DEFAULT, 99.7% improvement).

    w → (w * |w|) * |w * |w||  ≈ w^4 for positive weights

    Best performing transform for reducing infidelity.
    """

    EXTREME_POWER = "extreme_power"
    """Extreme power transform.

    w → w * |w|^p  where p is the configured exponent

    Sharper than double_power, may over-focus.
    """

    FEATURE_ADAPTIVE = "feature_adaptive"
    """Feature-adaptive power transform (97.7% improvement).

    Applies per-channel adaptive power based on feature map variance.
    More sophisticated but slightly slower.
    """

    FIXED_POWER = "fixed_power"
    """Fixed power transform.

    w → sign(w) * |w|^p  where p is the configured exponent

    Simple power law with configurable exponent.
    """

    @classmethod
    def default(cls) -> "WeightTransform":
        """Return the default weight transform (double_power)."""
        return cls.DOUBLE_POWER


class NormalizationMethod(str, Enum):
    """Heatmap normalization methods."""

    MINMAX = "minmax"
    """Min-max normalization to [0, 1].

    heatmap = (heatmap - min) / (max - min)

    Simple but sensitive to outliers.
    """

    QUANTILE = "quantile"
    """Quantile-based normalization (DEFAULT).

    heatmap = clip((heatmap - q_low) / (q_high - q_low), [0, 1])

    Robust to outliers. Uses 2nd and 98th percentiles by default.
    """

    SUM = "sum"
    """Sum normalization.

    heatmap = heatmap / sum(heatmap)

    Useful when comparing across images.
    """

    MAX = "max"
    """Max normalization.

    heatmap = heatmap / max(heatmap)

    Preserves relative structure but max is always 1.
    """

    @classmethod
    def default(cls) -> "NormalizationMethod":
        """Return the default normalization method (quantile)."""
        return cls.QUANTILE
