import numpy as np

from abc import ABC, abstractmethod

import shapely
from shapely import Point, Polygon

from dataclasses import dataclass
from typing import Optional, Type

from loguru import logger

Z_CONSTANT = 0.0
V_CONSTANT = 0

@dataclass(frozen=True)
class VerticalAction:
    xy: Point
    z: float = Z_CONSTANT
    V: int = V_CONSTANT

    def __repr__(self) -> str:
        return f"{{'x':{round(self.xy.x,3)},'y':{round(self.xy.y,3)},'z':{int(self.z)},'V':{self.V}}}"

@dataclass(frozen=True)
class LateralAction:
    xy: Point
    xy_dest: Point
    z: float = Z_CONSTANT
    V: int = V_CONSTANT

    def __repr__(self) -> str:
        return f"{{'x':{self.xy.x},'y':{self.xy.y},'x_dest':{self.xy_dest.x},'y_dest':{self.xy_dest.y},'z':{self.z},'V':{self.V}}}"


@dataclass
class Movement:
    x: float
    y: float
    rotation: int

    @property
    def translation(self) -> Point:
        return Point(self.x,self.y)

    @property
    def distance_moved(self) -> float:
        return np.linalg.norm((self.x, self.y))

    def moved(self, threshold: Optional[float] = 0.001) -> bool:
        return np.linalg.norm((self.x, self.y)) >= threshold

    def __str__(self) -> str:
        return f"Movement(x_delta={round(self.x,3)}, y_delta={round(self.y,3)}, rotation={int(self.rotation)}, travelled={self.distance_moved})"

    def __repr__(self) -> str:
        return f"{{'x':{round(self.x,3)},'y':{round(self.y,3)},'rot':{int(self.rotation)},'travelled':{self.distance_moved}}}"

@dataclass
class Object(ABC):
    position: Point
    shape: Polygon
    rotation: int

    @abstractmethod
    def __init__(self):
        pass

    @property
    def orientation(self):
        return self.rotation

    @property
    def rotated_shape(self):
        self._rotated_shape = shapely.affinity.rotate(self.shape, self.rotation)
        minx, miny, _, _ = self._rotated_shape.bounds
        self._rotated_shape = shapely.affinity.translate(self._rotated_shape, -minx, -miny)
        return self._rotated_shape


    @property
    def shape_coords(self):
        return shapely.get_coordinates(self.rotated_shape).tolist()

    @property
    def coords(self):
        return shapely.get_coordinates(self.polygon).tolist()

    @property
    def polygon(self) -> Polygon:
        if not hasattr(self,"_polygon") or not self._polygon:
            self._polygon = shapely.affinity.translate(self.rotated_shape, self.position.x - self.size_x/2, self.position.y - self.size_y/2)
        return self._polygon

    @property
    def size_x(self) -> float:
        if not hasattr(self, "_size_x") or not self._size_x:
            x, y = self.rotated_shape.envelope.exterior.coords.xy
            self._size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
            self._size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
        return self._size_x

    @property
    def size_y(self) -> float:
        if not hasattr(self, "_size_y") or not self._size_y:
            x, y = self.rotated_shape.envelope.exterior.coords.xy
            self._size_x = Point(x[0], y[0]).distance(Point(x[1], y[1]))
            self._size_y = Point(x[1], y[1]).distance(Point(x[2], y[2]))
        return self._size_y

    def move_by(self, x, y):
        return (self.position.x * x, self.position.y * y)

    def scale(self, x, y):
        scaled = shapely.affinity.scale(self.rotated_shape, x, y, origin=(0,0))
        return shapely.get_coordinates(scaled).tolist()

    def rotate(self):
        return

    def is_goal(self) -> bool:
        return False

    def is_obstacle(self) -> bool:
        return False

@dataclass
class Goal(Object):
    type: Type

    def is_goal(self) -> bool:
        return True

    def __hash__(self):
        return int(self.position.x * 2 + self.position.y * 3)

    def __repr__(self) -> str:
        return f"{{'x':{round(self.position.x,3)},'y':{round(self.position.y,3)},'rotation':{int(self.rotation)}}}"


@dataclass
class Obstacle(Object):
    def is_obstacle(self) -> bool:
        return True

    def set_buffer(self, buffer: Polygon):
        self.buffer = buffer
        self.__set_circumventing_anchors()

    def __set_circumventing_anchors(self):
        self.circumventing_anchors = [Point(p) for p in list(zip(*self.buffer.exterior.coords.xy))]

    def scaled_buffer(self, x, y):
        scaled_buffer = shapely.affinity.scale(self.buffer, x, y, origin=(0,0))
        return shapely.get_coordinates(scaled_buffer).tolist()
