from __future__ import annotations
import constants as C
from math import gcd, lcm
import numpy as np


class Frac:
    def __init__(self, num: int, den: int) -> None:
        self.num = num
        self.den = den
        self._simplify()

    def _simplify(self) -> None:
        g = gcd(self.num, self.den)
        self.num = self.num // g
        self.den = self.den // g

    def add(self, other_frac: Frac) -> Frac:
        l = lcm(self.den, other_frac.den)

        num = self.num * (l // self.den) + other_frac.num * (l // other_frac.den)

        f = Frac(num=num, den=l)
        return f

    def mult(self, other_frac: Frac) -> Frac:
        f = Frac(num=self.num * other_frac.num, den=self.den * other_frac.den)
        return f

    def inverse(self) -> Frac:
        f = Frac(num=self.den, den=self.num)
        return f

    def negation(self) -> Frac:
        f = Frac(num=-self.num, den=self.den)
        return f

    def divide(self, other_frac: Frac) -> Frac:
        inv_other_frac = other_frac.inverse()
        f = self.mult(other_frac=inv_other_frac)
        return f

    def minus(self, other_frac: Frac) -> Frac:
        inv_other_frac = other_frac.negation()
        f = self.add(other_frac=inv_other_frac)
        return f

    def power(self, power: int) -> Frac:
        frac = Frac(1, 1)
        for i in range(power):
            frac = frac.mult(self)
        return frac

    def bigger_than(self, other_frac: Frac) -> bool:
        v = self.minus(other_frac=other_frac)
        return (v.num * v.den) > 0

    def __str__(self):
        return f"{self.num}/{self.den}"

    def __eq__(self, value: Frac):
        self._simplify()
        value._simplify()

        return (self.num == value.num) & (self.den == value.den)

    def to_float(self) -> float:
        return self.num / self.den


def populate_answers_fractions(base_value: Frac):
    values = [base_value]
    den = base_value.den * 3

    for i in range(1, 3):
        num = (base_value.num) * 3 + i

        values.append(Frac(num=num, den=den))
        num = (base_value.num) * 3 - i
        values.append(Frac(num=num, den=den))

    choices = C.prompt_parameter.LETTERS[: len(values)]
    np.random.seed(123)
    np.random.shuffle(values)

    correct_answer = None
    for i in range(len(choices)):
        if values[i] == base_value:
            correct_answer = choices[i]
        values[i] = str(values[i])

    return choices, values, correct_answer


def get_valid_frac_from_couting(values: np.array) -> dict:
    unique, counts = np.unique(values, return_counts=True)
    n_tot = np.sum(counts)

    distribution = {}

    for i in range(len(unique)):
        num = counts[i]

        distribution[str(int(unique[i]))] = Frac(num=num, den=n_tot)

    return distribution
