#!/usr/bin/env python3
"""
Generate q,t-Narayana polynomials for fixed k and all n in [k..N]
using their combinatorial interpretation in terms of area and bounce
of rectangular polyominoes of shape k times n-k+1 (Aval et al., 2014),
normalized so the minimal area/bounce term is (0, 0).
"""

from __future__ import annotations

import argparse
from dataclasses import dataclass
from itertools import combinations
from pathlib import Path
from typing import Iterable

import json


@dataclass(frozen=True)
class PathInfo:
    area_under: int
    vertices: frozenset[tuple[int, int]]
    heights: tuple[int, ...]
    verticals: tuple[int, ...]


@dataclass(frozen=True)
class SingleLine:
    value: list[int]


class SingleLineJSONEncoder(json.JSONEncoder):
    """JSON encoder that keeps SingleLine lists on one line."""

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        self._single_line: dict[str, str] = {}

    def default(self, obj):
        if isinstance(obj, SingleLine):
            marker = f"__SL_{id(obj)}__"
            self._single_line[marker] = json.dumps(obj.value)
            return marker
        return super().default(obj)

    def encode(self, obj):
        result = super().encode(obj)
        for marker, payload in self._single_line.items():
            result = result.replace(f"\"{marker}\"", payload)
        return result


def path_info_from_steps(steps: list[bool]) -> PathInfo:
    x = 0
    y = 0
    area_under = 0
    heights: list[int] = []
    verticals: list[int] = []
    vertices = {(0, 0)}
    for step in steps:
        if step:  # North
            verticals.append(x)
            y += 1
        else:  # East
            heights.append(y)
            area_under += y
            x += 1
        vertices.add((x, y))
    return PathInfo(
        area_under=area_under,
        vertices=frozenset(vertices),
        heights=tuple(heights),
        verticals=tuple(verticals),
    )


def generate_paths(m: int, n: int) -> list[PathInfo]:
    total = m + n
    paths: list[PathInfo] = []
    for north_positions in combinations(range(total), n):
        steps = [False] * total
        for idx in north_positions:
            steps[idx] = True
        paths.append(path_info_from_steps(steps))
    return paths


def bounce(u_heights: tuple[int, ...], l_verticals: tuple[int, ...], m: int, n: int) -> int:
    if m == 0 or n == 0:
        return 0
    x = 1
    y = 0
    k = 1
    acc = 0
    fuel = m + n + 2
    for _ in range(fuel):
        if x >= m and y >= n:
            break
        if x > 0 and x - 1 < m:
            next_y = u_heights[x - 1]
        else:
            next_y = y
        if next_y > 0 and next_y - 1 < n:
            next_x = l_verticals[next_y - 1]
        else:
            next_x = x

        term = (next_y - y) * k + (next_x - x) * k
        if next_x == x and next_y == y:
            break
        acc += term
        x, y = next_x, next_y
        k += 1
    return acc


def narayana_terms(m: int, n: int) -> list[tuple[int, int, int]]:
    paths = generate_paths(m, n)
    endpoints = {(0, 0), (m, n)}
    shift = m + n - 1  # Normalize so the minimal area/bounce term is (0, 0).
    counts: dict[tuple[int, int], int] = {}
    for upper in paths:
        for lower in paths:
            if upper.area_under < lower.area_under:
                continue
            if (upper.vertices & lower.vertices) != endpoints:
                continue
            area = upper.area_under - lower.area_under - shift
            bounce_val = bounce(upper.heights, lower.verticals, m, n) - shift
            key = (bounce_val, area)
            counts[key] = counts.get(key, 0) + 1

    items = [(q_exp, t_exp, coeff) for (q_exp, t_exp), coeff in counts.items()]
    items.sort(key=lambda item: (item[0] + item[1], item[0]))
    return items


def write_json(path: Path, data: Iterable[dict[str, object]]) -> None:
    payload = {
        "data": [
            {
                "n": entry["n"],
                "terms": [SingleLine(list(term)) for term in entry["terms"]],
            }
            for entry in data
        ]
    }
    rendered = json.dumps(payload, indent=2, cls=SingleLineJSONEncoder)
    path.write_text(rendered + "\n", encoding="utf-8")


def main() -> None:
    parser = argparse.ArgumentParser(
        description="Generate qt-Narayana polynomials for fixed k and all n up to N."
    )
    parser.add_argument("n", type=int, help="Maximum n (must satisfy n >= k >= 1).")
    parser.add_argument("k", type=int, help="Fixed k (number of columns).")
    args = parser.parse_args()

    if args.k < 1 or args.n < args.k:
        parser.error("Require n >= k >= 1.")

    data = []
    for n_val in range(args.k, args.n + 1):
        rows = n_val - args.k + 1
        terms = narayana_terms(args.k, rows)
        data.append({"n": n_val, "terms": terms})

    output = Path(__file__).resolve().parent / f"qt_narayana_n{args.n}_k{args.k}.json"
    write_json(output, data)
    print(f"Wrote {output}")


if __name__ == "__main__":
    main()
