import numpy as np
import gymnasium as gym

import shapely
from shapely import Point

import os
from typing import Optional, Type

from loguru import logger

from shapely import LineString

from ..colour_utils import Highlight

from molecule_movement.envs.MoleculeEnvironment import MoleculeEnvironment
from molecule_movement import Goal, Obstacle, Molecule, AngleSymmetry
from molecule_movement.sampling import CircularSampler, Sampler, PoissonDiskSampler
from molecule_movement.matching import RandomMatching, GreedyMatching, HungarianMatching
from molecule_movement.shapes import compute_ellipse, FePc, interpolate_line, points_along_direction, ATOM



class Playground(MoleculeEnvironment):
    def __init__(
            self,
            **kwargs
            ):
        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        self.shape = self.system_configuration[self.type].moiety.shape

    def _parse_molecule_data(self) -> None:
        return super()._parse_molecule_data()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        sampler = Sampler(lambda x: 1,
                          lambda y: y >= 0.65 or y <= 0.35,
                          min_distance=4,
                          rejection=self._overlaps_obstacle_func(6),
                          width=self.surface_width, height=self.surface_height, seed=seed)

        names = list(range(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),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[i],
                                           type=configuration.moiety.type,
                                           molecule_point_symmetry=configuration.moiety.point_symmetry,
                                           substrate_point_symmetry=self.substrate_point_symmetry,
                                           action_space=configuration.action_space))
        for i, goal in enumerate(self.goals[0:2]):
            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,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=f"{names[i]}_",
                                           type=configuration.moiety.type,
                                           molecule_point_symmetry=configuration.moiety.point_symmetry,
                                           substrate_point_symmetry=self.substrate_point_symmetry,
                                           action_space=configuration.action_space))
        #self.molecules.pop()
        self.molecules.append(Molecule(self.goals[-1].position,
                                       configuration.moiety.shape,
                                       configuration.response_map,
                                       self.num_sensors,
                                       orientation=self.initial_orientation,
                                       name="extra",
                                       type=configuration.moiety.type,
                                       molecule_point_symmetry=configuration.moiety.point_symmetry,
                                       substrate_point_symmetry=self.substrate_point_symmetry,
                                       action_space=configuration.action_space))
        self.molecules.append(Molecule(self.goals[-3].position,
                                       configuration.moiety.shape,
                                       configuration.response_map,
                                       self.num_sensors,
                                       orientation=self.initial_orientation,
                                       name="extra1",
                                       type=configuration.moiety.type,
                                       molecule_point_symmetry=configuration.moiety.point_symmetry,
                                       substrate_point_symmetry=self.substrate_point_symmetry,
                                       action_space=configuration.action_space))
        self.molecules.append(Molecule(self.goals[0].position,
                                       configuration.moiety.shape,
                                       configuration.response_map,
                                       self.num_sensors,
                                       orientation=self.initial_orientation,
                                       name="extra2",
                                       type=configuration.moiety.type,
                                       molecule_point_symmetry=configuration.moiety.point_symmetry,
                                       substrate_point_symmetry=self.substrate_point_symmetry,
                                       action_space=configuration.action_space))

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        pts = points_along_direction((3,10), (4,10), 3.0, n=17)
        pts.extend(points_along_direction((3,14), (4,14), 3.0, n=2))

        for p in pts:
            self.goals.append(Goal(p, self.shape, 0, self.type))


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

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




class Playground2(MoleculeEnvironment):
    def __init__(
            self,
            **kwargs
            ):
        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        self.shape = self.system_configuration[self.type].moiety.shape

    def _parse_molecule_data(self) -> None:
        return super()._parse_molecule_data()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()


        positions = [[     70.574,      42.888], [     76.599,      45.588], [     67.775,      22.752], [     42.433,      32.783], [     49.024,      34.092], [     54.584,      37.733], [     64.025,      41.038], [     43.549,      28.633], [     73.883,      25.458], [     57.822,      38.232], [     68.094,      39.596], [     74.303,      41.401], [     66.901,       42.84], [     52.099,      35.588], [     45.466,      33.322], [      79.68,      28.173]]

        names = list(range(len(positions)))
        molecule_configurations = self.system_configuration
        for i, initial_pos in enumerate(positions):
            #initial_pos = positions[i]
            initial_pos_x = initial_pos[0]
            initial_pos_y = initial_pos[1]
            configuration = molecule_configurations[self.type]
            self.molecules.append(Molecule(Point(initial_pos_x, initial_pos_y),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[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 _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()

        pts = points_along_direction((47,26), (72,34), 3.0, n=9)
        for p in pts:
            self.goals.append(Goal(p, self.shape, 00, self.type))
        for p, q in zip(pts, pts[1:]):
            logger.info(p.distance(q))
        for p, q in zip(self.goals, self.goals[1:]):
            logger.info(p.polygon.distance(q.polygon))
        for p, q in zip(self.goals, self.goals[1:]):
            logger.info(p.position.buffer(1.1).distance(q.position.buffer(1.1)))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = list()
        #self.obstacles.append(Obstacle(Point(70.574 - 40, 45.888 - 20), shape=FePc, rotation=0))
        #self.obstacles.append(Obstacle(Point(8,7), shape=FePc, rotation=0))
        #super()._set_obstacles(seed)

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

class Playground3(MoleculeEnvironment):
    def __init__(
            self,
            **kwargs
            ):
        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]
        self.shape = self.system_configuration[self.type].moiety.shape

    def _parse_molecule_data(self) -> None:
        return super()._parse_molecule_data()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()
        names = list(range(len(self.goals)))
        molecule_configurations = self.system_configuration
        configuration = molecule_configurations[self.type]
        #self.molecules.append(Molecule(self.goals[-1].position,
        #                               #ATOM,
        #                               configuration.moiety.shape,
        #                               configuration.response_map,
        #                               self.num_sensors,
        #                               orientation=self.initial_orientation,
        #                               name="positioned",
        #                               type=configuration.moiety.type,
        #                               molecule_point_symmetry=configuration.moiety.point_symmetry,
        #                               substrate_point_symmetry=self.substrate_point_symmetry,
        #                               action_space=configuration.action_space))
        self.molecules.append(Molecule(Point(60, 11),
                                       configuration.moiety.shape,
                                       configuration.response_map,
                                       self.num_sensors,
                                       orientation=self.initial_orientation,
                                       name="extra",
                                       type=configuration.moiety.type,
                                       molecule_point_symmetry=configuration.moiety.point_symmetry,
                                       substrate_point_symmetry=self.substrate_point_symmetry,
                                       action_space=configuration.action_space))

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        pts = points_along_direction((47,10), (51,10), 3.0, n=2)

        for p in pts:
            self.goals.append(Goal(p, self.shape, 0, self.type))


    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = [Obstacle(Point(50,10), shape=FePc, rotation=0)]
        #super()._set_obstacles(seed)

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


class Playground4(MoleculeEnvironment):
    def __init__(
            self,
            **kwargs
            ):
        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.types = list(self.system_configuration.keys())

    def _parse_molecule_data(self) -> None:
        return super()._parse_molecule_data()

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = list()


        positions = [[     70.574,      42.888], [     76.599,      45.588], [     67.775,      22.752], [     42.433,      32.783], [     49.024,      34.092], [     54.584,      37.733], [     64.025,      41.038], [     43.549,      28.633], [     73.883,      25.458], [     57.822,      38.232], [     68.094,      39.596], [     74.303,      41.401], [     66.901,       42.84], [     52.099,      35.588], [     45.466,      33.322], [      79.68,      28.173]]

        names = list(range(len(positions)))
        system_configuration = self.system_configuration
        for i, initial_pos in enumerate(positions):
            try:
                goal_type = self.goals[i].type
            except IndexError:
                goal_type = self.goals[-1].type
            #initial_pos = positions[i]
            initial_pos_x = initial_pos[0]
            initial_pos_y = initial_pos[1]
            configuration = system_configuration[goal_type]
            self.molecules.append(Molecule(Point(initial_pos_x, initial_pos_y),
                                           configuration.moiety.shape,
                                           configuration.response_map,
                                           self.num_sensors,
                                           orientation=self.initial_orientation,
                                           name=names[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 _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()

        pts = points_along_direction((47,26), (72,34), 3.0, n=9)
        t = self.types[0]
        for i, p in enumerate(pts):
            self.goals.append(Goal(p, self.system_configuration[t].moiety.shape, 00, t))
            if i >= len(pts) / 2:
                t = self.types[1]
        for p, q in zip(pts, pts[1:]):
            logger.info(p.distance(q))
        for p, q in zip(self.goals, self.goals[1:]):
            logger.info(p.polygon.distance(q.polygon))
        for p, q in zip(self.goals, self.goals[1:]):
            logger.info(p.position.buffer(1.1).distance(q.position.buffer(1.1)))

    def _set_obstacles(self, seed: Optional[int] = None) -> None:
        self.obstacles = list()
        #self.obstacles.append(Obstacle(Point(70.574 - 40, 45.888 - 20), shape=FePc, rotation=0))
        #self.obstacles.append(Obstacle(Point(8,7), shape=FePc, rotation=0))
        #super()._set_obstacles(seed)

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