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.parsing import FixedVoltageDataParser, MoleculeDataProcessor, MockUpData, FePcMockUpData, CircularMockUpData
from molecule_movement import Goal, Obstacle, Molecule
from molecule_movement.sampling import CircularSampler, Sampler, PoissonDiskSampler
from molecule_movement.matching import RandomMatching, GreedyMatching, HungarianMatching
from molecule_movement.scheduling import SATBasedScheduling, SortScheduling
from molecule_movement.shapes import RECTANGLE, TRIANGLE, DDNB, resize, compute_ellipse, simplify_and_segmentize_polygon, FePc, CIRCLE
from molecule_movement.constants import MoietyType



class EllipseEnvironment(MoleculeEnvironment):
    def __init__(
            self,
            #molecule_transition_data_x: str | os.PathLike,
            #molecule_transition_data_y: str | os.PathLike,
            #molecule_rotation_data: str | os.PathLike,
            ellipse_width: int,
            ellipse_height: int,
            num_ellipse_points: int,
            **kwargs
            ):
        #self.molecule_transition_data_x = molecule_transition_data_x
        #self.molecule_transition_data_y = molecule_transition_data_y
        #self.molecule_rotation_data     = molecule_rotation_data
        self.ellipse_width = ellipse_width
        self.ellipse_height = ellipse_height
        self.num_ellipse_points = num_ellipse_points
        self.initial_orientation = "random"
        #self.shape = FePc
        super().__init__(**kwargs)


    def _parse_molecule_data(self):
        pass
        #self.data_processor = MoleculeDataProcessor(FixedVoltageDataParser(self.molecule_transition_data_x, self.molecule_transition_data_y, self.molecule_rotation_data, voltage=1700, name="DDNB"),
        #                                            dimensions_x=(-2.1, 2.1),
        #                                            dimensions_y=(-2.1, 2.1),
        #                                            step_x=0.3,
        #                                            step_y=0.3)
        #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()
        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)
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y), FePc, self.molecule_transition_data, 0, orientation=self.initial_orientation, name=names[i]))

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

        goal_x, goal_y = compute_ellipse(self.ellipse_width, self.ellipse_height, self.num_ellipse_points, self.surface_width // 2, self.surface_height // 2)
        assert len(self.system_configuration) == 1
        type = list(self.system_configuration.keys())[0]
        shape = self.system_configuration[type].moiety.shape

        for x,y in zip(goal_x, goal_y):
            self.goals.append(Goal(Point(x,y), shape, 0, 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 MockUpEllipseEnvironment(EllipseEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)

class MockUpEllipseUniformInitEnvironment(MockUpEllipseEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        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 = self._sample_n_random_names(len(self.goals) *2)
        names = list(range(len(self.goals)*2))
        molecule_configurations = self.system_configuration
        #for i, goal in enumerate(self.goals + self.goals):
        for i, goal in enumerate(self.goals):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = molecule_configurations[goal.type]
            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 MockUpEllipseCircularInitEnvironment(MockUpEllipseEnvironment):
    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)*2))
        molecule_configurations = self.system_configuration
        #for i, goal in enumerate(self.goals + self.goals):
        for i, goal in enumerate(self.goals):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = molecule_configurations[goal.type]
            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 MockUpEllipsePoissonDiskInitEnvironment(MockUpEllipseEnvironment):
    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 = self._sample_n_random_names(len(self.goals))
        positions = sampler.sample_positions(len(self.goals))
        for i in range(len(self.goals)):
            initial_pos = positions[i]
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y), self.shape, self.molecule_transition_data, 0, orientation=self.initial_orientation, name=names[i]))
