import numpy as np
import gymnasium as gym

import shapely
from shapely import Point

import os
from typing import Optional, Type

from loguru import logger

from shapely import LineString

from ..colour_utils import Highlight

from molecule_movement.envs.MoleculeEnvironment import MoleculeEnvironment
from molecule_movement import Goal, Obstacle, Molecule
from molecule_movement.parsing import FixedVoltageDataParser, MoleculeDataProcessor, CircularMockUpData, FePcMockUpData, MockUpData
from molecule_movement.sampling import CircularSampler, Sampler
from molecule_movement.matching import RandomMatching, GreedyMatching, HungarianMatching
from molecule_movement.scheduling import SATBasedScheduling, SortScheduling
from molecule_movement.shapes import RECTANGLE, TRIANGLE, DDNB, FePc, resize, compute_ellipse, random_convex_polygon, hexagon, remove_closing_point

class SATSchedulingExampleEnv(MoleculeEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
    def _parse_molecule_data(self):
        pass

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        names = [f"i{i}" for i in range(len(self.goals))] #self._sample_n_random_names(len(self.goals))
        i = iter(range(len(self.goals)))
        configuration = self.system_configuration[self.type]
        shape = configuration.moiety.shape
        response_map = configuration.response_map
        action_space = configuration.action_space
        self.molecules.append(Molecule(Point(2,2),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(5,5),   shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(9,9), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(22,19), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(27,23),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(3,22),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        orientation = "pointy"
        xs, ys = remove_closing_point(hexagon(width=10.5, height=13.5, orientation=orientation, center=(self.surface_width//2, self.surface_height//2)).exterior.xy)
        for x,y in zip(xs, ys):
            self.goals.append(Goal(Point(x,y), FePc, 0, type=self.type))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        #self.obstacles = [Obstacle(Point(22,15), random_convex_polygon(), 0)]
        super()._set_obstacles()

    def _get_matching(self, seed: Optional[int] = None, options: Optional[dict] = None):
        try:
            corridor_config = options["corridor_config"]
            self.matching_factory = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True, corridor_config=corridor_config)
        except TypeError as e:
            logger.error(e)
            self.matching_factory = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True)
        self.matching = self.matching_factory.compute_matching()

class SATSchedulingEllipseExampleEnv(MoleculeEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
    def _parse_molecule_data(self):
        pass

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        names = [f"i{i}" for i in range(len(self.goals))] #self._sample_n_random_names(len(self.goals))
        i = iter(range(len(self.goals)))
        configuration = self.system_configuration[self.type]
        shape = configuration.moiety.shape
        response_map = configuration.response_map
        action_space = configuration.action_space
        self.molecules.append(Molecule(Point(11,9),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(2,1),   shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(13,21), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(20,28), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(7,34),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(3,57),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(14,49), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(18,41), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(38,6),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(37,12), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(44,3),  shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))
        self.molecules.append(Molecule(Point(48,21), shape, response_map, self.num_sensors, orientation="random", name=names[next(i)],type=configuration.moiety.type, molecule_point_symmetry=configuration.moiety.point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=action_space))


    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        goal_x, goal_y = compute_ellipse(25, 18, 12, self.surface_width // 2, self.surface_height // 2)
        for x,y in zip(goal_x, goal_y):
            self.goals.append(Goal(Point(x,y), FePc, 0, type=self.type))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        #self.obstacles = [Obstacle(Point(22,15), random_convex_polygon(), 0)]
        super()._set_obstacles()

    def _get_matching(self, seed: Optional[int] = None, options: Optional[dict] = None):
        try:
            corridor_config = options["corridor_config"]
            self.matching_factory = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True, corridor_config=corridor_config)
        except TypeError as e:
            logger.error(e)
            self.matching_factory = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True)
        self.matching = self.matching_factory.compute_matching()


class CorridorExampleEnv(MoleculeEnvironment):
    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        self.shape = self.system_configuration[self.type].moiety.shape

    def _parse_molecule_data(self):
        return

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        configuration = self.system_configuration[self.type]
        logger.info(self.num_sensors)
        self.molecules.append(Molecule(Point(self.surface_width // 2 - 2, self.surface_height // 5),
                                       configuration.moiety.shape,
                                       configuration.response_map,
                                       self.num_sensors,
                                       orientation=0,
                                       name="extra2",
                                       type=configuration.moiety.type,
                                       molecule_point_symmetry=configuration.moiety.point_symmetry,
                                       substrate_point_symmetry=self.substrate_point_symmetry,
                                       action_space=configuration.action_space))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = [Obstacle(Point(self.surface_width // 2, self.surface_height // 2), random_convex_polygon(), 0)]

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        self.goals.append(Goal(Point(self.surface_width // 2, 4 * self.surface_height // 5), self.shape, 30, self.type))

    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None) -> None:
        corridor_config = options["corridor_config"]
        self.matching = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True, corridor_config=corridor_config).compute_matching()


class FirstPaperExampleEnv(MoleculeEnvironment):
    def __init__(self,
                 mockup_data: Type[MockUpData],
                 **kwargs):
        self.mockup_data = mockup_data
        super().__init__(**kwargs)
    def _parse_molecule_data(self):
        self.data_processor = self.mockup_data(dimensions_x=(-2.1, 2.1),
                                               dimensions_y=(-2.1, 2.1),
                                               step_x=0.3,
                                               step_y=0.3)
        if isinstance(self.data_processor, CircularMockUpData):
            self.shape = Point(0,0).buffer(1.0)
        else:
            self.shape = FePc
        self.molecule_transition_data = self.data_processor.get_molecular_data()
        self.molecule_transition_data.clear_name()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        self.molecules.append(Molecule(Point(8,6), FePc, self.molecule_transition_data, 0, orientation="random", name="i"))
        self.molecules.append(Molecule(Point(43,5), FePc, self.molecule_transition_data, 0, orientation="random", name="j"))
        self.molecules.append(Molecule(Point(20,20), FePc, self.molecule_transition_data, 0, orientation="random", name="k"))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = list()
        self.obstacles.append(Obstacle(Point(12,20), random_convex_polygon(), 0))
        self.obstacles.append(Obstacle(Point(20,10), random_convex_polygon(), 0))

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        self.goals.append(Goal(Point(30,10), FePc, 00))
        self.goals.append(Goal(Point(26,15), FePc, 60))
        self.goals.append(Goal(Point(33,15), FePc, 60))

    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None) -> None:
        corridor_config = options["corridor_config"]
        self.matching = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True, corridor_config=corridor_config).compute_matching()

class DescriptionEnv(MoleculeEnvironment):
    def __init__(self,
                 mockup_data: Type[MockUpData],
                 **kwargs):
        self.mockup_data = mockup_data
        super().__init__(**kwargs)
    def _parse_molecule_data(self):
        self.data_processor = self.mockup_data(dimensions_x=(-2.1, 2.1),
                                               dimensions_y=(-2.1, 2.1),
                                               step_x=0.3,
                                               step_y=0.3)
        if isinstance(self.data_processor, CircularMockUpData):
            self.shape = Point(0,0).buffer(1.0)
        else:
            self.shape = FePc
        self.molecule_transition_data = self.data_processor.get_molecular_data()
        self.molecule_transition_data.clear_name()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        self.molecules.append(Molecule(Point(5,5), FePc, self.molecule_transition_data, 0, orientation="random", name="i"))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = []

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        self.goals.append(Goal(Point(16,5), FePc, 00))

    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None) -> None:
        self.matching = HungarianMatching(self.molecules, self.goals, self.obstacles).compute_matching()

class Figure3Env(MoleculeEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        self.molecule_configuration = self.system_configuration[self.type]
        self.shape = self.molecule_configuration.moiety.shape

    def _parse_molecule_data(self):
        pass

    def _create_initial_distribution(self, seed: Optional[int] = None):
        response_map = self.molecule_configuration.response_map
        point_symmetry = self.molecule_configuration.moiety.point_symmetry
        action_space = self.molecule_configuration.action_space
        self.molecules = list()
        self.molecules.append(Molecule(Point(2.5,6),  self.shape, response_map, 0, orientation=30, name="i",type=self.type,molecule_point_symmetry=point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry,action_space=action_space))
        self.molecules.append(Molecule(Point(8,8), self.shape, response_map, 0, orientation=0, name="j",type=self.type,molecule_point_symmetry=point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry,action_space=action_space))
        #self.molecules.append(Molecule(Point(20,20),self.shape, response_map, 0, orientation="random", name="k",type=self.type,molecule_point_symmetry=point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry,action_space=action_space))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = list()

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        self.goals.append(Goal(Point(12,8), self.shape, 00, self.type))
        self.goals.append(Goal(Point(18,6), self.shape, 00, self.type))
        #self.goals.append(Goal(Point(33,15), self.shape, 60, self.type))

    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None) -> None:
        corridor_config = options["corridor_config"]
        self.matching = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True, corridor_config=corridor_config).compute_matching()

class Figure4GoalPrecedenceEnv(MoleculeEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        self.molecule_configuration = self.system_configuration[self.type]
        self.shape = self.molecule_configuration.moiety.shape

    def _parse_molecule_data(self):
        pass

    def _create_initial_distribution(self, seed: Optional[int] = None):
        response_map = self.molecule_configuration.response_map
        point_symmetry = self.molecule_configuration.moiety.point_symmetry
        action_space = self.molecule_configuration.action_space
        self.molecules = list()
        self.molecules.append(Molecule(Point(2.5,11),  self.shape, response_map, 0, orientation=30, name="i",type=self.type,molecule_point_symmetry=point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry,action_space=action_space))
        self.molecules.append(Molecule(Point(2,7.5), self.shape, response_map, 0, orientation=0, name="j",type=self.type,molecule_point_symmetry=point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry,action_space=action_space))
        #self.molecules.append(Molecule(Point(20,20),self.shape, response_map, 0, orientation="random", name="k",type=self.type,molecule_point_symmetry=point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry,action_space=action_space))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = list()

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        self.goals.append(Goal(Point(8,8), self.shape, 00, self.type))
        self.goals.append(Goal(Point(18,6), self.shape, 00, self.type))
        #self.goals.append(Goal(Point(33,15), self.shape, 60, self.type))

    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None) -> None:
        corridor_config = options["corridor_config"]
        self.matching = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True, corridor_config=corridor_config).compute_matching()
