
from typing import Dict, Any, List, Tuple
import numpy as np


def _merge_intervals(intervals: List[Tuple[float, float]], eps: float = 1e-12) -> List[Tuple[float, float]]:
    intervals = sorted((a, b) for a, b in intervals if b > a)
    out: List[List[float]] = []
    for a, b in intervals:
        if (not out) or (a > out[-1][1] + eps):
            out.append([a, b])
        else:
            out[-1][1] = max(out[-1][1], b)
    return [(float(a), float(b)) for a, b in out]


def beta_paper_case(fine_t: np.ndarray, case_id: int = 2) -> Tuple[np.ndarray, Dict[str, Any]]:
    """
    Minimal beta(t) generator used in demo.
    Returns:
      betat: shape (Tf,)
      info:  {"case_id":..., "domain":(...,...), "intervals_true":[(a,b),...]}
    """
    t = np.asarray(fine_t, dtype=float)

    if case_id == 1:
        betat = np.zeros_like(t)
        intervals_true: List[Tuple[float, float]] = []

    elif case_id == 2:
        intervals_true = [(0.05, 0.15), (0.75, 0.85)]
        freq = 1.0
        betat = np.zeros_like(t)

        for a, b in intervals_true:
            m = (t >= a) & (t <= b)
            if not np.any(m):
                continue
            taper = (t[m] - a) * (b - t[m])
            taper = taper / taper.max() if taper.max() > 0 else taper
            betat[m] += 2.5 * taper * np.sin(freq * 2.0 * np.pi * (t[m] + 0.1))

    elif case_id == 3:
        intervals_true = [(0.4, 0.6)]
        betat = np.zeros_like(t)
        a, b = intervals_true[0]
        m = (t >= a) & (t <= b)
        if np.any(m):
            taper = (t[m] - a) * (b - t[m])
            taper = taper / taper.max() if taper.max() > 0 else taper
            betat[m] = 5.0 * taper

    elif case_id == 4:
        intervals_true = [(0.1, 0.3)]
        betat = np.zeros_like(t)
        a, b = intervals_true[0]
        m = (t >= a) & (t <= b)
        if np.any(m):
            taper = (t[m] - a) * (b - t[m])
            taper = taper / taper.max() if taper.max() > 0 else taper
            betat[m] = 5.0 * taper

    else:
        raise ValueError("case_id must be 1,2,3,4")

    info = {
        "case_id": int(case_id),
        "domain": (float(t.min()), float(t.max())),
        "intervals_true": _merge_intervals(intervals_true),
    }
    return betat, info
