import os
from typing import Optional, Type

from loguru import logger

from molecule_movement.shapes import QuantumCorral
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, CircularMockUpData
from molecule_movement.sampling import Sampler
from molecule_movement.matching import HungarianMatching
from molecule_movement.shapes import TRIANGLE, RECTANGLE, CIRCLE, resize, FePc
import numpy as np

class QuantumCorralEnvironment(MoleculeEnvironment):
    metadata = {
        "render_modes": ["none", "human"],
    }
    def __init__(
            self,
            **kwargs
    ):
        self.scale = 170
        self.QuantumCorral = QuantumCorral(scale=self.scale, offset=Point(15,10))
        self.goal_positions = self.QuantumCorral.get_points()

        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        configuration = self.system_configuration[self.type]
        self.shape = configuration.moiety.shape
        self.response_map = configuration.response_map
        self.molecule_point_symmetry = configuration.moiety.point_symmetry
        self._action_space = configuration.action_space

        super().__init__(**kwargs)
        #if check_containment:
        #    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(boundary):
        #        raise ValueError(f"Cannot contain word with boundary {boundary=} on surface with dimensions: {self.surface_width}x{self.surface_height}")

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        sampler = Sampler(lambda x: 1,
                          lambda y: 1,
                          min_distance=6,
                          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))
        molecule_configurations = self.system_configuration
        for i, goal in enumerate(self.goals):
            initial_pos = sampler.sample_position(self.molecules)
            configuration = molecule_configurations[self.type]
            self.molecules.append(Molecule(Point(initial_pos.x, initial_pos.y),
                                           self.shape,
                                           self.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[i],
                                           type=self.type,
                                           molecule_point_symmetry=self.molecule_point_symmetry,
                                           substrate_point_symmetry=self.substrate_point_symmetry,
                                           action_space = self._action_space))


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

    def _set_goals(self, seed: Optional[int] = None) -> None:
        offset = 30
        self.goals = [Goal(Point(pos[0] + offset, pos[1] + offset), self.shape, 0, self.type) for pos in self.goal_positions.coords]

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

