import numpy as np

from dataclasses import dataclass
from typing import Optional

@dataclass(frozen=True)
class AngleSymmetry:
    molecule_point_symmetry : int
    substrate_point_symmetry : int

    def __post_init__(self):
        combined_point_symmetry = np.lcm(self.molecule_point_symmetry, self.substrate_point_symmetry)

        super().__setattr__('_symmetry_angle_moiety', 360 / self.molecule_point_symmetry)
        super().__setattr__('_discretization_adsorption_angles', 360 / combined_point_symmetry)
        super().__setattr__('_possible_adsorption_angles', np.arange(0, self._symmetry_angle_moiety, self._discretization_adsorption_angles))
        super().__setattr__('_num_adsorption_angles', len(self._possible_adsorption_angles))

    def random_angle(self, seed: Optional[int] = None):
        rng = np.random.default_rng(seed=seed)
        return int(self._possible_adsorption_angles[rng.choice(self._num_adsorption_angles)])

    def rotate(self, current_orientation, rotation) -> int:
        return int((current_orientation + rotation) % self._symmetry_angle_moiety)
