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
from molecule_movement.parsing import MockUpData
from molecule_movement.shapes import DDNB


class DebugSATBasedEnv(MoleculeEnvironment):
    """
    A **satisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)
        assert len(self.system_configuration) == 1
        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_action_space = configuration.action_space
        self.moiety_point_symmetry = configuration.moiety.point_symmetry

    def _create_molecule(self, position: Point, name : str) -> Molecule:
        return Molecule(position, self.shape, self.response_map, self.num_sensors, orientation="random", name=name, type=self.type, molecule_point_symmetry=self.moiety_point_symmetry, substrate_point_symmetry=self.substrate_point_symmetry, action_space=self.molecule_action_space, omit_type_in_name=True)

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

class DebugSATBasedSimple1Env(DebugSATBasedEnv):
    """
    A **satisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(10,18), name="i"),
                          self._create_molecule(Point(18,18), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(10,2), self.shape, 0, type=self.type),
                      Goal(Point(2,18), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[0]),
                         Matching(self.molecules[1], self.goals[1])]

class DebugSATBasedSimple2Env(DebugSATBasedEnv):
    """
    A **satisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(10,2),  name="i"),
                          self._create_molecule(Point(18,18), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(10,18), self.shape, 0, type=self.type),
                      Goal(Point(2,18),  self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[0]),
                         Matching(self.molecules[1], self.goals[1])]

class DebugSATBasedSimple3Env(DebugSATBasedEnv):
    """
    A **satisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,10),  name="i"),
                          self._create_molecule(Point(10,10), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(12,10), self.shape, 0, type=self.type),
                      Goal(Point(18,10), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[1], self.goals[0]),
                         Matching(self.molecules[0], self.goals[1])]

class DebugSATBasedSimple4Env(DebugSATBasedEnv):
    """
    An **unsatisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,10),  name="i"),
                          self._create_molecule(Point(18,10), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(11,10), self.shape, 0, type=self.type),
                      Goal(Point(9,10),  self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[0]),
                         Matching(self.molecules[1], self.goals[1])]

class DebugSATBasedSimple5Env(DebugSATBasedEnv):
    """
    A **satisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,10), name="i"),
                          self._create_molecule(Point(18,10), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(12,10), self.shape, 0, type=self.type),
                      Goal(Point(8,10),  self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[1]),
                         Matching(self.molecules[1], self.goals[0])]


class DebugSATBasedSimple6Env(DebugSATBasedEnv):
    """
    An **unsatisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(8,10),  name="i"),
                          self._create_molecule(Point(12,10), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(8,4), self.shape, 0, type=self.type),
                      Goal(Point(12,4),self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[0]),
                         Matching(self.molecules[1], self.goals[1])]

class DebugSATBasedSimple7Env(DebugSATBasedEnv):
    """
    A **unsatisfiable** environment
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(1,10),  name="i"),
                          self._create_molecule(Point(18,10), name="j")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(9,4), self.shape, 0, type=self.type),
                      Goal(Point(11,4), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[0]),
                         Matching(self.molecules[1], self.goals[1])]


class DebugSATBasedNestedConflictsEnv(DebugSATBasedEnv):
    """
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,2),  name="i"),
                          self._create_molecule(Point(4,10), name="j"),
                          self._create_molecule(Point(6,18), name="k")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(20,30), self.shape, 0, type=self.type),
                      Goal(Point(28,34), self.shape, 0, type=self.type),
                      Goal(Point(36,38), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[2]),
                         Matching(self.molecules[1], self.goals[1]),
                         Matching(self.molecules[2], self.goals[0])]


class DebugSATBasedNestedConflicts2Env(DebugSATBasedEnv):
    """
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,2),  name="i"),
                          self._create_molecule(Point(3,7),  name="j"),
                          self._create_molecule(Point(6,14), name="k")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(20,30), self.shape, 0, type=self.type),
                      Goal(Point(28,34), self.shape, 0, type=self.type),
                      Goal(Point(36,38), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[2]),
                         Matching(self.molecules[1], self.goals[1]),
                         Matching(self.molecules[2], self.goals[0])]

class DebugSATBasedNestedConflicts3Env(DebugSATBasedEnv):
    """
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,2),  name="i"),
                          self._create_molecule(Point(4,10), name="j"),
                          self._create_molecule(Point(6,18), name="k"),
                          self._create_molecule(Point(10,4), name="m"),
                          self._create_molecule(Point(18,6), name="n")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(20,30), self.shape, 0, type=self.type),
                      Goal(Point(28,34), self.shape, 0, type=self.type),
                      Goal(Point(36,38), self.shape, 0, type=self.type),
                      Goal(Point(34,28), self.shape, 0, type=self.type),
                      Goal(Point(36,34), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[2]),
                         Matching(self.molecules[1], self.goals[1]),
                         Matching(self.molecules[2], self.goals[0]),
                         Matching(self.molecules[3], self.goals[4]),
                         Matching(self.molecules[4], self.goals[3])]


class DebugSATBasedNestedConflicts4Env(DebugSATBasedEnv):
    """
    """
    def __init__(
            self,
            **kwargs
            ):
        super().__init__(**kwargs)

    def _create_initial_distribution(self, seed: Optional[int] = None):
        self.molecules = [self._create_molecule(Point(2,2),  name="i"),
                          self._create_molecule(Point(4,10), name="j"),
                          self._create_molecule(Point(10,4), name="m")]
    def _set_goals(self, seed: Optional[int] = None) -> None:
        self.goals = [Goal(Point(20,30), self.shape, 0, type=self.type),
                      Goal(Point(28,34), self.shape, 0, type=self.type),
                      Goal(Point(28,28), self.shape, 0, type=self.type)]
    def _get_matching(self, options: Optional[dict] = None, seed: Optional[int] = None):
        self.matching = [Matching(self.molecules[0], self.goals[1]),
                         Matching(self.molecules[1], self.goals[0]),
                         Matching(self.molecules[2], self.goals[2])]

