import gymnasium as gym
import numpy as np

import os
from typing import Optional, Type

from loguru import logger

import shapely
from shapely import Point

from molecule_movement.envs.MoleculeEnvironment import MoleculeEnvironment
from molecule_movement import Molecule, Matching, Goal, Obstacle
from molecule_movement.parsing import MockUpData
from molecule_movement.sampling import CircularSampler
from molecule_movement.shapes import DDNB, TRIANGLE, RECTANGLE, resize
from molecule_movement.matching import HungarianMatching
from molecule_movement.scheduling import SATBasedScheduling


class DebugShortestPathEnv(MoleculeEnvironment):
    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()

    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:
            self.matching = HungarianMatching(self.molecules, self.goals, self.obstacles, respect_obstacles=True).compute_matching()

class DebugShortestPath1Env(DebugShortestPathEnv):
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)
        self._parse_molecule_data()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [Molecule(Point(10,5), DDNB, self.molecule_transition_data, self.num_sensors, orientation=0, name="i")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(30,4), DDNB, 0),
                      Goal(Point(30,6), DDNB, 0)]
    def _set_obstacles(self, seed: Optional[int] = None):
        logger.info("setting obstacles")
        self.obstacles = [Obstacle(Point(20,5), Point(0,0).buffer(0.5), 0)]

