from gymnasium.envs.registration import register

from .logging import log_and_raise, deprecated
import warnings
warnings.filterwarnings("once", "", DeprecationWarning, module=r"^molecule_movement\.")

from molecule_movement.AngleSymmetry import AngleSymmetry
from molecule_movement.Molecule import Molecule, MoleculeExperiment
from molecule_movement.Matching import Matching, Pair
from molecule_movement.Objects import *

from .constants import configurations as system_configurations

from dataclasses import dataclass

@dataclass(frozen=True)
class EllipseConfiguration:
    ellipse_width: int
    ellipse_height: int
    num_ellipse_points: int
    surface_width: int
    surface_height: int

@dataclass(frozen=True)
class RectangleConfiguration:
    width: int
    height: int | None = None
    num_points: int = 0
    surface_width: int = 0
    surface_height: int | None = None

    def __post_init__(self) -> None:
        if self.width <= 0:
            raise ValueError("width must be positive")
        if self.num_points <= 0:
            raise ValueError("num_points must be positive")
        if self.surface_width <= 0:
            raise ValueError("surface_width must be positive")

        if self.height is None:
            object.__setattr__(self, "height", self.width)
        elif self.height <= 0:
            raise ValueError("height must be positive")

        if self.surface_height is None:
            object.__setattr__(self, "surface_height", self.surface_width)
        elif self.surface_height <= 0:
            raise ValueError("surface_height must be positive")

import os, sys
d = os.path.dirname(sys.modules[__name__].__file__)

from loguru import logger
logger.disable(__name__)


register(
  id='MoleculeMovement/CiRc-MockUp-Star-40x40-v0',
  entry_point='molecule_movement.envs:StarTrainingEnv',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)

register(
  id='MoleculeMovement/FePc-MockUp-Star-40x40-v0',
  entry_point='molecule_movement.envs:StarTrainingEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)

register(
  id='MoleculeMovement/FePc-MockUp-HighVar-Star-40x40-v0',
  entry_point='molecule_movement.envs:StarTrainingEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-HighVar-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)

register(
  id='MoleculeMovement/CiRc-MockUp-Star-100x100-v0',
  entry_point='molecule_movement.envs:StarTrainingEnv',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":100,
          "surface_width":100,
          }
)

register(
  id='MoleculeMovement/FePc-MockUp-Star-100x100-v0',
  entry_point='molecule_movement.envs:StarTrainingEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":100,
          "surface_width":100,
          }
)

register(
  id='MoleculeMovement/CircMockUpPositioning-10x10-v0',
  entry_point='molecule_movement.envs:MockUpPositioningTrainingEnv',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
    "surface_height":10,
    "surface_width":10,
    }
)

register(
  id='MoleculeMovement/FePcMockUpPositioning-10x10-v0',
  entry_point='molecule_movement.envs:MockUpPositioningTrainingEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
    "surface_height":10,
    "surface_width":10,
    }
)

################# EllipseEnvironments #################

ELLIPSE_CONFIGURATIONS=[
            #EllipseConfiguration(29, 23, 12, 70, 60),
            #EllipseConfiguration(9, 7, 6, 40, 40),
            #EllipseConfiguration(18, 10, 10, 50, 50),
            #EllipseConfiguration(28, 23, 20, 50, 50),
            #EllipseConfiguration(40, 35, 30, 100, 100),
            #EllipseConfiguration(83, 57, 50, 200, 200),
            #EllipseConfiguration(150, 110, 100, 300, 300),

            #EllipseConfiguration(60, 50, 20, 100, 100),
            #EllipseConfiguration(50, 40, 20, 100, 100),
            #EllipseConfiguration(40, 30, 20, 100, 100),
            #EllipseConfiguration(30, 28, 20, 100, 100),
            #EllipseConfiguration(80,   60,  50, 140, 140),

            #EllipseConfiguration(123,  99,  50, 160, 160),
            #EllipseConfiguration(242, 197, 100, 370, 250),
            #EllipseConfiguration(368, 296, 150, 480, 400),
            #EllipseConfiguration(490, 395, 200, 580, 520),
            #EllipseConfiguration(749, 592, 300, 850, 750),
            #EllipseConfiguration(986, 793, 400, 1050, 900),
            #EllipseConfiguration(1210, 995, 500, 1250, 1050),

            #EllipseConfiguration(108,  85,  50, 160, 120),
            #EllipseConfiguration(220, 175, 100, 320, 220),
            #EllipseConfiguration(330, 265, 150, 400, 360),
            #EllipseConfiguration(440, 350, 200, 560, 480),
            #EllipseConfiguration(680, 545, 300, 750, 650),
            ##EllipseConfiguration(880, 700, 400, 1050, 900),
            ##EllipseConfiguration(1080, 870, 500, 1250, 1050),

            #EllipseConfiguration(98,   75,  50, 108, 85),
            #EllipseConfiguration(190, 145, 100, 200, 155),
            #EllipseConfiguration(285, 218, 150, 290, 228),
            #EllipseConfiguration(380, 290, 200, 390, 300),
            #EllipseConfiguration(570, 435, 300, 580, 445),
            #EllipseConfiguration(760, 580, 400, 1020, 700),
            #EllipseConfiguration(1000, 760, 500, 1220, 900),
            #EllipseConfiguration(90,   68,  50, 120, 105),
            #EllipseConfiguration(180, 134, 100, 220, 165),
            #EllipseConfiguration(267, 197, 150, 290, 228),
            #EllipseConfiguration(350, 260, 200, 400, 300),
            #EllipseConfiguration(530, 390, 300, 580, 445),

            # HoneyComb Sizes
            EllipseConfiguration(70,   48,  36, 110, 90),
            EllipseConfiguration(174, 128,  96, 205, 160),
            EllipseConfiguration(307, 237, 180, 340, 270),
            EllipseConfiguration(515, 375, 288, 545, 410),
            EllipseConfiguration(700, 560, 420, 730, 590),
        ]
for name, system_configuration in system_configurations.items():
    name = name.replace("(","").replace(")","")
    for initial_distribution in ["Uniform", "Circular", "PoissonDisk"]:
        for c in ELLIPSE_CONFIGURATIONS:
            register(
              id=f'MoleculeMovement/{name}-{initial_distribution}-Ellipse{c.ellipse_width}x{c.ellipse_height}N{c.num_ellipse_points}-{c.surface_width}x{c.surface_height}-v0',
              entry_point=f'molecule_movement.envs:MockUpEllipse{initial_distribution}InitEnvironment',
              kwargs={"system_configuration": system_configuration,
                      "surface_height":c.surface_height,
                      "surface_width":c.surface_width,
                      "ellipse_width":c.ellipse_width,
                      "ellipse_height":c.ellipse_height,
                      "num_ellipse_points":c.num_ellipse_points,
                      }
            )


RECTANGLE_CONFIGURATIONS=[
    RectangleConfiguration(40,  None, 6, 90, None),
    RectangleConfiguration(40,  None, 36, 90, None),
    RectangleConfiguration(110, None, 96, 180, None),
    RectangleConfiguration(220, None, 180, 310, None),
    RectangleConfiguration(360, None, 288, 470, None),
    RectangleConfiguration(520, None, 420, 640, None),
        ]

for name, system_configuration in system_configurations.items():
    name = name.replace("(", "").replace(")", "")
    for initial_distribution in ["Uniform", "Circular", "PoissonDisk"]:
        for c in RECTANGLE_CONFIGURATIONS:

            if c.width == c.height:
                shape_part = f"Square{c.width}N{c.num_points}"
            else:
                shape_part = f"Rectangle{c.width}x{c.height}N{c.num_points}"

            register(
                id=(
                    f"MoleculeMovement/"
                    f"{name}-{initial_distribution}-"
                    f"{shape_part}-"
                    f"{c.surface_width}x{c.surface_height}-v0"
                ),
                entry_point=f"molecule_movement.envs:Rectangle{initial_distribution}InitEnvironment",
                kwargs={
                    "system_configuration": system_configuration,
                    "surface_height": c.surface_height,
                    "surface_width": c.surface_width,
                    "width": c.width,
                    "height": c.height,
                    "num_points": c.num_points,
                },
            )

register(
  id=f'MoleculeMovement/FePc-Au111-MockUp-Uniform-SmallHoneyComb-v0',
  entry_point=f'molecule_movement.envs:UniformHoneyCombEnv',
  kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":150,
          "surface_width":200,
          "honeycomb_cols":1,
          "honeycomb_rows":1
          }
)

register(
  id=f'MoleculeMovement/CiRc-Au111-MockUp-Uniform-SmallHoneyComb-v0',
  entry_point=f'molecule_movement.envs:UniformHoneyCombEnv',
  kwargs={"system_configuration": system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":150,
          "surface_width":200,
          "honeycomb_cols":1,
          "honeycomb_rows":1
          }
)

for colsrows, (width, height) in zip(range(1,6), [(90,90),(150,150),(225,225),(270,270), (330,330)]):
    for config in ["FePc-Au(111)-MockUp", "FePc-Au(111)-HighVar-MockUp", "CiRc-Au(111)-MockUp"]:
        name = config.replace("(","").replace(")","")
        register(
          id=f'MoleculeMovement/{name}-Uniform-HoneyComb-{colsrows}-v0',
          entry_point=f'molecule_movement.envs:UniformHoneyCombEnv',
          kwargs={"system_configuration": system_configurations[config],
                  "surface_height":height,
                  "surface_width":width,
                  "honeycomb_cols":colsrows,
                  "honeycomb_rows":colsrows
                  }
        )

register(
  id=f'MoleculeMovement/FePc-Au111-MockUp-Uniform-HoneyComb-v0',
  entry_point=f'molecule_movement.envs:UniformHoneyCombEnv',
  kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":330,
          "surface_width":330,
          "honeycomb_cols":5,
          "honeycomb_rows":5
          }
)

register(
  id=f'MoleculeMovement/CiRc-MockUp-Uniform-HoneyComb-v0',
  entry_point=f'molecule_movement.envs:UniformHoneyCombEnv',
  kwargs={"system_configuration": system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":330,
          "surface_width":330,
          "honeycomb_cols":5,
          "honeycomb_rows":5
          }
)

register(
  id='MoleculeMovement/Debug-Silver-TwoCorrals-Uniform-50x100-v0',
  entry_point='molecule_movement.envs:MockUpSilverPlateTwoCorralsUniformInitEnvironment',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":50,
          "surface_width":100,
          }
)

register(
    id='MoleculeMovement/Letter-50x50',
    entry_point='molecule_movement.envs:MockUpLetterTrainingEnv',
    kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
            "surface_height":50,
            "surface_width":50,
            }
)

register(
    id='MoleculeMovement/CiRc-Quantum-Corral-250x250',
    entry_point='molecule_movement.envs:QuantumCorralEnvironment',
    kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
            "surface_height":250,
            "surface_width":250,
            }
)

register(
    id='MoleculeMovement/FePc-Quantum-Corral-250x250',
    entry_point='molecule_movement.envs:QuantumCorralEnvironment',
    kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
            "surface_height":250,
            "surface_width":250,
            }
)

register(
    id='MoleculeMovement/FePc-HighVar-Quantum-Corral-250x250',
    entry_point='molecule_movement.envs:QuantumCorralEnvironment',
    kwargs={"system_configuration":system_configurations["FePc-Au(111)-HighVar-MockUp"],
            "surface_height":250,
            "surface_width":250,
            }
)

register(
  id='MoleculeMovement/FePc-DoubleEllipse-40x40-v0',
  entry_point='molecule_movement.envs:MockUpFePcDoubleEllipseEnvironment',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)

#register(
#  id='MoleculeMovement/Testing-Measurements-v0',
#  entry_point='molecule_movement.envs:TestingMeasurementsEnv',
#  kwargs={"orientation_map":os.path.join(d, "data/orientation_map.pickle"),
#          "translation_map":os.path.join(d, "data/translation_map.pickle"),
#          "surface_height":40,
#          "surface_width":40,
#          }
#)

#register(
#  id='MoleculeMovement/Measurement-Training-v0',
#  entry_point='molecule_movement.envs:OrientationPositioningEnv',
#  kwargs={"system_configuration":system_configurations["FePc-Au(111)"],
#          "surface_height":10,
#          "surface_width":10,
#          }
#)
#
#register(
#  id='MoleculeMovement/CoPc-Measurement-Training-v0',
#  entry_point='molecule_movement.envs:OrientationPositioningEnv',
#  kwargs={"system_configuration":system_configurations["CoPc-Au(111)"],
#          "surface_height":10,
#          "surface_width":10,
#          }
#)
#
#
#register(
#  id='MoleculeMovement/CoPc-new-data-Measurement-Training-v0',
#  entry_point='molecule_movement.envs:OrientationPositioningEnv',
#  kwargs={"system_configuration":system_configurations["CoPc-Au(111)-new_data"],
#          "surface_height":10,
#          "surface_width":10,
#          }
#)
#register(
#  id='MoleculeMovement/LateralFePc-StarTraining-30x30-v0',
#  entry_point='molecule_movement.envs:LateralFePcStarTrainingEnv',
#  kwargs={"system_configuration":system_configurations["FePc-Au(111)"],
#          "surface_height":30,
#          "surface_width":30,
#          }
#)
#
#register(
#  id='MoleculeMovement/Linking-Training-v0',
#  entry_point='molecule_movement.envs:LinkingTrainingEnv',
#  kwargs={"system_configuration":system_configurations["CoPc-Au(111)"],
#          "surface_height":10,
#          "surface_width":10,
#          }
#)
#
#register(
#  id='MoleculeMovement/CoPc-new-data-Linking-Training-v0',
#  entry_point='molecule_movement.envs:LinkingTrainingEnv',
#  kwargs={"system_configuration":system_configurations["CoPc-Au(111)-new_data"],
#          "surface_height":10,
#          "surface_width":10,
#          }
#)
#
#register(
#  id='MoleculeMovement/FePc-Au111-MockUp-Linking-Training-v0',
#  entry_point='molecule_movement.envs:LinkingTrainingEnv',
#  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
#          "surface_height":15,
#          "surface_width":15,
#          }
#)

########################### SATBasedScheduling Debug Environments ###########################

for i in list(range(1,9)):
    register(
      id=f'MoleculeMovement/DebugSATBasedSimple{i}Env-20x20-v0',
      entry_point=f'molecule_movement.envs:DebugSATBasedSimple{i}Env',
      kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
              "surface_height":20,
              "surface_width":20,
              }
    )

register(
  id=f'MoleculeMovement/DebugSATBasedNestedConflictsEnv-20x20-v0',
  entry_point=f'molecule_movement.envs:DebugSATBasedNestedConflictsEnv',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)

register(
  id=f'MoleculeMovement/DebugSATBasedNestedConflicts2Env-20x20-v0',
  entry_point=f'molecule_movement.envs:DebugSATBasedNestedConflicts2Env',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)

register(
  id=f'MoleculeMovement/DebugShortestPath1Env-10x40-v0',
  entry_point=f'molecule_movement.envs:DebugShortestPath1Env',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":10,
          "surface_width":40,
          }
)

#register(
#  id='MoleculeMovement/SATSchedulingExample-60x60-v0',
#  entry_point='molecule_movement.envs:SATSchedulingExampleEnv',
#  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
#          "surface_height":60,
#          "surface_width":60,
#          }
#)

register(
  id='MoleculeMovement/SATSchedulingExample-25x35-v0',
  entry_point='molecule_movement.envs:SATSchedulingExampleEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":25,
          "surface_width":35,
          }
)

register(
  id='MoleculeMovement/CorridorExample-40x20-v0',
  entry_point='molecule_movement.envs:CorridorExampleEnv',
  kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":20,
          }
)


register(
  id='MoleculeMovement/FirstPaperExample-50x25-v0',
  entry_point='molecule_movement.envs:FirstPaperExampleEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":25,
          "surface_width":50,
          }
)


register(
  id='MoleculeMovement/Figure3Env-20x15-v0',
  entry_point='molecule_movement.envs:Figure3Env',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":15,
          "surface_width":20,
          }
)

register(
  id='MoleculeMovement/Figure4GoalPrecedenceEnv-20x15-v0',
  entry_point='molecule_movement.envs:Figure4GoalPrecedenceEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":15,
          "surface_width":20,
          }
)



register(
  id='MoleculeMovement/Description-20x10-v0',
  entry_point='molecule_movement.envs:DescriptionEnv',
  kwargs={"system_configuration":system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":10,
          "surface_width":20,
          }
)

register(
  id=f'MoleculeMovement/DebugSATBasedNestedConflicts3Env-20x20-v0',
  entry_point=f'molecule_movement.envs:DebugSATBasedNestedConflicts3Env',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)


register(
  id=f'MoleculeMovement/DebugSATBasedNestedConflicts4Env-20x20-v0',
  entry_point=f'molecule_movement.envs:DebugSATBasedNestedConflicts4Env',
  kwargs={"system_configuration":system_configurations["CiRc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)



register(
  id=f'MoleculeMovement/Playground4-v0',
  entry_point=f'molecule_movement.envs:Playground4',
  kwargs={"system_configuration": [system_configurations["FePc-Au(111)-MockUp"],system_configurations["CiRc-Au(111)-MockUp"]],
  #kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)




register(
  id=f'MoleculeMovement/Playground-v0',
  entry_point=f'molecule_movement.envs:Playground',
  kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":20,
          "surface_width":100,
          }
)

register(
  id=f'MoleculeMovement/Playground2-v0',
  entry_point=f'molecule_movement.envs:Playground2',
  kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":40,
          "surface_width":40,
          }
)


register(
  id=f'MoleculeMovement/Playground3-v0',
  entry_point=f'molecule_movement.envs:Playground3',
  kwargs={"system_configuration": system_configurations["FePc-Au(111)-MockUp"],
          "surface_height":20,
          "surface_width":100,
          }
)
