import constants as C

from numpy.random import shuffle
import frac as F


NUM_TO_TEXT = {
    1: "one",
    2: "two",
    3: "three",
    4: "four",
    6: "six",
    8: "eight",
    10: "ten",
    12: "twelve",
}


class Prompt_Coins_REVB:
    _BASE_INTRO_PLUR = "There are {num_text} coins. "  # num_text = {Two, Three, etc.}

    _BASE_PROBABILITY_UNFAIR = "Each coin is biased and is {num_bias} times more likely to land on {focus_side} than on {other_side}. "
    _BASE_PROBABILITY_FAIR = "Each coin is fair and is equally likely to land on {focus_side} and {other_side}. "

    _BASE_CONCLUSION = (
        "The coins are flipped and the resulting number of {focus_side} is equal to"
    )

    _PREVIOUS_LAUNCH = "The coins are flipped a first time and the resulting number of {focus_side} is {previous_value}. "

    _INDEPENDANT_CONCLUSION = "The coins are flipped again and the new resulting number of {focus_side} is equal to"

    _DEPENDANT_CONCLUSION = "The coins are flipped again and the total number of {focus_side} shown in the two launches is equal to"

    @staticmethod
    def regular_experiment(number_of_coins, weight_of_focus, focus) -> str:
        if weight_of_focus == 1:
            proba_prompt = Prompt_Coins_REVB._BASE_PROBABILITY_FAIR
        else:
            proba_prompt = Prompt_Coins_REVB._BASE_PROBABILITY_UNFAIR

        other = C.TAILS if focus == C.HEADS else C.HEADS

        base_prompt = (
            Prompt_Coins_REVB._BASE_INTRO_PLUR
            + proba_prompt
            + Prompt_Coins_REVB._BASE_CONCLUSION
        )

        return base_prompt.format(
            num_text=number_of_coins,
            focus_side=focus,
            other_side=other,
            num_bias=weight_of_focus,
        )

    @staticmethod
    def independant_previous_launch(
        number_of_coins, weight_of_focus, focus, previous_value
    ) -> str:
        if weight_of_focus == 1:
            proba_prompt = Prompt_Coins_REVB._BASE_PROBABILITY_FAIR
        else:
            proba_prompt = Prompt_Coins_REVB._BASE_PROBABILITY_UNFAIR

        other = C.TAILS if focus == C.HEADS else C.HEADS

        base_prompt = (
            Prompt_Coins_REVB._BASE_INTRO_PLUR
            + proba_prompt
            + Prompt_Coins_REVB._PREVIOUS_LAUNCH
            + Prompt_Coins_REVB._INDEPENDANT_CONCLUSION
        )

        return base_prompt.format(
            num_text=number_of_coins,
            focus_side=focus,
            other_side=other,
            num_bias=weight_of_focus,
            previous_value=previous_value,
        )

    @staticmethod
    def dependant_previous_launch(
        number_of_coins, weight_of_focus, focus, previous_value
    ) -> str:
        if weight_of_focus == 1:
            proba_prompt = Prompt_Coins_REVB._BASE_PROBABILITY_FAIR
        else:
            proba_prompt = Prompt_Coins_REVB._BASE_PROBABILITY_UNFAIR

        other = C.TAILS if focus == C.HEADS else C.HEADS

        base_prompt = (
            Prompt_Coins_REVB._BASE_INTRO_PLUR
            + proba_prompt
            + Prompt_Coins_REVB._PREVIOUS_LAUNCH
            + Prompt_Coins_REVB._DEPENDANT_CONCLUSION
        )

        return base_prompt.format(
            num_text=number_of_coins,
            focus_side=focus,
            other_side=other,
            num_bias=weight_of_focus,
            previous_value=previous_value,
        )


class Prompt_Coins_StaA:
    _BASE_INTRO_PLUR = "There are {num_text} coins. "  # num_text = {Two, Three, etc.}

    _BASE_PROBABILITY_UNFAIR = "Each coin is biased and is {num_bias} times more likely to land on {focus_side} than on {other_side}. "
    _BASE_PROBABILITY_FAIR = "Each coin is fair and is equally likely to land on {focus_side} and {other_side}. "

    _BASE_CONCLUSION = (
        "The coins are flipped and the resulting number of {focus_side} is equal to"
    )

    _PREVIOUS_LAUNCH = "The coins are flipped a first time and the resulting number of {focus_side} is {previous_value}. "

    _INDEPENDANT_CONCLUSION = "The coins are flipped again and the new resulting number of {focus_side} is equal to"

    _DEPENDANT_CONCLUSION = "The coins are flipped again and the total number of {focus_side} shown in the two launches is equal to"

    _QUESTION_BASE = "What is the probability that the resulting number of {focus_side} is equal to {question_value} after flipping the coins?"

    _QUESTION_INDEPENDANT = "What is the probability that the new number of {focus_side} is equal to {question_value} after flipping the coins again?"

    _QUESTION_DEPENDANT = "What is the probability that the total number of {focus_side} in the two launches is equal to {question_value} after flipping the coins again?"

    @staticmethod
    def regular_experiment(
        number_of_coins, weight_of_focus, focus, question_for_value, true_distribution
    ):
        true_answer = true_distribution[question_for_value]

        if weight_of_focus == 1:
            proba_prompt = Prompt_Coins_StaA._BASE_PROBABILITY_FAIR
        else:
            proba_prompt = Prompt_Coins_StaA._BASE_PROBABILITY_UNFAIR

        other = C.TAILS if focus == C.HEADS else C.HEADS

        base_prompt = Prompt_Coins_StaA._BASE_INTRO_PLUR + proba_prompt
        question = Prompt_Coins_StaA._QUESTION_BASE.format(
            focus_side=focus, question_value=question_for_value
        )

        choices, values, correct_answer = F.populate_answers_fractions(true_answer)

        bp = base_prompt.format(
            num_text=number_of_coins,
            focus_side=focus,
            other_side=other,
            num_bias=weight_of_focus,
        )

        return bp, question, choices, values, correct_answer

    @staticmethod
    def independant_previous_launch(
        number_of_coins,
        weight_of_focus,
        focus,
        previous_value,
        question_for_value,
        true_distribution,
    ):
        true_answer = true_distribution[question_for_value]

        if weight_of_focus == 1:
            proba_prompt = Prompt_Coins_StaA._BASE_PROBABILITY_FAIR
        else:
            proba_prompt = Prompt_Coins_StaA._BASE_PROBABILITY_UNFAIR

        other = C.TAILS if focus == C.HEADS else C.HEADS

        base_prompt = (
            Prompt_Coins_StaA._BASE_INTRO_PLUR
            + proba_prompt
            + Prompt_Coins_StaA._PREVIOUS_LAUNCH
        )

        question = Prompt_Coins_StaA._QUESTION_INDEPENDANT.format(
            focus_side=focus, question_value=question_for_value
        )

        base = str(previous_value)

        choices, values, correct_answer = F.populate_answers_fractions(true_answer)

        bp = base_prompt.format(
            num_text=number_of_coins,
            focus_side=focus,
            other_side=other,
            num_bias=weight_of_focus,
            previous_value=base,
        )

        return bp, question, choices, values, correct_answer

    @staticmethod
    def dependant_previous_launch(
        number_of_coins,
        weight_of_focus,
        focus,
        previous_value,
        question_for_value,
        true_distribution,
    ):
        true_answer = true_distribution[question_for_value]

        if weight_of_focus == 1:
            proba_prompt = Prompt_Coins_StaA._BASE_PROBABILITY_FAIR
        else:
            proba_prompt = Prompt_Coins_StaA._BASE_PROBABILITY_UNFAIR

        other = C.TAILS if focus == C.HEADS else C.HEADS

        true_answer = true_distribution[question_for_value]

        base_prompt = (
            Prompt_Coins_StaA._BASE_INTRO_PLUR
            + proba_prompt
            + Prompt_Coins_StaA._PREVIOUS_LAUNCH
        )
        question = Prompt_Coins_StaA._QUESTION_DEPENDANT.format(
            focus_side=focus, question_value=question_for_value
        )

        choices, values, correct_answer = F.populate_answers_fractions(true_answer)

        bp = base_prompt.format(
            num_text=number_of_coins,
            focus_side=focus,
            other_side=other,
            num_bias=weight_of_focus,
            previous_value=previous_value,
        )

        return bp, question, choices, values, correct_answer


if __name__ == "__main__":
    if 0:
        print(
            Prompt_Coins_REVB.regular_experiment(
                number_of_coins=2, weight_of_focus=1, focus=C.HEADS
            )
        )
        print(
            Prompt_Coins_REVB.regular_experiment(
                number_of_coins=2, weight_of_focus=4, focus=C.TAILS
            )
        )

    if 0:
        for i in range(3):
            print(
                Prompt_Coins_REVB.independant_previous_launch(
                    number_of_coins=2,
                    weight_of_focus=1,
                    focus=C.HEADS,
                    previous_value=i,
                )
            )
            print(
                Prompt_Coins_REVB.independant_previous_launch(
                    number_of_coins=2,
                    weight_of_focus=4,
                    focus=C.TAILS,
                    previous_value=i,
                )
            )

    if 1:
        for i in range(3):
            print(
                Prompt_Coins_REVB.dependant_previous_launch(
                    number_of_coins=2,
                    weight_of_focus=1,
                    focus=C.HEADS,
                    previous_value=i,
                )
            )
            print(
                Prompt_Coins_REVB.dependant_previous_launch(
                    number_of_coins=2,
                    weight_of_focus=4,
                    focus=C.TAILS,
                    previous_value=i,
                )
            )
