{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "6e5c6d2b",
   "metadata": {},
   "source": [
    "VectorAdam for Rotation Equivariant Geometry Optimization\n",
    "======================================"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bfa45efe",
   "metadata": {},
   "source": [
    "This is an example on how to use VectorAdam for Laplacian smoothing in 2D."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "2264271e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import seaborn as sns\n",
    "import math\n",
    "from util import *\n",
    "from vectoradam import * "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1fa94a6a",
   "metadata": {},
   "source": [
    "# Laplacian Smoothing in 2D "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d33aea7f",
   "metadata": {},
   "source": [
    "### 1. Create the mesh."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ec3bd5e8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "n = 12\n",
    "v1,l1 = create_circle(n_points=n, noise_level=0)\n",
    "v2,l2 = create_circle(n_points=n, noise_level=0)\n",
    "eps = 1\n",
    "x_lim = [np.min(v1[:,0]) - eps, np.max(v1[:,0]) + eps]\n",
    "y_lim = [np.min(v1[:,1]) - eps, np.max(v1[:,1]) + eps]\n",
    "\n",
    "# keep track of meshes for later plotting\n",
    "data= {}\n",
    "data['original mesh'] = (v1, l1)\n",
    "\n",
    "# plot the original mesh \n",
    "plot_mesh2d(v1,l1,x_lim=x_lim, y_lim=y_lim, showfig=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "2ce01491",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set optimizer hyperparameters\n",
    "lr = 0.5\n",
    "betas = (0.9, 0.999)\n",
    "eps = 1e-8"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "1172e412",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Torch the arrays\n",
    "v1 = torch.from_numpy(v1).to(torch.float32).cuda()\n",
    "l1 = torch.from_numpy(l1).cuda()\n",
    "v2 = torch.from_numpy(v2).to(torch.float32).cuda()\n",
    "l2 = torch.from_numpy(l2).cuda()\n",
    "v1.requires_grad = True\n",
    "v2.requires_grad = True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "4c8db3c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Initialize VectorAdam optimizers\n",
    "regadam = VectorAdam([{'params': v1, 'axis': None}], lr=lr, betas=betas, eps=eps)\n",
    "vadam = VectorAdam([{'params': v2, 'axis': -1}], lr=lr, betas=betas, eps=eps)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eb0d8d58",
   "metadata": {},
   "source": [
    "### 2. One Optimization step with Adam"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "ed3f2937",
   "metadata": {},
   "outputs": [],
   "source": [
    "# One optimization step with Adam\n",
    "# Laplacian smoothing loss\n",
    "regadam.zero_grad()\n",
    "vbf = v1.detach().cpu()\n",
    "L1 = laplacian_uniform_2d(v1,l1)\n",
    "loss1 = (v1 * (L1 @ v1)).mean()\n",
    "loss1.backward()\n",
    "regadam.step()\n",
    "vaf = v1\n",
    "adam_step = vaf-vbf.cuda()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "9a622589",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Save the results from Adam\n",
    "data[\"radam grad\"] = -adam_step.detach().cpu().numpy()\n",
    "data[\"original grad\"] = v1.grad.detach().cpu().numpy()\n",
    "data[\"radam mesh step1\"] = (v1.detach().cpu().numpy(), l1.detach().cpu().numpy())\n",
    "\n",
    "fig, ax = plot_mesh2d(data['original mesh'][0], data['original mesh'][1], \n",
    "                      x_lim = x_lim,\n",
    "                      y_lim = y_lim,\n",
    "                      return_ax = True)\n",
    "ax.quiver(data['original mesh'][0][:,0], data['original mesh'][0][:,1], \n",
    "          -data[\"original grad\"][:,0], -data[\"original grad\"][:,1], scale=1.0, color=\"#EC8A19\", width=0.015)\n",
    "ax.quiver(data['original mesh'][0][:,0], data['original mesh'][0][:,1], \n",
    "          -data[\"radam grad\"][:,0], -data[\"radam grad\"][:,1], scale=2.5, color=\"#226843\", width=0.015)\n",
    "plt.show()\n",
    "\n",
    "# step 1 image \n",
    "data[f'radam mesh step1'] = (v1.detach().cpu().numpy(), l1.detach().cpu().numpy())\n",
    "plot_mesh2d(data[f'radam mesh step1'][0], data[f'radam mesh step1'][1], \n",
    "            x_lim = x_lim,\n",
    "            y_lim = y_lim,\n",
    "            showfig=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4d4c2538",
   "metadata": {},
   "source": [
    "### 3. One optimization step with VectorAdam"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "fe3c830b",
   "metadata": {},
   "outputs": [],
   "source": [
    "vadam.zero_grad()\n",
    "vbf = v2.detach().cpu()\n",
    "L2 = laplacian_uniform_2d(v2,l2)\n",
    "loss2 = (v2 * (L2 @ v2)).mean()\n",
    "loss2.backward()\n",
    "vadam.step()\n",
    "vaf = v2\n",
    "vadam_step = vaf-vbf.cuda()\n",
    "data[\"vadam grad\"] = -vadam_step.detach().cpu().numpy()\n",
    "data[\"vadam mesh step1\"] = (v2.detach().cpu().numpy(), l2.detach().cpu().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "ca224d4c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot \n",
    "# gradient image\n",
    "fig, ax = plot_mesh2d(data['original mesh'][0], data['original mesh'][1], \n",
    "                      x_lim = x_lim,\n",
    "                      y_lim = y_lim,\n",
    "                      return_ax = True)\n",
    "ax.quiver(data['original mesh'][0][:,0], data['original mesh'][0][:,1], \n",
    "          -data[\"original grad\"][:,0], -data[\"original grad\"][:,1], scale=1.0, color=\"#EC8A19\", width=0.015)\n",
    "ax.quiver(data['original mesh'][0][:,0], data['original mesh'][0][:,1], \n",
    "          -data[\"vadam grad\"][:,0], -data[\"vadam grad\"][:,1], scale=2.5, color=\"#2A63AD\", width=0.015)\n",
    "plt.show()\n",
    "\n",
    "# step 1 image \n",
    "data[f'vadam mesh step1'] = (v2.detach().cpu().numpy(), l2.detach().cpu().numpy())\n",
    "plot_mesh2d(data[f'vadam mesh step1'][0], data[f'vadam mesh step1'][1], \n",
    "            x_lim = x_lim,\n",
    "            y_lim = y_lim,\n",
    "            showfig=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5d1a940f",
   "metadata": {},
   "source": [
    "### 4. Five optimization steps with Adam"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "b2a5cab0",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 5/5 [00:00<00:00, 393.43it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# optimization loop\n",
    "# regular adam\n",
    "steps = 5\n",
    "from tqdm import tqdm\n",
    "for i in tqdm(range(steps)):\n",
    "    regadam.zero_grad()\n",
    "    #laplacian loss\n",
    "    L1 = laplacian_uniform_2d(v1,l1)\n",
    "    loss1 = (v1 * (L1 @ v1)).mean()\n",
    "    loss1.backward()\n",
    "    regadam.step()\n",
    "\n",
    "data[f'radam mesh step{steps}'] = (v1.detach().cpu().numpy(), l1.detach().cpu().numpy())\n",
    "plot_mesh2d(data[f'radam mesh step{steps}'][0], data[f'radam mesh step{steps}'][1], \n",
    "            x_lim = x_lim,\n",
    "            y_lim = y_lim,\n",
    "            showfig=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "adf86437",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 5/5 [00:00<00:00, 442.19it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x360 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# vector adam\n",
    "steps = 5\n",
    "for i in tqdm(range(steps)):\n",
    "    vadam.zero_grad()\n",
    "    #laplacian loss\n",
    "    L2 = laplacian_uniform_2d(v2,l2)\n",
    "    loss2 = (v2 * (L2 @ v2)).mean()\n",
    "    loss2.backward()\n",
    "    vadam.step()\n",
    "\n",
    "data[f'vadam mesh step{steps}'] = (v2.detach().cpu().numpy(), l2.detach().cpu().numpy())\n",
    "plot_mesh2d(data[f'vadam mesh step{steps}'][0], data[f'vadam mesh step{steps}'][1], \n",
    "            x_lim = x_lim,\n",
    "            y_lim = y_lim,\n",
    "            showfig=True)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
