import os
import math
import numpy as np
import matplotlib.pyplot as plt
import imageio

os.environ["KMP_DUPLICATE_LIB_OK"] = "TRUE"


class ParabolaSimulator:
    def __init__(self,
                 newton_pos=(0, 0),
                 fruit_pos=(0, 10),
                 vx=1.0,
                 vy=5.0,
                 newton_img_path='newton.png',
                 fruit_img_path='apple.png',
                 newton_zoom=0.25,
                 fruit_zoom=0.25,
                 fixed_max_x=18,
                 fixed_max_y=12,
                 min_y=0.2):
        self.newton_pos = newton_pos
        self.x0, self.y0 = fruit_pos
        self.vx = vx
        self.vy = vy
        self.g = 9.81
        self.min_y = min_y

        self.newton_img = plt.imread(newton_img_path)
        self.fruit_img = plt.imread(fruit_img_path)
        self.newton_zoom = newton_zoom
        self.fruit_zoom = fruit_zoom

        self._set_fixed_axis(fixed_max_x, fixed_max_y)

        self.fruit_half_height = (self.fruit_img.shape[0] * self.fruit_zoom) / 80.0

        if self.y0 > (self.min_y + self.fruit_half_height):
            self.t_stop = (self.vy + math.sqrt(
                self.vy ** 2 + 2 * self.g * (self.y0 - (self.min_y + self.fruit_half_height)))) / self.g
        else:
            self.t_stop = 0

    def _set_fixed_axis(self, max_x, max_y):
        self.x_lim = (0, max_x)
        self.y_lim = (0, max_y)

    def _compute_position(self, t):
        if t <= 0:
            return (self.x0, max(self.y0, self.min_y + self.fruit_half_height))
        if t >= self.t_stop:
            x = self.x0 + self.vx * self.t_stop
            y = self.min_y + self.fruit_half_height
            return (x, y)
        x = self.x0 + self.vx * t
        y = self.y0 + self.vy * t - 0.5 * self.g * t ** 2
        return (x, max(y, self.min_y + self.fruit_half_height))

    def _add_image(self, ax, img, pos, zoom):
        H, W = img.shape[0], img.shape[1]
        phys_w = (W / 100.0) * zoom
        phys_h = (H / 100.0) * zoom
        x0 = pos[0] - phys_w / 2
        x1 = pos[0] + phys_w / 2
        y0 = pos[1] - phys_h / 2
        y1 = pos[1] + phys_h / 2
        ax.imshow(img, extent=[x0, x1, y0, y1], aspect='auto', zorder=20, interpolation='bilinear')

    def _compute_shape_params(self):
        H, W = self.fruit_img.shape[0], self.fruit_img.shape[1]
        phys_w = (W / 100.0) * self.fruit_zoom
        phys_h = (H / 100.0) * self.fruit_zoom
        s = min(phys_w, phys_h)
        l = max(phys_w, phys_h)
        a = math.pi * (s / 2.0) * (l / 2.0)
        theta = 0.0
        omega = 0.0
        return theta, omega, s, l, a


if __name__ == "__main__":
    test_times = [i * 0.01 for i in range(101)]
    base_output_dir = ""
    os.makedirs(base_output_dir, exist_ok=True)

    n_samples = 10

    fig_w, fig_h = 2880, 1920

    for i in range(n_samples):
        x0 = float(np.random.uniform(1.0, 2.0))
        y0 = float(np.random.uniform(8.0, 9.0))
        vx = float(np.random.uniform(0.0, 15.0))
        vy = float(np.random.uniform(-2.0, 2.0))

        print(f"Simulating {i+1}/{n_samples}: vx={vx:.3f}, vy={vy:.3f}, x0={x0:.3f}, y0={y0:.3f}")

        sim = ParabolaSimulator(
            newton_pos=(10, 1),
            fruit_pos=(x0, y0),
            vx=vx,
            vy=vy,
            newton_img_path='',
            fruit_img_path='',
            newton_zoom=0.6,
            fruit_zoom=0.35,
            fixed_max_x=18,
            fixed_max_y=12,
            min_y=0.2
        )

        first_pos = sim._compute_position(0)
        px = int(first_pos[0] / sim.x_lim[1] * fig_w)
        py = int((sim.y_lim[1] - first_pos[1]) / sim.y_lim[1] * fig_h)

        theta0, omega0, s0, l0, a0 = sim._compute_shape_params()

        base_name = (
            f"px_{px}_py_{py}_x_{x0:.2f}_y_{y0:.2f}_vx_{vx:.2f}_vy_{vy:.2f}_"
            f"theta_{theta0:.2f}_omega_{omega0:.2f}_s_{s0:.4f}_l_{l0:.4f}_a_{a0:.4f}"
        ).replace('.', '_')

        video_path = os.path.join(base_output_dir, f"{base_name}.mp4")
        txt_path = os.path.join(base_output_dir, f"{base_name}.txt")

        writer = imageio.get_writer(video_path, fps=25)
        with open(txt_path, 'w') as ftxt:
            ftxt.write("t\tx\ty\tvx\tvy\ttheta\tomega\ts\tl\ta\n")

            for t in test_times:
                fig, ax = plt.subplots(figsize=(28.8, 19.2), dpi=100)
                ax.set_position([0, 0, 1, 1])
                ax.axis('off')
                ax.set_xlim(*sim.x_lim)
                ax.set_ylim(*sim.y_lim)
                ax.set_aspect('equal')


                ground_y = 0.18
                ax.axhline(ground_y, color='#689919', linewidth=10, zorder=10)


                sim._add_image(ax, sim.newton_img, sim.newton_pos, sim.newton_zoom)
                fruit_pos = sim._compute_position(t)
                sim._add_image(ax, sim.fruit_img, fruit_pos, sim.fruit_zoom)


                fig.canvas.draw()
                w, h = fig.canvas.get_width_height()
                buf = np.frombuffer(fig.canvas.tostring_argb(), dtype=np.uint8).reshape(h, w, 4)
                frame = buf[:, :, [1, 2, 3]]
                writer.append_data(frame)
                plt.close(fig)


                vx_t = sim.vx if t < sim.t_stop else 0.0
                vy_t = sim.vy - sim.g * t if t < sim.t_stop else 0.0
                theta, omega, s, l, a = sim._compute_shape_params()
                ftxt.write(
                    f"{t:.4f}\t{fruit_pos[0]:.4f}\t{fruit_pos[1]:.4f}\t"
                    f"{vx_t:.4f}\t{vy_t:.4f}\t{theta:.4f}\t{omega:.4f}\t{s:.4f}\t{l:.4f}\t{a:.4f}\n"
                )

        writer.close()
        print(f"Saved: {video_path}, {txt_path}")
