import os
from typing import Optional, Type

from loguru import logger

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

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

class LinkingTrainingEnv(MoleculeEnvironment):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)
        assert len(self.system_configuration) == 1
        self.type = list(self.system_configuration.keys())[0]
        self.shape = self.system_configuration[self.type].moiety.shape
        self.response_map = self.system_configuration[self.type].response_map

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        #sampler = Sampler(lambda x: 1,
        #                  lambda y: 1,
        #                  rejection=self._overlaps_obstacle_func(3),
        #                  width=self.surface_width, height=self.surface_height, seed=seed)
        sampler = CircularSampler(mean=0.8,
                                  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=0,
                                  rejection=self._overlaps_obstacle_func(3),
                                  seed=seed)

        molecule_configurations = self.system_configuration
        initial_pos = sampler.sample_position(self.molecules)
        configuration = molecule_configurations[self.type]
        self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y),
                                       configuration.moiety.shape,
                                       self.response_map,
                                       self.num_sensors,
                                       orientation=0,
                                       name="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 _parse_molecule_data(self):
        pass

    def reset(self, seed: Optional[int] = None, options: Optional[dict] = None) -> tuple[dict, dict]:
        if options:
            self.options = options
        self.seed = seed
        self.steps = 0
        self.name_generator = np.random.default_rng(seed=seed)
        self.current_matching_index = 0

        ### In this env, we need to sample the obstacle first to then set the goal
        self._set_obstacles(seed)
        self._create_initial_distribution(seed=seed)
        self._set_goals(seed)
        self.name_to_molecule_map = {molecule.name : molecule for molecule in self.molecules}
        self.simulator = Simulator(self.molecules, self.goals)
        self._initialize_renderer()
        if self.renderer:
            self.renderer.clear(force=True)
            self.render(None)
            self.renderer.update()
        logger.trace("compute matching in reset")
        self._get_matching(seed=seed, options=options)

        #super().reset(seed=seed)

        if self.renderer:
            self.render(None)
            self.renderer.update()
        return self.observation(), {}

    def _set_goals(self, seed: Optional[int] = None) -> None:
        obstacle_pos = self.obstacles[0].position
        obstacle_orientation = self.obstacles[0].orientation
        molecule_pos = self.molecules[0].center

        goal_pos = line_interpolate_point(LineString([obstacle_pos, molecule_pos]), 2 * 1.1 + 0.6)
        self.goals = [Goal(goal_pos, self.shape, obstacle_orientation, type=self.type)]

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        random_angle = self.system_configuration[self.type].angle_symmetry.random_angle()
        self.obstacles = [Obstacle(Point(self.surface_width//2, self.surface_height//2), self.shape, random_angle)]

    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=False, corridor_config=corridor_config)
        except TypeError as e:
            #logger.error(e)
            self.matching_factory = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=False)
        self.matching = self.matching_factory.compute_matching()


    def reached_goal_orientation(self) -> bool:
        molecule_orientation = self.current_molecule.orientation % 90
        goal_orientation = self.current_matching.goal.rotation % 90
        difference = np.abs(molecule_orientation - goal_orientation)
        difference = np.min([difference, 90 - difference])
        return difference <= 7.5

