class CommonPrompts:
    PROBLEM_DESCRIPTION = """
A Bongard Problem is composed of left and right sides separated by a line. Each side contains six images. All images belonging to one side present a common concept, which is lacking in all images from the other side. The goal is to describe the rule that fits all images on the left side, but none on the right, and, conversely, the rule that fits all images on the right side, but none on the left. The description of the rule should be simple and concise.
Example 1: All shapes on left are small. All shapes on right are big.
Example 2: The left side contains circles. The right side contains triangles.
""".strip()

    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem. What is the difference between the two sides of the problem?
""".strip()


class DirectPrompts:
    pass


class DescriptivePrompts:
    DESCRIBE_IMAGE_PROMPT = """
The provided image is a part of an abstract visual reasoning problem. Describe all crucial properties of the image. Your description should be as concise as possible. Focus on the most important details. The image is provided correctly. Respond only with descriptions.
""".strip()

    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem using descriptions of its images.

LEFT IMAGES:
{left_descriptions}

RIGHT IMAGES:
{right_descriptions}

What is the difference between the two sides of the problem?
""".strip()


class DescriptiveDirectPrompts:
    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem with the help of its image descriptions.

LEFT IMAGES:
{left_descriptions}

RIGHT IMAGES:
{right_descriptions}

What is the difference between the two sides of the problem?
""".strip()


class DescriptiveIterativePrompts:
    DESCRIBE_ITERATIVELY_SIDE_IMAGES_PROMPT = """
You'll receive a sequence of images that are a part of a single side of a Bongard Problem. The images will be provided one by one. Your goal is to find a common concept presented in all images. Your description should be as concise as possible. Focus on the most important details. Try to enhance the description of the concept after each image.

The image is always provided correctly. Respond only to the specific request. The first image will be provided in the next message.
""".strip()

    FINAL_ANSWER_PROMPT = "That was the last image. Now provide your final answer."

    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem using descriptions of two sides of the problem.

LEFT SIDE DESCRIPTION:
{left_description}

RIGHT SIDE DESCRIPTION:
{right_description}

What is the difference between the two sides of the problem?
""".strip()


class ContrastivePrompts:
    COMPARE_IMAGES_PROMPT = """
You are given two images extracted from the left and right side of a Bongard Problem, respectively. Your goal is to compare the images. Your comparison should be as concise as possible.
""".strip()

    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem using comparisons between pairs of images. Each pair contains one image from the left and one from the right side of the problem.

COMPARISONS:
{comparisons}

What is the difference between the two sides of the problem?
""".strip()


class ContrastiveDirectPrompts:
    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem with the help of comparisons between pairs of images. Each pair contains one image from the left and one from the right side of the problem.

COMPARISONS:
{comparisons}

What is the difference between the two sides of the problem?
""".strip()


class ContrastiveIterativePrompts:
    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to solve the provided Bongard Problem. You'll receive a sequence of image pairs. Each pair contains one image from the left and one from the right side of the problem. In each step compare the two images and refine the definitions of concepts that describe left and right sides of the problem. Your description should be as concise as possible. Focus on the most important details. The first pair will be provided in the next message.
""".strip()

    FINAL_QUESTION = """
It was the last pair of images. What is the difference between the two sides of the problem?
""".strip()


class IsLabelCorrectPrompts:
    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to evaluate the correctness of the provided answer to the given Bongard Problem. All images are provided correctly. Do not explain the answer, just evaluate it. Respond 'OK' if the answer is correct, otherwise respond 'WRONG'.
""".strip()


class ImageToSidePrompts:
    QUESTION = """
You are a vision understanding module designed to provide short, clear and accurate answers. Your goal is to classify two test images to the corresponding side of the Bongard Problem, LEFT or RIGHT. Each image belongs to exactly one class. The test images belong to different classes.

The images are always provided correctly. Respond only to the specific request. Respond in json using the following format.

FIRST EXAMPLE:
Left images: <small shapes>
Right images: <big shapes>

First test image: <small shape>
Second test image: <big shape>

Response: 
{
    "first": {
        "explanation": "The test image shows a small shape, similarly as all images on the left side. Conversely, the images on the right side feature big shapes.",
        "concept": "small vs big",
        "answer": "LEFT"
    }, 
    "second": {
        "explanation": "The test image shows a big shape, similarly as all images on right. The images on left, on the other hand, feature small shapes.",
        "concept": "small vs big",
        "answer": "RIGHT"
    }
}
END OF FIRST EXAMPLE

SECOND EXAMPLE:
Left images: <circles>
Right images: <triangles>

First test image: <triangle>
Second test image: <circle>

Response: 
{
    "first": {
        "explanation": "The test image shows a triangle, which matches all images on right. In contrast, the left side images feature circles.",
        "concept": "circles vs triangles",
        "answer": "RIGHT"
    }, 
    "second": {
        "explanation": "The test image shows a circle, which matches all images on left. Conversely, the right side images feature triangles.",
        "concept": "circles vs triangles",
        "answer": "LEFT"
    }
}
END OF SECOND EXAMPLE
""".strip()


class EvaluationPrompts:
    STRICT_LOGIC_PROMPT = """
You are a logic module designed to provide accurate answers. 
In a Bongard Problem the objective is to spot the difference between the contents of images located on the two opposite sides of the problem. 
You are given correct labels of these sides and must decide whether the answer provided by the user is correct and matches with those labels. Answer with 'OK' or 'WRONG'.
The user's answer has to strictly logically match the labels, as shown in examples.

FIRST EXAMPLE: 
LEFT SIDE LABEL: All shapes are small.
RIGHT SIDE LABEL: All shapes are big.
USER ANSWER: On the left side, one of the shapes is small. On the right side, all of the shapes are big.
EVALUATION: WRONG
END OF FIRST EXAMPLE.

SECOND EXAMPLE:
LEFT SIDE LABEL: All shapes are small.
RIGHT SIDE LABEL: All shapes are big.
USER ANSWER: On the left side, all of the shapes are small. On the right side, all of the shapes are big.
EVALUATION: OK
END OF SECOND EXAMPLE.


LEFT SIDE LABEL:
{first_label}

RIGHT SIDE LABEL:
{second_label}

USER ANSWER:
{model_answer}
""".strip()

    @staticmethod
    def from_name(name: str) -> str:
        if name == "STRICT_LOGIC_PROMPT":
            return EvaluationPrompts.STRICT_LOGIC_PROMPT
        else:
            raise ValueError(f"Unsupported name: {name}")
