from shapely import Point

from typing import Optional

from loguru import logger


from molecule_movement.envs.MoleculeEnvironment import MoleculeEnvironment
from molecule_movement.Objects import Goal
from molecule_movement.Molecule import Molecule
from molecule_movement.sampling import CircularSampler, Sampler, PoissonDiskSampler
from molecule_movement.matching import HungarianMatching
from molecule_movement.shapes import compute_rectangle



class RectangleEnvironment(MoleculeEnvironment):
    def __init__(
            self,
            width: int,
            height: int,
            num_points: int,
            **kwargs
            ):
        self.width = width
        self.height = height
        self.num_points = num_points
        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.moiety_type = list(self.system_configuration.keys())[0]
        self.molecule_configuration = self.system_configuration[self.moiety_type]
        self.shape = self.molecule_configuration.moiety.shape

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        sampler = CircularSampler(mean=0.6,
                                  scale=0.01,
                                  angle_distribution=lambda x: 1,
                                  x_offset=self.surface_width / 2, y_offset=self.surface_height / 2,
                                  radius=min(self.surface_height / 2, self.surface_width / 2),
                                  min_distance=8,
                                  rejection=self._overlaps_obstacle_func(8),
                                  seed=seed)

        names = self._sample_n_random_names(len(self.goals))
        for i in range(len(self.goals)):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = self.molecule_configuration
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[i],
                                           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_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        goal_x, goal_y = compute_rectangle(self.width, self.height, self.num_points, self.surface_width // 2, self.surface_height // 2)
        shape = self.shape
        moiety_type = self.moiety_type
        for x,y in zip(goal_x, goal_y):
            self.goals.append(Goal(Point(x,y), shape, 0, moiety_type))

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

    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 RectangleUniformInitEnvironment(RectangleEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        print(seed)
        sampler = Sampler(lambda x: 1, # 0.25 < x and x < 0.75, # 1,
                          lambda y: 1, # 0.25 < y and y < 0.75, # 1,
                          min_distance=4.0,
                          rejection=self._overlaps_obstacle_func(8),
                          width=self.surface_width, height=self.surface_height, seed=seed)

        names = list(range(len(self.goals)))
        for i, goal in enumerate(self.goals):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = self.molecule_configuration
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[i],
                                           type=configuration.moiety.type,
                                           molecule_point_symmetry=configuration.moiety.point_symmetry,
                                           substrate_point_symmetry=self.substrate_point_symmetry,
                                           action_space=configuration.action_space))

class RectangleCircularInitEnvironment(RectangleEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        sampler = CircularSampler(mean=0.6,
                                  scale=0.10,
                                  angle_distribution=lambda x: 1,
                                  x_offset=self.surface_width / 2, y_offset=self.surface_height / 2,
                                  radius=min(self.surface_height / 2, self.surface_width / 2),
                                  min_distance=3.5,
                                  rejection=self._overlaps_obstacle_func(4),
                                  seed=seed)

        names = list(range(len(self.goals)))
        for i, goal in enumerate(self.goals):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = self.molecule_configuration
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[i],
                                           type=configuration.moiety.type,
                                           molecule_point_symmetry=configuration.moiety.point_symmetry,
                                           substrate_point_symmetry=self.substrate_point_symmetry,
                                           action_space=configuration.action_space))


class RectanglePoissonDiskInitEnvironment(RectangleEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        sampler = PoissonDiskSampler(width=self.surface_width,
                                     height=self.surface_height,
                                     seed=seed)

        names = list(range(len(self.goals)))
        for i, goal in enumerate(self.goals):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = self.molecule_configuration
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[i],
                                           type=configuration.moiety.type,
                                           molecule_point_symmetry=configuration.moiety.point_symmetry,
                                           substrate_point_symmetry=self.substrate_point_symmetry,
                                           action_space=configuration.action_space))
