{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "91dcc4b8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib.animation import FuncAnimation\n",
    "from scipy.integrate import solve_ivp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "12877d06",
   "metadata": {},
   "outputs": [],
   "source": [
    "def animate_phase_map(\n",
    "        qs=np.linspace(-1.0, 1.0, 300),\n",
    "        ps=np.linspace(-1.0, 1.0, 300),\n",
    "        energy_func=None,\n",
    "        hamiltonian_rhs=None,  # function (t, y) -> [dq/dt, dp/dt]\n",
    "        levels=None,\n",
    "        contour_line_numbers=60,\n",
    "        title=\"Phase Space Energy\",\n",
    "        q_spec_label=\"(position)\",\n",
    "        p_spec_label=\"(momentum)\",\n",
    "        t_vals=np.linspace(0, 10, 100),   # times for animation\n",
    "        init_points=[(0.5, 0.0), (-0.5, 0.5)],  # list of (q0, p0)\n",
    "        save_anim=False,\n",
    "        anim_path='visualizations/phase_map_animation.mp4',\n",
    "        cmap=\"viridis\",\n",
    "        linecolor=\"black\",\n",
    "        fps=15\n",
    "    ):\n",
    "    \"\"\"\n",
    "    Animate phase space energy contours over time with fixed color scale.\n",
    "    energy_func must be callable: energy_func(Q, P, t).\n",
    "    \"\"\"\n",
    "    Q, P = np.meshgrid(qs, ps)\n",
    "\n",
    "    # Precompute global min/max across all times for fixed color scale\n",
    "    H_min, H_max = np.inf, -np.inf\n",
    "    for t in t_vals:\n",
    "        H = energy_func(Q, P, t)\n",
    "        H_min = min(H_min, np.min(H))\n",
    "        H_max = max(H_max, np.max(H))\n",
    "\n",
    "    if levels is None:\n",
    "        levels = np.linspace(H_min, H_max, contour_line_numbers)\n",
    "\n",
    "     # Integrate trajectories\n",
    "    trajectories = []\n",
    "    for q0, p0 in init_points:\n",
    "        sol = solve_ivp(\n",
    "            hamiltonian_rhs, \n",
    "            t_span=(t_vals[0], t_vals[-1]),\n",
    "            y0=[q0, p0],\n",
    "            t_eval=t_vals,\n",
    "            rtol=1e-9, atol=1e-9\n",
    "        )\n",
    "        trajectories.append(sol.y)  # shape (2, len(t_vals))\n",
    "\n",
    "    fig, ax = plt.subplots(figsize=(8, 6))\n",
    "    H0 = energy_func(Q, P, t_vals[0])\n",
    "    contourf = ax.contourf(Q, P, H0, levels=levels, cmap=cmap)\n",
    "    cbar = fig.colorbar(contourf, ax=ax, label='Hamiltonian Energy $\\\\mathcal{H}$')\n",
    "    ax.set_title(title + f\"\\nTime t = {t_vals[0]:.2f}\")\n",
    "    ax.set_xlabel('$q$ ' + q_spec_label)\n",
    "    ax.set_ylabel('$p$ ' + p_spec_label)\n",
    "    ax.grid(True)\n",
    "\n",
    "\n",
    "    # Line + marker for each trajectory\n",
    "    lines = []\n",
    "    points = []\n",
    "    for _ in init_points:\n",
    "        (line,) = ax.plot([], [], lw=1.5)\n",
    "        (point,) = ax.plot([], [], 'o', markersize=6)\n",
    "        lines.append(line)\n",
    "        points.append(point)\n",
    "\n",
    "    def update(frame_idx):\n",
    "        t = t_vals[frame_idx]\n",
    "        H = energy_func(Q, P, t)\n",
    "\n",
    "        # Remove previous contours\n",
    "        for coll in ax.collections:\n",
    "            coll.remove()\n",
    "\n",
    "        # # Update trajectories\n",
    "        # for traj, line, point in zip(trajectories, lines, points):\n",
    "        #     q_traj, p_traj = traj\n",
    "        #     line.set_data(q_traj[:frame_idx+1], p_traj[:frame_idx+1])\n",
    "        #     point.set_data(q_traj[frame_idx], p_traj[frame_idx])\n",
    "\n",
    "        cf = ax.contourf(Q, P, H, levels=levels, cmap=cmap)\n",
    "        ax.set_title(title + f\"\\nTime t = {t:.2f}\")\n",
    "        return ax.collections #+ lines + points\n",
    "\n",
    "    anim = FuncAnimation(fig, update, frames=len(t_vals), blit=False, interval=1000/fps)\n",
    "\n",
    "    if save_anim:\n",
    "        anim.save(anim_path, fps=fps, dpi=200)\n",
    "        plt.close(fig)\n",
    "    else:\n",
    "        plt.show()\n",
    "\n",
    "    return anim\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "96620f84",
   "metadata": {},
   "outputs": [],
   "source": [
    "def H_pendulum(q, p, t):\n",
    "    m, l, g = 1.0, 1.0, 9.81\n",
    "    return p**2 / (2 * m * l**2) - m * g * l * np.cos(q)\n",
    "\n",
    "def dynamics_pendulum(t, x):\n",
    "    m, l, g = 1.0, 1.0, 9.81\n",
    "    q, p = x\n",
    "    dqdt = p / (m * l**2)\n",
    "    dpdt = -m * g * l * np.sin(q)\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_damped_pendulum(q, p, t):\n",
    "    m, l, g = 1.0, 1.0, 9.81   \n",
    "    F0 = 0.1\n",
    "    gamma = 0.1\n",
    "    return np.exp(-gamma*t) * (p**2 / (2 * m * l**2) - m * g * l * np.cos(q))\n",
    "\n",
    "def dynamics_damped_pendulum(t, x):\n",
    "    m, l, g = 1.0, 1.0, 9.81   \n",
    "    gamma = 0.1\n",
    "    q, p = x\n",
    "    dqdt = p / (m * l**2)\n",
    "    dpdt = -m * g * l * np.sin(q) - gamma * p / (m * l**2)\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_windy_pendulum(q, p, t):\n",
    "    m, l, g = 1.0, 1.0, 9.81   \n",
    "    F0 = 0.1\n",
    "    gamma = 0.1\n",
    "    F_ext = F0 * t\n",
    "    return np.exp(-gamma*t) * (p**2 / (2 * m * l**2) - m * g * l * np.cos(q) - q * F_ext)\n",
    "\n",
    "def dynamics_windy_pendulum(t, x):\n",
    "    m, l, g = 1.0, 1.0, 9.81   \n",
    "    F0 = 0.1\n",
    "    gamma = 0.1\n",
    "    F_ext = F0 * t\n",
    "    q, p = x\n",
    "    dqdt = p / (m * l**2)\n",
    "    dpdt = -m * g * l * np.sin(q) - gamma * p / (m * l**2) + F_ext\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_conservative_spring(q, p, t):\n",
    "    k, m = 1.0, 1.0\n",
    "    return 0.5 * k * q**2 + 0.5 * p**2 / m\n",
    "\n",
    "def dynamics_conservative_spring(t, x):\n",
    "    k, m = 1.0, 1.0\n",
    "    q, p = x\n",
    "    dqdt = p / m\n",
    "    dpdt = -k * q\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_damped_spring(q, p, t):\n",
    "    k, m = 1.0, 1.0\n",
    "    gamma = 0.1\n",
    "    return np.exp(-gamma*t) * (0.5 * k * q**2 + 0.5 * p**2 / m)\n",
    "\n",
    "def dynamics_damped_spring(t, x):\n",
    "    k, m = 1.0, 1.0\n",
    "    gamma = 0.1\n",
    "    q, p = x\n",
    "    dqdt = p / m\n",
    "    dpdt = -k * q - gamma * p / m\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_forced_spring(q, p, t):\n",
    "    k, m = 1.0, 1.0\n",
    "    F0, omega = 1.0, 1.0\n",
    "    gamma = 0.1\n",
    "    F_ext = F0 * np.sin(omega * t) * np.sin(2 * omega * t)\n",
    "    return np.exp(-gamma*t) * (0.5 * k * q**2 + 0.5 * p**2 / m - q * F_ext)\n",
    "\n",
    "def dynamics_forced_spring(t, x):\n",
    "    k, m = 1.0, 1.0\n",
    "    F0, omega = 1.0, 1.0\n",
    "    gamma = 0.1\n",
    "    F_ext = F0 * np.sin(omega * t) * np.sin(2 * omega * t)\n",
    "    q, p = x\n",
    "    dqdt = p / m\n",
    "    dpdt = -k * q - gamma * p / m + F_ext\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_henon_heiles(q, p, t):\n",
    "    q1, p1 = 1.0, 1.0\n",
    "    q2, p2 = q, p\n",
    "    a = 1.0\n",
    "    H = 0.5 * (q1 ** 2 + q2 ** 2) +  0.5 * (p1 ** 2 + p2 ** 2) + a * (q1 * q1 * q2 - q2 ** 3 / 3.0)\n",
    "    return H\n",
    "\n",
    "def dynamics_henon_heiles(t, x):\n",
    "    q1, p1 = 1.0, 1.0\n",
    "    q2, p2 = x\n",
    "    a = 1.0\n",
    "    dq1dt = p1\n",
    "    dp1dt = -q1 - 2 * a * q1 * q2\n",
    "    dq2dt = p2\n",
    "    dp2dt = -q2 - a * (q1 ** 2 - q2 ** 2)\n",
    "    return [dq2dt, dp2dt]\n",
    "\n",
    "def H_duffing_unforced(q, p, t):\n",
    "    alpha, beta, m = -1.0, 1.0, 1.0\n",
    "    gamma = 0.3\n",
    "    return np.exp(-gamma*t) * (0.5 * alpha * q**2 + 0.25 * beta * q**4 + 0.5 * p**2 / m)\n",
    "\n",
    "def dynamics_duffing_unforced(t, x):\n",
    "    alpha, beta, m = -1.0, 1.0, 1.0\n",
    "    gamma = 0.3\n",
    "    q, p = x\n",
    "    dqdt = p / m\n",
    "    dpdt = -alpha * q - beta * q**3 - gamma * p / m\n",
    "    return [dqdt, dpdt]\n",
    "\n",
    "def H_duffing_chaotic(q, p, t):\n",
    "    alpha, beta, m = -1.0, 1.0, 1.0\n",
    "    f = 0.39\n",
    "    omega = 1.4\n",
    "    gamma = 0.1\n",
    "    F_ext = f * np.sin(omega * t)\n",
    "    return np.exp(-gamma*t) * (0.5 * alpha * q**2 + 0.25 * beta * q**4 + 0.5 * p**2 / m - q * F_ext)\n",
    "\n",
    "def dynamics_duffing_chaotic(t, x):\n",
    "    alpha, beta, m = -1.0, 1.0, 1.0\n",
    "    f = 0.39\n",
    "    omega = 1.4\n",
    "    F_ext = f * np.sin(omega * t)\n",
    "    gamma = 0.1\n",
    "    q, p = x\n",
    "    dqdt = p / m\n",
    "    dpdt = -alpha * q - beta * q**3 - gamma * p / m + F_ext\n",
    "    return [dqdt, dpdt]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "b1b06459",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_pendulum,\n",
    "    hamiltonian_rhs=dynamics_pendulum,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Conservative Pendulum\",\n",
    "    save_anim=True,\n",
    "    anim_path='single_pendulum.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "343c7c05",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_damped_pendulum,\n",
    "    hamiltonian_rhs=dynamics_damped_pendulum,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Damped Pendulum\",\n",
    "    save_anim=True,\n",
    "    anim_path='damped_pendulum.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "74d46e36",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_windy_pendulum,\n",
    "    hamiltonian_rhs=dynamics_windy_pendulum,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Windy Pendulum\",\n",
    "    save_anim=True,\n",
    "    anim_path='windy_pendulum.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "912698c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_conservative_spring,\n",
    "    hamiltonian_rhs=dynamics_conservative_spring,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Conservative Spring\",\n",
    "    save_anim=True,\n",
    "    anim_path='conservative_spring.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "c748cb2a",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_damped_spring,\n",
    "    hamiltonian_rhs=dynamics_damped_spring,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Damped Spring\",\n",
    "    save_anim=True,\n",
    "    anim_path='damped_spring.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "b8f8638a",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_forced_spring,\n",
    "    hamiltonian_rhs=dynamics_forced_spring,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Forced Spring\",\n",
    "    save_anim=True,\n",
    "    anim_path='forced_spring.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "f21802c2",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_duffing_unforced,\n",
    "    hamiltonian_rhs=dynamics_duffing_unforced,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Unforced Duffing Oscillator\",\n",
    "    save_anim=True,\n",
    "    anim_path='unforced_duffing.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "0e01635e",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_duffing_chaotic,\n",
    "    hamiltonian_rhs=dynamics_duffing_chaotic,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Forced Duffing Oscillator\",\n",
    "    save_anim=True,\n",
    "    anim_path='chaotic_duffing.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "6d6f81ae",
   "metadata": {},
   "outputs": [],
   "source": [
    "anim = animate_phase_map(\n",
    "    energy_func=H_henon_heiles,\n",
    "    hamiltonian_rhs=dynamics_henon_heiles,\n",
    "    t_vals=np.linspace(0, 10, 100),  # animate from t=0 to 20\n",
    "    init_points=[(0.5, 0.0), (-0.5, 0.5)],\n",
    "    title=\"Phase Space Energy for Henon-Heiles System\",\n",
    "    save_anim=True,\n",
    "    anim_path='henon_heiles.mp4',\n",
    "    cmap='coolwarm'\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "hbond",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.13.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
