import os
import re
import tempfile
from functools import partial

import pandas as pd
import ast

from ..image_base import ImageBaseDataset, img_root_map
from ..utils import build_judge, DEBUG_MESSAGE
from ...smp import *
from ...utils import track_progress_rich
from ipdb import set_trace as st

logger = get_logger("RUN")

"""
{
    "img_filename": "web_3b0ad239-da6b-4f6f-8f12-f674dc90ff33.png",
    "bbox": [42, 1102, 197, 70],
    "question": "view the details of the item",
    "data_type": "text",
    "data_source": "shop"
},
{
    "img_filename": "web_3b0ad239-da6b-4f6f-8f12-f674dc90ff33.png",
    "bbox": [93, 74, 86, 132],
    "question": "view the previous photo",
    "data_type": "icon",
    "data_source": "shop"
}
"""

SYSTEM_PROMPT = """You are a GUI agent. You are given a task and a screenshot of the screen. You need to perform pyautogui click/moveTo action to complete the task."""  # noqa: E501

USER_INSTRUCTION = """Please complete the following tasks by clicking using `pyautogui.click`:\n{instruction}"""  # noqa: E501

SYSTEM_PROMPT_V2 = """You are a GUI agent. You are given a screenshot of the screen and the description of a target element. You need to click the target element using `pyautogui.click`."""  # noqa: E501
USER_INSTRUCTION_V2 = """Please click the following target element using `pyautogui.click`:\n{description}"""


def parse_bbox_aguvis(response):
    match = re.search(r"x=([\d.]+), y=([\d.]+)", response)
    if match:
        click_point = [float(match.group(1)), float(match.group(2))]
    else:
        click_point = [0.0, 0.0]
    return click_point


def compute_iou(box1, box2):
    """
    Compute the Intersection over Union (IoU) of two bounding boxes.

    Parameters:
    - box1 (list of float): Bounding box [x_min, y_min, x_max, y_max].
    - box2 (list of float): Bounding box [x_min, y_min, x_max, y_max].

    Returns:
    - float: IoU of box1 and box2.
    """
    # Determine the coordinates of the intersection rectangle
    x_left = max(box1[0], box2[0])
    y_top = max(box1[1], box2[1])
    x_right = min(box1[2], box2[2])
    y_bottom = min(box1[3], box2[3])

    # Compute the area of intersection
    intersection_area = max(0, x_right - x_left) * max(0, y_bottom - y_top)

    # Compute the area of both bounding boxes
    box1_area = (box1[2] - box1[0]) * (box1[3] - box1[1])
    box2_area = (box2[2] - box2[0]) * (box2[3] - box2[1])

    # Compute the area of the union
    union_area = box1_area + box2_area - intersection_area

    # Compute the Intersection over Union
    iou = intersection_area / union_area

    return iou


def compute_accuracy(box1, box2, threshold=0.5):
    """
    Compute the accuracy of two bounding boxes based on a specified threshold.

    Parameters:
    - box1 (list of float): Bounding box [x_min, y_min, x_max, y_max].
    - box2 (list of float): Bounding box [x_min, y_min, x_max, y_max].
    - threshold (float): Threshold for the IoU to consider the prediction correct.

    Returns:
    - float: Accuracy of the prediction based on the IoU threshold.
    """
    iou = compute_iou(box1, box2)
    return iou >= threshold


def compute_center_accuracy(box1, box2):
    """
    Compute if the center point of box 2 is within box 1.

    Parameters:
    - box1 (list of float): Bounding box [x_min, y_min, x_max, y_max].
    - box2 (list of float): Bounding box [x_min, y_min, x_max, y_max].

    Returns:
    - bool: True if the center point of box 2 is within box 1, False otherwise.
    """
    # Compute the center point of box 2
    center_x = (box2[0] + box2[2]) / 2
    center_y = (box2[1] + box2[3]) / 2

    # Check if the center point is within box 1
    return box1[0] <= center_x <= box1[2] and box1[1] <= center_y <= box1[3]


def convert_bbox(bbox, image_path, convert_xywh_to_x1y1x2y2=True):
    new_bbox = bbox if isinstance(bbox, list) else ast.literal_eval(bbox)
    if convert_xywh_to_x1y1x2y2:
        new_bbox = [
            new_bbox[0],
            new_bbox[1],
            new_bbox[0] + new_bbox[2],
            new_bbox[1] + new_bbox[3],
        ]
    image = Image.open(image_path)
    img_size = image.size
    new_bbox = [
        new_bbox[0] / img_size[0],
        new_bbox[1] / img_size[1],
        new_bbox[2] / img_size[0],
        new_bbox[3] / img_size[1],
    ]
    return new_bbox


class ScreenSpot_Pro(ImageBaseDataset):
    MODALITY = "IMAGE"
    TYPE = "GUI"
    DATASET_URL = {
        "ScreenSpot_Pro_Development": "ScreenSpot_Pro_Development",
        "ScreenSpot_Pro_Creative": "ScreenSpot_Pro_Creative",
        "ScreenSpot_Pro_CAD": "ScreenSpot_Pro_CAD",
        "ScreenSpot_Pro_Scientific": "ScreenSpot_Pro_Scientific",
        "ScreenSpot_Pro_Office": "ScreenSpot_Pro_Office",
        "ScreenSpot_Pro_OS": "ScreenSpot_Pro_OS",
        # "ScreenSpot_Pro_Development": [
        #     "vscode_macos",
        #     "pycharm_macos",
        #     "android_studio_macos",
        #     "quartus_windows",
        #     "vmware_macos",
        # ],
        # "ScreenSpot_Pro_Creative": [
        #     "photoshop_windows",
        #     "premiere_windows",
        #     "illustrator_windows",
        #     "blender_windows",
        #     "fruitloops_windows",
        #     "unreal_engine_windows",
        #     "davinci_macos",
        # ],
        # "ScreenSpot_Pro_CAD": [
        #     "autocad_windows",
        #     "solidworks_windows",
        #     "inventor_windows",
        #     "vivado_windows",
        # ],
        # "ScreenSpot_Pro_Scientific": [
        #     "matlab_macos",
        #     "origin_windows",
        #     "stata_windows",
        #     "eviews_windows",
        # ],
        # "ScreenSpot_Pro_Office": ["word_macos", "excel_macos", "powerpoint_windows"],
        # "ScreenSpot_Pro_OS": [
        #     "macos_common_macos",
        #     "windows_common_windows",
        #     "linux_common_linux",
        # ],
    }  # path
    DATASET_MD5 = {}
    EVAL_TYPE = "point"  # point or rectangle
    RE_TYPE = "functional"  # type of referring expressions: functional or composite

    def __init__(
        self,
        dataset="ScreenSpot_Pro_Development",
        skip_noimg=True,
        skeleton=False,
        re_type="functional",
    ):
        # st()
        ROOT = LMUDataRoot()
        # You can override this variable to save image files to a different directory
        self.dataset_name = dataset
        self.img_root = osp.join(ROOT, "ScreenSpot_Pro", "images")
        self.RE_TYPE = re_type
        if skeleton:
            return

        data = self.load_data(dataset)
        self.skip_noimg = skip_noimg
        if skip_noimg and "image" in data:
            data = data[~pd.isna(data["image"])]

        data["index"] = [str(idx + 1) for idx, x in enumerate(data["bbox"])]

        self.meta_only = True
        self.parse_response_func = parse_bbox_aguvis  # TODO: parse function can be specified through kwargs when initializing the dataset # noqa: E501

        # The image field can store the base64 encoded image or another question index (for saving space) # noqa: E501
        if "image" in data:
            data["image"] = [str(x) for x in data["image"]]
            image_map = {x: y for x, y in zip(data["index"], data["image"])}
            for k in image_map:
                if len(image_map[k]) <= 64:
                    idx = image_map[k]
                    assert idx in image_map and len(image_map[idx]) > 64
                    image_map[k] = image_map[idx]

            images = [toliststr(image_map[k]) for k in data["index"]]
            data["image"] = [x[0] if len(x) == 1 else x for x in images]
            self.meta_only = False

        if "img_filename" in data:
            paths = [toliststr(x) for x in data["img_filename"]]
            data["image_path"] = [x[0] if len(x) == 1 else x for x in paths]
            data["image_path"] = [x.replace("matlab_mac/", "matlab_macos/") for x in data["image_path"]]

        # if np.all([istype(x, int) for x in data["index"]]):
        #     data["index"] = [int(x) for x in data["index"]]

        self.data = data
        self.post_build(dataset)

    def prepare_tsv(self, url, file_md5=None):
        # st()
        data = []
        if isinstance(url, str):
            url = [url]
        data_root = LMUDataRoot()
        for single_url in url:
            data_path = osp.join(
                data_root, "ScreenSpot_Pro", single_url + ".tsv"
            )
            task_data = load(data_path)
            data.extend(task_data)
        return pd.DataFrame(task_data)

    # actually retrieve the image path
    def dump_image(self, line):
        assert "image_path" in line
        tgt_path = toliststr(osp.join(self.img_root, line["image_path"]))
        return tgt_path

    @classmethod
    def get_action_space(self):
        return ""

    @classmethod
    def get_trajectory(self, line):
        traj_dict = {}
        if self.RE_TYPE == "functional":
            traj_dict["task"] = line["question"]
        else:
            traj_dict["task"] = line["description"]
        return traj_dict

    def build_prompt(self, line):
        if isinstance(line, int):
            line = self.data.iloc[line]
        tgt_path = self.dump_image(line)

        if self.RE_TYPE == "functional":
            user_instruction = USER_INSTRUCTION.format(instruction=line["question"])
        else:
            user_instruction = USER_INSTRUCTION_V2.format(
                description=line["description"]
            )

        msgs = []
        # add system prompt
        if self.RE_TYPE == "functional":
            msgs.append(dict(role="system", type="text", value=SYSTEM_PROMPT))
        else:
            msgs.append(dict(role="system", type="text", value=SYSTEM_PROMPT_V2))
        if isinstance(tgt_path, list):
            msgs.extend([dict(type="image", value=p) for p in tgt_path])
        else:
            msgs = [dict(type="image", value=tgt_path)]
        msgs.append(dict(type="text", value=user_instruction))
        return msgs

    def evaluate(self, eval_file, **judge_kwargs):
        # st()
        if self.EVAL_TYPE == "point":
            return self.evaluate_point(eval_file, **judge_kwargs)

        elif self.EVAL_TYPE == "rectangle":
            return self.evaluate_rectangle(eval_file, **judge_kwargs)

    def evaluate_rectangle(self, eval_file, **judge_kwargs):
        scorers = {
            "IoU": compute_iou,
            "ACC@0.1": lambda x, y: compute_accuracy(x, y, 0.1),
            "ACC@0.3": lambda x, y: compute_accuracy(x, y, 0.3),
            "ACC@0.5": lambda x, y: compute_accuracy(x, y, 0.5),
            "ACC@0.7": lambda x, y: compute_accuracy(x, y, 0.7),
            "ACC@0.9": lambda x, y: compute_accuracy(x, y, 0.9),
            "Center_ACC": compute_center_accuracy,
        }
        results_dict = {}
        for key in scorers.keys():
            results_dict.update(
                {
                    key: [],
                    key + "_text": [],
                    key + "_icon": [],
                }
            )

        result = []
        data = load(eval_file)
        assert "bbox" in data and "prediction" in data
        lt = len(data)
        lines = [data.iloc[i] for i in range(lt)]
        for i in tqdm(range(len(lines))):
            line = lines[i]
            bbox = convert_bbox(
                line["bbox"], os.path.join(self.img_root, line["image_path"]), convert_xywh_to_x1y1x2y2=False
            )
            prediction = str(line["prediction"])
            try:
                click_point = parse_bbox_aguvis(prediction)

                match = {}
                for score_key, score_value in scorers.items():
                    score = score_value(bbox, click_point)
                    if score_key != "IoU":
                        match[score_key.replace("ACC", "match")] = score
                    results_dict[score_key].append(score)
                    if line["ui_type"] == "text":
                        results_dict[score_key + "_text"].append(score)
                    else:
                        results_dict[score_key + "_icon"].append(score)
            except:
                click_point = None
                match = {score_key: False for score_key in scorers.keys() if score_key != "IoU"}
            result.append(
                {
                    "img_path": os.path.join(self.img_root, line["image_path"]),
                    "text": line["question"],
                    "bbox": line["bbox"],
                    "parsed_bbox": bbox,
                    "type": line["ui_type"],
                    "source": line["application"],
                    "pred": click_point,
                    "num_matched": sum(match.values()),
                    **match,
                }
            )
        for key in results_dict:
            if len(results_dict[key]) == 0:
                results_dict[key] = str(0)
            else:
                results_dict[key] = str(sum(results_dict[key]) / len(results_dict[key]))
        score_pth = eval_file.replace(".xlsx", "_score.json")
        dump(results_dict, score_pth)

        failure_cases_path = os.environ.get("FAILURE_CASES_PATH", None)
        if failure_cases_path is not None:
            failure_cases = [res for res in result if not res["match"] and res["is_wrong_format"]]
            failure_cases.sort(key=lambda r: r["num_matched"], reverse=True)

            with open(failure_cases_path, "w") as f:
                json.dump(failure_cases, f, indent=4, ensure_ascii=False)
        return results_dict

    def evaluate_point(self, eval_file, **judge_kwargs):
        # st()
        SCREENSPOT_result = dict(
            num_action=0,
            corr_action=0,
            text_correct=[],
            icon_correct=[],
            num_wrong_format=0,
            text_num=0,
            icon_num=0,
        )
        result = []
        data = load(eval_file)
        assert "bbox" in data and "prediction" in data
        lt = len(data)
        lines = [data.iloc[i] for i in range(lt)]
        for i in tqdm(range(len(lines))):
            SCREENSPOT_result["num_action"] += 1
            line = lines[i]
            bbox = (
                line["bbox"]
                if isinstance(line["bbox"], list)
                else ast.literal_eval(line["bbox"])
            )
            # bbox = [bbox[0], bbox[1], bbox[0] + bbox[2], bbox[1] + bbox[3]]  # f**k
            image = Image.open(os.path.join(self.img_root, line["image_path"]))
            img_size = image.size

            def make_safe(value):
                if value == -1:
                    # we can tolerate -1 as a special value and nomalize it to 0
                    return 0
                else:
                    return value

            bbox = [
                make_safe(bbox[0]) / img_size[0],
                make_safe(bbox[1]) / img_size[1],
                make_safe(bbox[2]) / img_size[0],
                make_safe(bbox[3]) / img_size[1],
            ]

            if any([x < 0 or x > 1 for x in bbox]):
                raise ValueError(f"bbox out of range: {bbox} | {line['bbox']} | {img_size}")

            prediction = str(line["prediction"])
            try:
                click_point = self.parse_response_func(prediction)
                match = (bbox[0] <= click_point[0] <= bbox[2]) and \
                    (bbox[1] <= click_point[1] <= bbox[3])
                if match:
                    SCREENSPOT_result["corr_action"] += 1
                    if line["ui_type"] == "text":
                        SCREENSPOT_result["text_correct"].append(1)
                        SCREENSPOT_result["text_num"] += 1
                    else:
                        SCREENSPOT_result["icon_correct"].append(1)
                        SCREENSPOT_result["icon_num"] += 1
                    logger.debug(
                        "match "
                        + str(
                            SCREENSPOT_result["corr_action"]
                            / SCREENSPOT_result["num_action"]
                        )
                    )
                else:
                    if line["ui_type"] == "text":
                        SCREENSPOT_result["text_correct"].append(0)
                        SCREENSPOT_result["text_num"] += 1
                    else:
                        SCREENSPOT_result["icon_correct"].append(0)
                        SCREENSPOT_result["icon_num"] += 1
                    logging.debug(
                        "unmatch "
                        + str(
                            SCREENSPOT_result["corr_action"]
                            / SCREENSPOT_result["num_action"]
                        )
                    )
                is_wrong_format = False

            except Exception as e:
                logger.warning(f"exception in screenspot eval:{e}")
                SCREENSPOT_result["num_wrong_format"] += 1
                if line["ui_type"] == "text":
                    SCREENSPOT_result["text_correct"].append(0)
                    SCREENSPOT_result["text_num"] += 1
                else:
                    SCREENSPOT_result["icon_correct"].append(0)
                    SCREENSPOT_result["icon_num"] += 1

                match, is_wrong_format, click_point = False, True, None

            result.append(
                {
                    "img_path": os.path.join(self.img_root, line["image_path"]),
                    "text": line["question"],
                    "bbox": line["bbox"],
                    "parsed_bbox": bbox,
                    "type": line["ui_type"],
                    "source": line["application"],
                    "match": match,
                    "is_wrong_format": is_wrong_format,
                    "pred": click_point,
                }
            )

        action_acc = SCREENSPOT_result["corr_action"] / SCREENSPOT_result["num_action"] * 100
        text_acc = (
            sum(SCREENSPOT_result["text_correct"])
            / len(SCREENSPOT_result["text_correct"])
            if len(SCREENSPOT_result["text_correct"]) != 0
            else 0
        ) * 100
        icon_acc = (
            sum(SCREENSPOT_result["icon_correct"])
            / len(SCREENSPOT_result["icon_correct"])
            if len(SCREENSPOT_result["icon_correct"]) != 0
            else 0
        ) * 100

        final_score_dict = {}
        final_score_dict["Action Acc"] = str(action_acc)
        final_score_dict["Total num"] = str(SCREENSPOT_result["num_action"])
        final_score_dict["Wrong format num"] = str(
            SCREENSPOT_result["num_wrong_format"]
        )
        final_score_dict["Text Num"] = str(SCREENSPOT_result["text_num"])
        final_score_dict["Icon Num"] = str(SCREENSPOT_result["icon_num"])
        final_score_dict["Text Acc"] = str(text_acc)
        final_score_dict["Icon Acc"] = str(icon_acc)

        score_pth = eval_file.replace(".xlsx", "_score.json")
        dump(final_score_dict, score_pth)

        failure_cases_path = os.environ.get("FAILURE_CASES_PATH", None)
        if failure_cases_path is not None:
            def click_distance(bbox, click_point):
                x, y = click_point
                x1, y1, x2, y2 = bbox
                xc, yc = (x1 + x2) / 2, (y1 + y2) / 2
                w, h = x2 - x1, y2 - y1
                abs_shift_to_center = [abs(x - xc), abs(y - yc)]  # noqa: E501
                width_outside, height_outside = [max(0, abs_shift_to_center[0] - w / 2), max(0, abs_shift_to_center[1] - h / 2)]  # noqa: E501
                return (width_outside ** 2 + height_outside ** 2) ** 0.5  # noqa: E501

            wrong_format_result = [res for res in result if res["is_wrong_format"]]
            missed_result = [res for res in result if not res["match"] and not res["is_wrong_format"]]
            missed_result.sort(key=lambda r: click_distance(r["parsed_bbox"], r["pred"]), reverse=True)
            failure_cases = wrong_format_result + missed_result

            with open(failure_cases_path, "w") as f:
                json.dump(failure_cases, f, indent=4, ensure_ascii=False)

        successful_cases_path = os.environ.get("SUCCESSFUL_CASES_PATH", None)
        if successful_cases_path is not None:
            def _click_distance(bbox, click_point):
                x, y = click_point
                x1, y1, x2, y2 = bbox
                xc, yc = (x1 + x2) / 2, (y1 + y2) / 2
                x_shift, y_shift = x - xc, y - yc
                return (x_shift ** 2 + y_shift ** 2) ** 0.5

            successful_cases = [res for res in result if res["match"]]
            successful_cases.sort(key=lambda r: _click_distance(r["parsed_bbox"], r["pred"]), reverse=True)
            with open(successful_cases_path, "w") as f:
                json.dump(successful_cases, f, indent=4, ensure_ascii=False)
        return final_score_dict
