import os
from typing import Optional, Type

from loguru import logger

from molecule_movement.shapes import Characters
from shapely import Point, GeometryCollection, Polygon
import shapely

from molecule_movement.envs.MoleculeEnvironment import MoleculeEnvironment
from molecule_movement import Molecule, Matching, Goal
from molecule_movement.parsing import FixedVoltageDataParser, MoleculeDataProcessor, MockUpData
from molecule_movement.sampling import Sampler
from molecule_movement.matching import HungarianMatching
from molecule_movement.shapes import DDNB, TRIANGLE, RECTANGLE, resize
import numpy as np

class LetterEnvironment(MoleculeEnvironment):
    metadata = {
        "render_modes": ["none", "human"],
    }
    def __init__(
            self,
            molecule_transition_data_x: str | os.PathLike,
            molecule_transition_data_y: str | os.PathLike,
            molecule_rotation_data: str | os.PathLike,
            word: str,
            check_containment: bool = True,
            center: bool = True,
            **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.word = word
        self.scale = 40
        self.Letters = Characters(scale=self.scale, spacing=12, max_segment_length=12, characters=self.word)
        if center:
            self.goal_positions = self.Letters.get_positions(self.word, center=(kwargs["surface_width"], kwargs["surface_height"]))
        else:
            self.goal_positions = self.Letters.get_positions(self.word)

        super().__init__(**kwargs)
        if check_containment:
            word_boundary = GeometryCollection(self.goal_positions).convex_hull
            if not Polygon([(0,0), (self.surface_width, 0), (self.surface_width, self.surface_height), (0, self.surface_height), (0,0)]).contains(word_boundary):
                raise ValueError(f"Cannot contain word with boundary {word_boundary=} on surface with dimensions: {self.surface_width}x{self.surface_height}")

    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="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()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        sampler = Sampler(lambda x: 1,
                          lambda y: 1,
                          min_distance=8,
                          width=self.surface_width, height=self.surface_height, seed=seed)
        self.molecules = list()
        for _ in range(len(self.goals)):
            initial_pos = sampler.sample_position(self.molecules)
            angle = np.random.randint(6, size=1)[0]
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y), DDNB, angle * 60, self.molecule_transition_data, self.num_sensors, name=self._sample_random_name()))
        #self.molecules = [Molecule(Point(0,0), DDNB, 0, self.molecule_transition_data, self.num_sensors, name=self._sample_random_name())]

    def _get_matching(self, seed: Optional[int] = None) -> None:
        self.matching = HungarianMatching(self.molecules, self.goals).compute_matching()
    def _set_goals(self, seed: Optional[int] = None) -> None:
        ##goals = [Point(x, y) for line in self.current_letter for (x, y) in line.coords]
        self.goals = [Goal(pos, Point(0,0).buffer(2.0), 0) for pos in self.goal_positions]

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

class MockUpLetterTrainingEnv(LetterEnvironment):
    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.molecule_transition_data = self.data_processor.get_molecular_data()
        self.molecule_transition_data.clear_name()


