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
from molecule_movement import Goal, Obstacle, Molecule
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 DDNB, FePc, resize, compute_ellipse, simplify_and_segmentize_polygon



class FePcDoubleEllipseEnvironment(MoleculeEnvironment):
    def __init__(
            self,
            molecule_transition_data_x: str | os.PathLike,
            molecule_transition_data_y: str | os.PathLike,
            molecule_rotation_data: str | os.PathLike,
            **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
        super().__init__(**kwargs)

    def _parse_molecule_data(self):
        self.data_processor = MoleculeDataProcessor(FixedVoltageDataParser(self.molecule_transition_data_x, self.molecule_transition_data_y, self.molecule_rotation_data, voltage=1700, name="FePc"),
                                                    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()

    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=1,
        #                          rejection=self._overlaps_obstacle_func(1),
        #                          seed=seed)
        sampler = Sampler(lambda x: 1,# if x < 0.2 or x > 0.8 else 0,
                          lambda y: 1,
                          min_distance=3,
                          #rejection=self._overlaps_goal_func(3),
                          width=self.surface_width, height=self.surface_height, seed=seed)

        names = self._sample_n_random_names(len(self.goals))
        for i in range(len(self.goals)):
        #for i in range(min(20, len(self.goals))):
            initial_pos = sampler.sample_position(self.molecules)
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y), FePc, 0, self.molecule_transition_data, self.num_sensors, name=names[i]))


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

        goal_x, goal_y = compute_ellipse(22.4, 18.4, 24, self.surface_width // 2, self.surface_height // 2)
        inner_goal_x, inner_goal_y = compute_ellipse(14, 10, 14, self.surface_width // 2, self.surface_height // 2)
        #g_x, g_y = goal_x, goal_y
        g_x, g_y = inner_goal_x, inner_goal_y
        for x,y in zip(np.concatenate((inner_goal_x, goal_x)), np.concatenate((inner_goal_y, goal_y))):
        #for x,y in zip(g_x, g_y):
            self.goals.append(Goal(Point(x,y), FePc, 0))


    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        sampler = Sampler(lambda x: 1,
                          lambda y: 1,
                          rejection=self._overlaps_goal_func(0),
                          width=self.surface_width, height=self.surface_height, seed=seed)
        self.obstacles = list()
        #for _ in range(3):
        #    self.obstacles.append(Obstacle(sampler.sample_position([]), Point(0,0).buffer(1.2), 0))

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

class MockUpFePcDoubleEllipseEnvironment(FePcDoubleEllipseEnvironment):
    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)
        #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,
        #                                       dimensions_dest_x=(-2.1, 2.1),
        #                                       dimensions_dest_y=(-2.1, 2.1),
        #                                       step_dest_x=0.3,
        #                                       step_dest_y=0.3)
        self.molecule_transition_data = self.data_processor.get_molecular_data()
        self.molecule_transition_data.clear_name()

