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
from molecule_movement.sampling import CircularSampler, Sampler, PoissonDiskSampler
from molecule_movement.matching import RandomMatching, GreedyMatching, HungarianMatching
from molecule_movement.shapes import compute_ellipse, FePc, CIRCLE, honeycomb, hexagon, remove_closing_point



class HoneyCombEnv(MoleculeEnvironment):
    def __init__(
            self,
            honeycomb_cols: int,
            honeycomb_rows: int,
            **kwargs
            ):
        self.honeycomb_cols = honeycomb_cols
        self.honeycomb_rows = honeycomb_rows
        self.initial_orientation = "random"
        super().__init__(**kwargs)
        self.type = list(self.system_configuration.keys())[0]

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

    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = list()
        orientation = "pointy"

        # for smaller figure
        #hexagon_poly = hexagon(width=10.5, height=13.5, orientation=orientation)
        #honeycomb_x, honeycomb_y = honeycomb(cols=self.honeycomb_cols, rows=self.honeycomb_rows, hex_w=40.5, hex_h=61.0, center=(self.surface_height / 2, self.surface_width / 2), orientation=orientation)

        hexagon_poly = hexagon(width=12.5, height=16.5, orientation=orientation)
        honeycomb_x, honeycomb_y = honeycomb(cols=self.honeycomb_cols, rows=self.honeycomb_rows, hex_w=47.5, hex_h=68.0, center=(self.surface_height / 2, self.surface_width / 2), orientation=orientation)

        goal_x, goal_y = list(), list()

        for x, y in zip(honeycomb_x, honeycomb_y):
            #logger.info(f"{x=}, {y=}")
            xs, ys = remove_closing_point(shapely.affinity.translate(hexagon_poly, x, y).exterior.xy)
            #logger.info(len(xs))
            goal_x.extend(xs)
            goal_y.extend(ys)


        assert len(self.system_configuration) == 1
        shape = self.system_configuration[self.type].moiety.shape

        for x,y in zip(goal_x, goal_y):
            self.goals.append(Goal(Point(x,y), 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)
        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()

class UniformHoneyCombEnv(HoneyCombEnv):
    def __init__(self,
                 **kwargs):
        super().__init__(**kwargs)

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

        names = list(range(len(self.goals)*2))
        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))
