import gymnasium as gym
import numpy as np

import os
from typing import Optional, Type

from loguru import logger
from molecule_movement.logging import log_and_raise

import shapely
from shapely import Point

from molecule_movement.sampling.Sampler import pol2cart

from molecule_movement.envs.MoleculeEnvironment import MoleculeEnvironment
from molecule_movement import Molecule, Goal, Matching
from molecule_movement.parsing import FixedVoltageDataParser, MoleculeDataProcessor, MockUpData, LateralMockUpData, VerticalMockUpData
from molecule_movement.sampling import CircularSampler
from molecule_movement.shapes import DDNB, TRIANGLE, RECTANGLE, resize


class MeasurementSamplingEnv(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

        self.current_goal_index = 0

        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="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):
        self.molecules = list()
        self.initial_pos = Point(self.surface_width /2, self.surface_height / 2)
        angle = np.random.randint(6, size=1)[0]
        self.molecules = [Molecule(self.initial_pos, DDNB, 60 * angle, self.molecule_transition_data, self.num_sensors, name=self._sample_random_name())]

    def _set_goals(self, seed: Optional[int] = None) -> None:
        scaling = 5
        orientation_step = 30
        self.goals = list()
        for _ in range(3):
            angle = np.random.random_sample() * 2*np.pi
            radius = np.random.randint(1, high=4, size=1)[0] * scaling
            orientation = np.random.randint(1, high=4, size=1)[0] * orientation_step
            position = shapely.affinity.translate(Point(*pol2cart(angle, radius)), self.surface_width / 2, self.surface_height / 2)
            self.goals.append(Goal(position, Point(0,0).buffer(1), orientation))
        self.center_goal = Goal(Point(self.surface_width / 2, self.surface_height / 2), Point(0,0).buffer(1), np.random.randint(1, high=4, size=1)[0] * orientation_step)

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

    def _get_matching(self, seed: Optional[int] = None):
        #self.matching = [Matching(self.molecules[0], self.goals[0])]
        self.matching = list()
        for goal in self.goals:
            self.matching.append(Matching(self.molecules[0], goal))
            self.matching.append(Matching(self.molecules[0], self.center_goal))
