import numpy as np
from scipy.ndimage import gaussian_filter


def rotate_polygon(polygon, rotation_deg=30):
    """
    Rotate the polygon.
    """
    theta = np.deg2rad(rotation_deg)

    rotation_matrix = np.array([
        [np.cos(theta), -np.sin(theta)],
        [np.sin(theta), np.cos(theta)]
    ])

    # Apply the rotation
    rotated_polygon = np.dot(polygon, rotation_matrix.T)

    return rotated_polygon


def translate_polygon(polygon, translation=(10, 10)):
    """
    Translate the polygon.
    """
    translated_polygon = polygon + translation
    return translated_polygon

def Gaussian(x, z, sigma=0.5):
    return np.exp((-(np.linalg.norm(x - z, axis=1) ** 2)) / (2 * sigma ** 2))

def create_deformation(points, control_point, VECTOR):
    deformed = Gaussian(points, control_point)[:, None] @ VECTOR[None, :]  # + points
    return deformed

def random_deform_polygon(arr_contour_pts, std_deviation=1.0, scale=2.0):
    """
    Add random deformation to the polygon.
    """
    # num_points = polygon.shape[0]
    # random_deformation = (np.random.rand(num_points, 2) * 2 - 1) * scale
    # random_deformation = gaussian_filter(random_deformation, sigma=std_deviation, mode='wrap')
    #
    # deformed_polygon = polygon + random_deformation

    control_points = np.array([[-2, ]])

    nx, ny = (5, 5)
    x = np.linspace(-2, 2, nx)
    y = np.linspace(-2, 2, ny)
    xv, yv = np.meshgrid(x, y)
    arr_control_pts = np.concatenate((xv.ravel()[:, None], yv.ravel()[:, None]), axis=1)

    arr_deformed = arr_contour_pts.copy()
    for ith_con_pt in arr_control_pts:
        #Gaussian(x, ith_con_pt)
        arr_deformed += create_deformation(arr_contour_pts, ith_con_pt, np.random.rand(2) * 0.5)
    return arr_deformed


def deform_polygon(polygon, rotation_deg=130, translation=(10, 10), std_deviation=1.0, deformation_scale=2.0):
    """
    Combine the rotation, translation, and random deformation.
    """
    #rotated = rotate_polygon(polygon, rotation_deg)
    #translated = translate_polygon(rotated, translation)
    deformed = random_deform_polygon(polygon, std_deviation, deformation_scale)

    return deformed

def show_shapely_polygon(polygon):
    fig, ax = plt.subplots()
    ax.plot(*polygon.exterior.xy, 'b-', label='Polygon')
    ax.set_aspect('equal', adjustable='datalim')
    ax.legend()
    plt.show()
    return

# Example usage
polygon = np.array([
    [0, 0],
    [1, 0],
    [1, 1],
    [0, 1]
])

import pyvista as pv
import numpy as np
from shapely.geometry import Polygon, Point
from shapely.geometry import Point, LineString, Polygon
import matplotlib.pyplot as plt
import pyvista as pv

path_starman_vtk = "../examples/starman/data_ground_truth/ForSimulation__Template__GroundTruth.vtk"
path_control_points = "../examples/starman/data_ground_truth/ForSimulation__ControlPoints__GoundTruth.txt"
mesh = pv.read(path_starman_vtk)
arr_contour_pts = mesh.points[:, [0,1]]
deformed_polygon = deform_polygon(arr_contour_pts )
polygon = Polygon(deformed_polygon)
show_shapely_polygon(polygon)

print(deformed_polygon)
