import numpy as np
import matplotlib.pyplot as plt
from numpy.polynomial import Polynomial, Chebyshev, Legendre, Hermite
from scipy.integrate import quad

def get_basis_constructor(name):
    basis_map = {
        'monomial': Polynomial,
        'chebyshev': Chebyshev,
        'legendre': Legendre,
        'hermite': Hermite,
    }
    if name.lower() not in basis_map:
        raise ValueError(f"Unknown basis '{name}'. Choose from {list(basis_map.keys())}")
    return basis_map[name.lower()]

def poly_to_basis(poly_coeffs, from_basis='monomial', to_basis='chebyshev', domain=[-1, 1], deg=None):
    """
    Convert a univariate polynomial from one basis to another.
    
    poly_coeffs: list or np.array of coefficients (assumed from 'from_basis')
    from_basis, to_basis: str, one of ['monomial', 'chebyshev', 'legendre', 'hermite']
    domain: interval over which the polynomial is defined
    deg: degree of target approximation (if None, use input degree)
    """
    n = len(poly_coeffs)
    deg = deg or n - 1

    # Construct input polynomial
    from_cls = get_basis_constructor(from_basis)
    to_cls = get_basis_constructor(to_basis)

    p_from = from_cls(poly_coeffs, domain=domain)

    # Define inner product on [-1,1]
    def inner_product(f, g):
        return quad(lambda x: f(x) * g(x), domain[0], domain[1])[0]

    # Build target basis
    coeffs_out = []
    for i in range(deg + 1):
        b = to_cls.basis(i, domain=domain)
        norm = inner_product(b, b)
        proj = inner_product(p_from, b) / norm
        coeffs_out.append(proj)

    return np.array(coeffs_out), to_cls(coeffs_out, domain=domain)