from dataclasses import dataclass, field
from typing import Callable, Final, Iterable, Mapping, Tuple
from enum import Enum

from molecule_movement.AngleSymmetry import AngleSymmetry
from molecule_movement.parsing import MockUpData, FePcMockUpData, CircularMockUpData, MoleculeActionSpace, MoleculeDataProcessor, PickleDataParser, LateralMoleculeDataProcessor
from molecule_movement.shapes import *

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

class MoietyType(Enum):
    CoPc = "CoPc"
    FePc = "FePc"
    Circular = "CiRc"

class SubstrateType(Enum):
    Au111 = "Au(111)"
    Au101 = "Au(101)"
    Ag111 = "Ag(111)"

@dataclass(frozen=True, slots=True)
class MoleculeConfig:
    type: MoietyType
    point_symmetry: int
    shape: Polygon

@dataclass(frozen=True, slots=True)
class SubstrateConfig:
    type: SubstrateType

@dataclass(frozen=True, slots=True)
class SystemConfig:
    moiety: MoleculeConfig
    substrate: SubstrateConfig
    substrate_point_symmetry: int
    data: MockUpData | MoleculeDataProcessor
    variant: str = None
    angle_symmetry: AngleSymmetry = field(init=False, repr=False)

    def __post_init__(self):
        value = AngleSymmetry(
            molecule_point_symmetry=self.moiety.point_symmetry,
            substrate_point_symmetry=self.substrate_point_symmetry,
        )
        object.__setattr__(self, "angle_symmetry", value)

    @property
    def action_space(self) -> MoleculeActionSpace:
        return self.data.action_space

    @property
    def response_map(self) -> MoleculeActionSpace:
        return self.data.get_response_map()

    @property
    def key(self) -> Tuple[MoietyType, SubstrateType]:
        return (self.moiety.type, self.substrate.type)

    @property
    def id(self) -> str:
        return f"{self.moiety.type.value}-{self.substrate.type.value}{f'-{self.variant}' if self.variant else ''}"

def make_system_config(
    moiety: MoleculeConfig,
    substrate: SubstrateConfig,
    substrate_point_symmetry: int,
    data_factory: Callable[[], MockUpData | MoleculeDataProcessor],
    variant: str = None
) -> SystemConfig:
    return SystemConfig(moiety=moiety, substrate=substrate, substrate_point_symmetry=substrate_point_symmetry, data=data_factory(), variant=variant)





FePcConfig: Final = MoleculeConfig(
    MoietyType.FePc, point_symmetry=4, shape=FePc
)
CoPcConfig: Final = MoleculeConfig(
    MoietyType.CoPc, point_symmetry=4, shape=FePc
)
CircularConfig: Final = MoleculeConfig(
    MoietyType.Circular, point_symmetry=360, shape=CIRCLE
)

Au111: Final = SubstrateConfig(SubstrateType.Au111)
Au101: Final = SubstrateConfig(SubstrateType.Au101)


_PRESETS: Final[Iterable[tuple[MoleculeConfig, SubstrateConfig, Callable[[], MockUpData | MoleculeDataProcessor], str | None]]] = [
    (
        FePcConfig,
        Au111,
        6,
        lambda: FePcMockUpData(
            dimensions_x=(-2.1, 2.1), dimensions_y=(-2.1, 2.1), step_x=0.3, step_y=0.3
        ),
        "MockUpBig"
    ),
    (
        CircularConfig,
        Au111,
        6,
        lambda: CircularMockUpData(
            dimensions_x=(-2.1, 2.1), dimensions_y=(-2.1, 2.1), step_x=0.3, step_y=0.3
        ),
        "MockUpBig"
    ),
    (
        FePcConfig,
        Au111,
        6,
        lambda: FePcMockUpData(
            dimensions_x=(-1.5, 1.5), dimensions_y=(-1.5, 1.5), step_x=0.3, step_y=0.3
        ),
        "MockUp"
    ),
    (
        FePcConfig,
        Au111,
        6,
        lambda: FePcMockUpData(
            dimensions_x=(-1.5, 1.5), dimensions_y=(-1.5, 1.5), step_x=0.3, step_y=0.3, cov=0.1
        ),
        "HighVar-MockUp"
    ),
    (
        CircularConfig,
        Au111,
        6,
        lambda: CircularMockUpData(
            dimensions_x=(-1.5, 1.5), dimensions_y=(-1.5, 1.5), step_x=0.3, step_y=0.3
        ),
        "MockUp"
    ),
]

_ALL_SYSTEM_CONFIGS: Final[tuple[SystemConfig, ...]] = tuple(
    make_system_config(moiety, substrate, substrate_point_symmetry, data_factory, variant) for (moiety, substrate, substrate_point_symmetry, data_factory, variant) in _PRESETS
)

REGISTRY_BY_TUPLE: Final[Mapping[tuple[MoietyType, SubstrateType], SystemConfig]] = {
    cfg.key: cfg for cfg in _ALL_SYSTEM_CONFIGS
}

REGISTRY_BY_ID: Final[Mapping[str, SystemConfig]] = {
    cfg.id: cfg for cfg in _ALL_SYSTEM_CONFIGS
}

__all__ = [
    "MoietyType",
    "SubstrateType",
    "MoleculeConfig",
    "SubstrateConfig",
    "SystemConfig",
    "FePcConfig",
    "CoPcConfig",
    "CircularConfig",
    "Au111",
    "Au101",
    "REGISTRY_BY_ID",
    "REGISTRY_BY_TUPLE",
]
