import numpy as np
import math
from math import cos, sin, radians
from copy import deepcopy
from collections import deque
from typing import Optional

from shapely import Point, Polygon, LineString, box
import shapely

from molecule_movement.AngleSymmetry import AngleSymmetry
from molecule_movement.parsing import MoleculeTransitionData, MoleculeActionSpace
from molecule_movement.Objects import VerticalAction, LateralAction, Movement
from molecule_movement.constants import MoietyType

from loguru import logger
from .logging import log_and_raise, deprecated

SENSOR_LENGTH = 5

def find_nearest(array,value):
    idx = np.searchsorted(array, value, side="left")
    if idx > 0 and (idx == len(array) or math.fabs(value - array[idx-1]) < math.fabs(value - array[idx])):
        return array[idx-1]
    else:
        return array[idx]

class Molecule():
    def __init__(self,
                 center: Point,
                 shape: Polygon,
                 stochastic_updates: MoleculeTransitionData,
                 num_sensors: int,
                 name: str,
                 type: MoietyType,
                 orientation: int | str = "random",
                 action_space: Optional[dict[str, MoleculeActionSpace]] = None,
                 molecule_point_symmetry: int = 4,
                 substrate_point_symmetry: int = 6,
                 omit_type_in_name: bool = False
                 ):
        self._starting_position = center
        self.center = center
        self.positions = deque([self._starting_position], 2)
        self.__set_action_spaces(stochastic_updates, action_space)
        self.angle_symmetry = AngleSymmetry(molecule_point_symmetry=molecule_point_symmetry, substrate_point_symmetry=substrate_point_symmetry)

        if orientation == "random":
            self.orientation = self.angle_symmetry.random_angle()
        else:
            self.orientation = orientation

        self.shape = shape
        self.type = type
        self._rotation_sizes_map = dict()

        x, y = self.shape.envelope.exterior.coords.xy
        self.shape_size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
        self.shape_size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
        self._rotation_sizes_map[0] = (self.shape_size_x, self.shape_size_y)

        self.polygon = shapely.affinity.translate(shape, self.center.x - self.shape_size_x / 2, self.center.y - self.shape_size_y / 2)
        self.orientations = deque([self.orientation], 2)

        self.polygon = shapely.affinity.rotate(self.polygon, angle=self.orientation, origin=self.center)
        x, y = self.polygon.envelope.exterior.coords.xy
        size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
        size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
        self._rotation_sizes_map[self.orientation] = (size_x, size_y)
        self._polygon_at_start = deepcopy(self.polygon)

        if omit_type_in_name:
            self.name = name
        else:
            self.name = f"{name}({self.type.value})"
        self.num_sensors = num_sensors
        self._set_sensors()

        self.types = deque([self.type], 2)

        self._reached = False
        self._reached_orientation = False
        self._crashed = False

    def __set_action_spaces(self, stochastic_updates: MoleculeTransitionData, action_space: Optional[MoleculeActionSpace] = None):
        self.stochastic_updates = stochastic_updates
        if action_space is None:
            self.action_space = self.stochastic_updates.action_space
        if isinstance(action_space, dict):
            self.action_space_translation = action_space['translation']
            self.action_space_orientation = action_space['orientation']
        else:
            self.action_space_translation = action_space
            self.action_space_orientation = action_space

        self.action_space_translation_x = self.action_space_translation.x_space
        self.action_space_translation_y = self.action_space_translation.y_space

        try:
            self.action_space_translation_dest_x = self.action_space_translation.dest_x_space
            self.action_space_translation_dest_y = self.action_space_translation.dest_y_space
            self.action_space_orientation_x = self.action_space_orientation.x_space
            self.action_space_orientation_y = self.action_space_orientation.y_space
        except AttributeError:
            self.action_space_translation_dest_x = None
            self.action_space_translation_dest_y = None
            self.action_space_orientation_x = None
            self.action_space_orientation_y = None

        try:
            self.action_space_translation_z = self.action_space_translation.z_space
            self.action_space_translation_V = self.action_space_translation.V_space
            self.action_space_orientation_z = self.action_space_orientation.z_space
            self.action_space_orientation_V = self.action_space_orientation.V_space
        except AttributeError:
            self.action_space_translation_z = None
            self.action_space_translation_V = None
            self.action_space_orientation_z = None
            self.action_space_orientation_V = None

    def move(self, action: VerticalAction | LateralAction, lateral_dest_center: Optional[Point] = None) -> Movement:
        if not self.within_action_space(action):
            self.positions.append(self.center)
            return Movement(0.0, 0.0, 0)
        action_xy = Point(round(find_nearest(self.action_space_translation_x,action.xy.x),1), round(find_nearest(self.action_space_translation_y, action.xy.y),1))

        if isinstance(action, VerticalAction):
            action = VerticalAction(action_xy, action.z, action.V)
            sampled_translation = self.__sample_vertical_translation(action)
            rotation = self.__sample_vertical_rotation(action)
            translation = shapely.affinity.rotate(Point(sampled_translation[0], sampled_translation[1]), self.orientation, origin=(0,0))
        elif isinstance(action, LateralAction):
            molecule_rotation_vector = shapely.affinity.rotate(Point(1,0), self.orientation, origin=(0,0))
            molecule_rotation_vector = np.array([molecule_rotation_vector.x, molecule_rotation_vector.y])

            z = find_nearest(self.action_space_translation_z, action.z)
            V = find_nearest(self.action_space_translation_V, action.V)
            action_xy_dest = Point(round(find_nearest(self.action_space_translation_dest_x,action.xy_dest.x),1), round(find_nearest(self.action_space_translation_dest_y, action.xy_dest.y),1))
            if lateral_dest_center is None:
                lateral_dest_center = self.center
                angle_to_dest = 0
            absolute_end_tip_position = shapely.affinity.translate(action.xy_dest, lateral_dest_center.x, lateral_dest_center.y)
            action = LateralAction(action_xy, action_xy_dest, z, V)
            if True: # self.is_lateral_end_within_start(absolute_end_tip_position):
                translation = Point(*self.__sample_lateral_translation(action))
                rotation = self.__sample_lateral_rotation(action)
            else:
                molecule_lateral_dest_vector = np.array([lateral_dest_center.x - self.center.x, lateral_dest_center.y - self.center.y])
                length = np.linalg.norm(molecule_lateral_dest_vector)
                if length <= 0.0:
                    # warn here? this should, never happen
                    angle_to_dest = 0
                else:
                    molecule_lateral_dest_vector = molecule_lateral_dest_vector / length
                    angle_to_dest = int(np.degrees(np.arccos(np.clip(np.dot(molecule_rotation_vector, molecule_lateral_dest_vector), -1.0, 1.0))))
                translation, rotation = self.__sample_distant_lateral_translation(action, angle_to_dest)
                if np.any(translation == (np.nan, np.nan)):
                    translation = Point(0,0)
                    rotation = 0
                else:
                    translation = shapely.affinity.translate(translation, lateral_dest_center.x, lateral_dest_center.y)
                    translation = Point(translation.x - self.center.x, translation.y - self.center.y)


        movement = Movement(translation.x, translation.y, rotation)
        logger.bind(task="stats", movement=movement).trace("")

        self.polygon = shapely.affinity.translate(self.polygon, movement.x, movement.y)
        self.center = Point(self.center.x + movement.x, self.center.y + movement.y)
        self.polygon = shapely.affinity.rotate(self.polygon, angle=movement.rotation, origin=self.center)
        self.positions.append(self.center)

        self.orientation = self.angle_symmetry.rotate(self.orientation, movement.rotation)
        self.orientations.append(self.orientation)
        if not self.orientation in self._rotation_sizes_map:
            self._rotation_sizes_map[self.orientation] = (self.size_x, self.size_y)
        self._set_sensors()

        return movement


    def move_to(self, new_position: Point, new_orientation: int) -> Movement:
        translation = Point(new_position.x - self.center.x, new_position.y - self.center.y)

        # Check if self.orientation is NaN (not a number)
        if math.isnan(self.orientation):
            rotation_change = new_orientation
        else:
            rotation_change = new_orientation - self.orientation
        movement = Movement(translation.x, translation.y, rotation_change)

        self.polygon = shapely.affinity.translate(self.polygon, movement.x, movement.y)
        self.center = Point(self.center.x + movement.x, self.center.y + movement.y)
        self.types.append(self.type)
        self.positions.append(self.center)
        self.polygon = shapely.affinity.rotate(self.polygon, angle=movement.rotation, origin=self.center)
        self.orientation = int(new_orientation)
        self.orientations.append(self.orientation)
        if not self.orientation in self._rotation_sizes_map:
            self._rotation_sizes_map[self.orientation] = (self.size_x, self.size_y)
        self._set_sensors()
        return movement

    def set_position(self, new_position: Point) -> None:
        """
        Set the position of the molecule.

        Parameters:
            new_position (Point): New position of the molecule experiment
        """
        deprecated("Molecule::set_orientation() and Molecule::set_position() are deprecated. Use Molecule::move_to()")
        raise ValueError("")
        self.center = new_position
        self.polygon = shapely.affinity.translate(self.polygon, new_position.x - self.center.x, new_position.y - self.center.y)
        self.positions.append(new_position)
        self._set_sensors()

    def set_orientation(self, measured_angle_deg: int) -> None:
        """
        Map the measured orientation (in degrees) to all symmetry-equivalent angles
        based on combined molecule-substrate symmetry.

        Parameters:
            measured_angle_deg (float): Measured orientation in degrees (0–359°)

        Returns:
            np.ndarray: Array of symmetry-equivalent angles in degrees
        """
        deprecated("Molecule::set_orientation() and Molecule::set_position() are deprecated. Use Molecule::move_to()")
        raise ValueError("")

        # Normalize measured angle to [0, molecule_angle)
        discrete_symmetry_angle = measured_angle_deg % 360 #self._symmetry_angle_moiety

        # Discretize the measured angle bansed on angle_bins
        orientation_bins = np.arange(0, 360, self._discretization_adsorption_angles)
        binned_angle = find_nearest(orientation_bins, discrete_symmetry_angle)
        # Re-normalize measured angle to [0, molecule_angle)
        self.rotation = int(binned_angle % self._symmetry_angle_moiety)

    def get_size(self) -> tuple[float, float]:
        return (self.shape_size_x, self.shape_size_y)

    def get_rotation(self) -> int:
        deprecated("Molecule.get_rotation() is deprected. Use Molecule.orientation")
        return self.orientation

    def __sample_vertical_translation(self, action: VerticalAction) -> tuple[float, float]:
        try:
            return self.stochastic_updates.translations[action]()
        except KeyError as e:
            log_and_raise(e, f"Tried to fetch (x,y) translation at {action}: {e}.")
        else:
            return (0,0)

    def __sample_lateral_translation(self, action: LateralAction) -> tuple[float, float]:
        #logger.info(f"\t\tsampling NEAR     translation")
        try:
            return self.stochastic_updates.translations[action]()
        except KeyError as e:
            log_and_raise(e, f"Tried to fetch (x,y) translation at {action}: {e}.")
        else:
            return (0,0)

    def __sample_vertical_rotation(self, action: VerticalAction) -> int:
        try:
            return self.stochastic_updates.rotations[action]()
        except KeyError as e:
            log_and_raise(e, f"Tried to fetch (x,y) rotation at {action}: {e}.")

    def __sample_lateral_rotation(self, action: LateralAction) -> int:
        try:
            return self.stochastic_updates.rotations[action]()
        except KeyError as e:
            log_and_raise(e, f"Tried to fetch (x,y) rotation at {action}: {e}.")

    def __sample_distant_lateral_translation(self, action: LateralAction, angle_to_dest: int) -> tuple[tuple[float, float], int]:
        #logger.info(f"\t\tsampling DISTANCE translation with angle: {angle_to_dest=}")
        symmetry_shift = angle_to_dest // 90 ##FIXXME this needs to take actual symmetries into account
        angle_for_sampling = angle_to_dest % 90
        start_tip = shapely.affinity.rotate(action.xy, -symmetry_shift * 90, origin=(0,0))
        action_for_sampling = LateralAction(start_tip, action.xy_dest, z=action.z, V=action.V)
        try:
            support = self.stochastic_updates.distant_translations[angle_for_sampling][(np.float64(action.xy.x),np.float64(action.xy.y),np.float64(action.xy_dest.x),np.float64(action.xy_dest.y),np.float64(action.z),np.float64(action.V))]
            translation, rotation = support[np.random.choice(len(support))]
            return shapely.affinity.rotate(Point(translation), symmetry_shift * 90, origin=(0,0)), rotation
        except KeyError as e:
            #log_and_raise(e, f"Tried to fetch (x,y) translation at {action}: {e}.")
            return (np.nan, np.nan), np.nan
        else:
            return (0,0)

    @property
    def previous_type(self) -> str:
        return self.types[0]

    @property
    def previous_orientation(self) -> int:
        return self.orientations[0]

    @property
    def previous_position(self) -> Point:
        return self.positions[0]

    @property
    def rotation(self):
        deprecated("Molecule::rotation has been deprecated, use Molecule::orientation instead")
        return self.orientation

    @property
    def size_x(self) -> float:
        if self.orientation not in self._rotation_sizes_map:
            x, y = self.polygon.envelope.exterior.coords.xy
            size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
            size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
            self._rotation_sizes_map[self.orientation] = (size_x, size_y)
        return self._rotation_sizes_map[self.orientation][0]

    @property
    def size_y(self) -> float:
        if self.orientation not in self._rotation_sizes_map:
            x, y = self.polygon.envelope.exterior.coords.xy
            size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
            size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
            self._rotation_sizes_map[self.orientation] = (size_x, size_y)
        return self._rotation_sizes_map[self.orientation][1]

    @property
    def coords(self) -> list[tuple[float, float]]:
        return shapely.get_coordinates(self.polygon).tolist()

    def get_drawable_polygon(self, nm_size: int) -> list[tuple[int, int]]:
        shape = shapely.affinity.translate(self.shape, xoff=-self.center.x, yoff=-self.center.y)
        scaled = shapely.affinity.scale(shape, nm_size, nm_size, origin=(0, 0))
        rotated = shapely.affinity.rotate(scaled, angle=self.orientation, origin=(0, 0))

        minx, miny, _, _ = rotated.bounds
        tight = shapely.affinity.translate(rotated, xoff=-minx, yoff=-miny)
        coords = shapely.get_coordinates(tight).tolist()

        if len(coords) > 1 and coords[0] == coords[-1]:
            coords = coords[:-1]
        return [(int(x), int(y)) for x, y in coords]

    def __str__(self):
        return f"{self.name}: ({self.center.x:0.2f},{self.center.y:0.2f}|{self.orientation})"

    def _set_sensors(self):
        if self.num_sensors < 2:
            self.sensors = list()
            return

        cx, cy = self.center.x, self.center.y
        r = SENSOR_LENGTH
        step_deg = 360.0 / self.num_sensors

        sensors = []
        for i in range(self.num_sensors):
            a1 = radians(self.orientation + step_deg * i)
            a2 = radians(self.orientation + step_deg * (i + 1))
            p1 = (cx + r * cos(a1), cy + r * sin(a1))
            p2 = (cx + r * cos(a2), cy + r * sin(a2))
            sensors.append(Polygon([(cx, cy), p1, p2]))
        self.sensors = sensors

    def set_reached(self) -> None:
        deprecated("Molecule properties 'reached' and 'reached_orientation' are deprecated. Compute these values in the respective environment.")
        self._reached = True

    def set_reached_orientation(self) -> None:
        deprecated("Molecule properties 'reached' and 'reached_orientation' are deprecated. Compute these values in the respective environment.")
        self._reached_orientation = True

    def set_crashed(self) -> None:
        self._crashed = True

    def within_action_space(self, action: VerticalAction | LateralAction) -> bool:
        xy = action.xy
        return ((self.action_space_translation_x[0] <= xy.x and xy.x <= self.action_space_translation_x[-1]) and
                (self.action_space_translation_y[0] <= xy.y and xy.y <= self.action_space_translation_y[-1]))

    def is_lateral_end_within_start(self, absolute_end_tip_position: Point) -> bool:
        action_space_bounding_box = shapely.affinity.translate(box(self.action_space_translation_x[0],
                                                                   self.action_space_translation_y[0],
                                                                   self.action_space_translation_x[-1],
                                                                   self.action_space_translation_y[-1]), self.center.x, self.center.y)
        return action_space_bounding_box.intersects(absolute_end_tip_position)


    @property
    def reached(self) -> bool:
        deprecated("Molecule properties 'reached' and 'reached_orientation' are deprecated. Compute these values in the respective environment.")
        raise ValueError("")
        return self._reached

    @property
    def reached_orientation(self) -> bool:
        deprecated("Molecule properties 'reached' and 'reached_orientation' are deprecated. Compute these values in the respective environment.")
        raise ValueError("")
        return self._reached_orientation

    @property
    def crashed(self) -> bool:
        return self._crashed

    @property
    def starting_position(self) -> Point:
        return self._starting_position

    @property
    def maximum_movement(self) -> float:
        return self.stochastic_updates.maximum_movement

    @property
    def polygon_at_start(self) -> Polygon:
        return self._polygon_at_start

    def set_type(self, type: str = None):
        return

    @property
    def translated(self) -> Point:
        return Point(self.center.x - self.previous_position.x, self.center.y - self.previous_position.y)

    @property
    def rotated(self) -> int:
        return self.orientation - self.previous_orientation

    @property
    def movement(self) -> Movement:
        return Movement(self.translated.x, self.translated.y, self.rotated)

class MoleculeExperiment(Molecule):
    def __init__(self,
                 center: Point,
                 shape: Polygon,
                 orientation: int,
                 action_space: dict,
                 num_sensors: int,
                 name: str,
                 index: int,
                 type: str,
                 molecule_point_symmetry: int,
                 substrate_point_symmetry: int
                 ):
        self.types = deque([type], 2)
        self._starting_position = center
        self.center = center
        self.positions = deque([self._starting_position], 2)
        self.index = index

        self.action_space = action_space
        self.action_space_translation = action_space['translation']
        self.action_space_orientation = action_space['orientation']

        self.action_space_translation_x = self.action_space_translation.x_space
        self.action_space_translation_y = self.action_space_translation.y_space

        self.positions = deque([self._starting_position], 2)

        self._compute_adsorption_angles(molecule_point_symmetry, substrate_point_symmetry)

        try:
            self.action_space_translation_dest_x = self.action_space_translation.dest_x_space
            self.action_space_translation_dest_y = self.action_space_translation.dest_y_space
            self.action_space_orientation_x = self.action_space_orientation.x_space
            self.action_space_orientation_y = self.action_space_orientation.y_space
        except AttributeError:
            self.action_space_translation_dest_x = None
            self.action_space_translation_dest_y = None
            self.action_space_orientation_x = None
            self.action_space_orientation_y = None

        self.action_space_translation_z = self.action_space_translation.z_space
        self.action_space_translation_V = self.action_space_translation.V_space
        self.action_space_orientation_z = self.action_space_orientation.z_space
        self.action_space_orientation_V = self.action_space_orientation.V_space

        self.shape = shape
        self._rotation_sizes_map = dict()

        x, y = self.shape.envelope.exterior.coords.xy
        self.shape_size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
        self.shape_size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
        self._rotation_sizes_map[0] = (self.shape_size_x, self.shape_size_y)

        self.polygon = shapely.affinity.translate(shape, self.center.x - self.shape_size_x / 2, self.center.y - self.shape_size_y / 2)
        self.orientation = orientation
        self.orientations = deque([self.orientation], 2)

        self.polygon = shapely.affinity.rotate(self.polygon, angle=self.rotation, origin=self.center)
        self._polygon_at_start = deepcopy(self.polygon)
        x, y = self.polygon.envelope.exterior.coords.xy
        size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
        size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
        self._rotation_sizes_map[self.orientation] = (size_x, size_y)

        self.name = name

        self.type = type


        self.num_sensors = num_sensors
        self._set_sensors()

        self._reached = False
        self._crashed = False
        self._reached_orientation = False

    def set_type(self, type: str) -> None:
        """
        Set the type of the molecule experiment.

        Parameters:
            type (str): Type of the molecule experiment
        """
        self.type = type
        self.types.append(type)
        if types[0] != types[1]:
            pass
            # update AngleSymmetry

    def move(self, action: VerticalAction | LateralAction, lateral_dest_center: Optional[Point] = None) -> Movement:
        return Movement(0,0,0)

    @property
    def maximum_movement(self) -> float:
        return 1.0
