{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "cxAqUGugzFnH"
   },
   "source": [
    "# RacerCar HJ-Reachability Notebook\n",
    "\n",
    "Notebook dependencies:\n",
    "- System: python3, ffmpeg (for rendering animations)\n",
    "- Python: jupyter, jax, numpy, matplotlib, plotly, tqdm, hj_reachability\n",
    "\n",
    "Example setup for a Ubuntu system (Mac users, maybe `brew` instead of `sudo apt`; Windows users, learn to love [WSL](https://docs.microsoft.com/en-us/windows/wsl/install-win10)):\n",
    "```\n",
    "sudo apt install ffmpeg\n",
    "/usr/bin/python3 -m pip install --upgrade pip\n",
    "pip install --upgrade jupyter jax[cpu] numpy matplotlib plotly tqdm hj-reachability\n",
    "jupyter notebook  # from the directory of this notebook\n",
    "```\n",
    "Alternatively, view this notebook on [Google Colab](https://colab.research.google.com/github/StanfordASL/hj_reachability/blob/main/examples/quickstart.ipynb) and run a cell containing this command:\n",
    "```\n",
    "!pip install --upgrade hj-reachability\n",
    "```"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "id": "mgKB0FrTzFnM"
   },
   "outputs": [],
   "source": [
    "import jax\n",
    "import jax.numpy as jnp\n",
    "import numpy as np\n",
    "import math\n",
    "\n",
    "from IPython.display import HTML\n",
    "import matplotlib.animation as anim\n",
    "import matplotlib.pyplot as plt\n",
    "import plotly.graph_objects as go\n",
    "\n",
    "import hj_reachability as hj"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {
    "id": "3gZHSzLfzFnR"
   },
   "source": [
    "## Defining your own dynamics: `RacerCar`"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "id": "uPckPKqEzFnS"
   },
   "outputs": [],
   "source": [
    "class RacerCar(hj.ControlAndDisturbanceAffineDynamics):\n",
    "\n",
    "    def __init__(self,\n",
    "                 max_acceleration=0.75,\n",
    "                 max_disturbance=0.5,\n",
    "                 control_mode=\"min\",\n",
    "                 disturbance_mode=\"max\",\n",
    "                 control_space=None,\n",
    "                 disturbance_space=None):\n",
    "        if control_space is None:\n",
    "            control_space = hj.sets.Box(jnp.array([-max_acceleration]),\n",
    "                                        jnp.array([max_acceleration]))\n",
    "        if disturbance_space is None:\n",
    "            # disturbance_space = hj.sets.Ball(jnp.zeros(2), max_disturbance)\n",
    "            disturbance_space = hj.sets.Box(jnp.array([-max_disturbance]),\n",
    "                                            jnp.array([max_disturbance]))\n",
    "        super().__init__(control_mode, disturbance_mode, control_space, disturbance_space)\n",
    "\n",
    "    def open_loop_dynamics(self, state, time):\n",
    "        x, _, vx = state\n",
    "        return 0.5*jnp.array([0.85*vx-0.05*x, 1., -0.15*x-0.3*vx])\n",
    "\n",
    "    def control_jacobian(self, state, time):\n",
    "        return 0.5*jnp.array([\n",
    "            [0.5],\n",
    "            [0.],\n",
    "            [1.],\n",
    "        ])\n",
    "\n",
    "    def disturbance_jacobian(self, state, time):\n",
    "        return 0.5*jnp.array([\n",
    "            [1.],\n",
    "            [0.],\n",
    "            [0.],\n",
    "        ])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "id": "3txvuuMu2ZNG"
   },
   "outputs": [],
   "source": [
    "lo_x = -10. \n",
    "lo_y = 0. \n",
    "lo_xdot = -5. \n",
    "lo_xx = np.array([lo_x, lo_y, lo_xdot])\n",
    "\n",
    "r_x = 20. \n",
    "r_y = 200. \n",
    "r_xd = 10. \n",
    "\n",
    "hi_xx = np.array([lo_x+r_x, lo_y+r_y, lo_xdot+r_xd])\n",
    "\n",
    "n_x = 201\n",
    "n_y = 401\n",
    "n_xd = 101\n",
    "\n",
    "k_x = n_x/r_x\n",
    "k_y = n_y/r_y\n",
    "k_xd = n_xd/r_xd\n",
    "\n",
    "spacerD = 50.0\n",
    "firstObs = 12.5\n",
    "rObs = 3.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "TBmRsBm1zFnS",
    "outputId": "fe67d246-df6a-4351-8cae-016d67c51265"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(201, 401, 101, 3)\n",
      "(201, 401, 101)\n"
     ]
    }
   ],
   "source": [
    "dynamics = RacerCar()\n",
    "grid = hj.Grid.from_lattice_parameters_and_boundary_conditions(hj.sets.Box(lo=lo_xx,\n",
    "                                                                           hi=hi_xx),\n",
    "                                                               (n_x, n_y, n_xd)\n",
    "                                                              )\n",
    "\n",
    "print(grid.states.shape)\n",
    "\n",
    "states1 = jnp.stack(((grid.states[..., 1] - firstObs) % spacerD, spacerD - ((grid.states[..., 1] - firstObs) % spacerD)), axis=-1)\n",
    "states2 = jnp.min(states1, axis=-1)\n",
    "\n",
    "#values = jnp.linalg.norm(grid.states[..., :2], axis=-1) - 1\n",
    "#print(values.shape)\n",
    "\n",
    "tmp = jnp.stack((grid.states[..., 0], states2), axis=-1)\n",
    "values = jnp.exp(rObs - jnp.linalg.norm(tmp, axis=-1)) - 1.0 # + jnp.linalg.norm(grid.states[..., :1], axis=-1) + \n",
    "# values = 100.0*(((1.0+rObs)/(1.0+jnp.linalg.norm(tmp, axis=-1))) - 1.0) + 0.05*jnp.linalg.norm(grid.states[..., :1], axis=-1)\n",
    "#valueMin = -3.0*jnp.ones(values.shape)\n",
    "#values = jnp.max(jnp.stack((values, valueMin), axis=-1), axis=-1)\n",
    "print(values.shape)\n",
    "\n",
    "solver_settings = hj.SolverSettings.with_accuracy(\"medium\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "cu1buKTm9EJ1",
    "outputId": "d2b272c3-0814-4467-da85-5e98028b5b9d"
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "  0%|                                                                   |  0.0000/2.5 [00:00<?, ?sim_s/s]\u001b[A\n",
      "  1%|4                                                          |  0.0175/2.5 [00:01<02:31, 61.15s/sim_s]\u001b[A\n",
      "  1%|8                                                          |  0.0350/2.5 [00:01<02:02, 49.79s/sim_s]\u001b[A\n",
      "  2%|#2                                                         |  0.0525/2.5 [00:02<01:55, 46.99s/sim_s]\u001b[A\n",
      "  3%|#6                                                         |  0.0700/2.5 [00:03<01:48, 44.69s/sim_s]\u001b[A\n",
      "  3%|##                                                         |  0.0875/2.5 [00:03<01:44, 43.19s/sim_s]\u001b[A\n",
      "  4%|##4                                                        |  0.1042/2.5 [00:04<01:42, 42.90s/sim_s]\u001b[A\n",
      "  5%|##8                                                        |  0.1217/2.5 [00:05<01:42, 43.29s/sim_s]\u001b[A\n",
      "  6%|###2                                                       |  0.1392/2.5 [00:06<01:39, 42.35s/sim_s]\u001b[A\n",
      "  6%|###6                                                       |  0.1566/2.5 [00:06<01:38, 41.93s/sim_s]\u001b[A\n",
      "  7%|####1                                                      |  0.1741/2.5 [00:07<01:36, 41.53s/sim_s]\u001b[A\n",
      "  8%|####5                                                      |  0.1916/2.5 [00:08<01:36, 41.75s/sim_s]\u001b[A\n",
      "  8%|####9                                                      |  0.2083/2.5 [00:09<01:38, 42.86s/sim_s]\u001b[A\n",
      "  9%|#####3                                                     |  0.2258/2.5 [00:09<01:39, 43.68s/sim_s]\u001b[A\n",
      " 10%|#####7                                                     |  0.2433/2.5 [00:10<01:38, 43.57s/sim_s]\u001b[A\n",
      " 10%|######1                                                    |  0.2608/2.5 [00:11<01:35, 42.87s/sim_s]\u001b[A\n",
      " 11%|######5                                                    |  0.2783/2.5 [00:12<01:34, 42.35s/sim_s]\u001b[A\n",
      " 12%|######9                                                    |  0.2958/2.5 [00:12<01:32, 42.10s/sim_s]\u001b[A\n",
      " 12%|#######3                                                   |  0.3125/2.5 [00:13<01:33, 42.68s/sim_s]\u001b[A\n",
      " 13%|#######7                                                   |  0.3300/2.5 [00:14<01:33, 43.24s/sim_s]\u001b[A\n",
      " 14%|########2                                                  |  0.3475/2.5 [00:15<01:32, 43.07s/sim_s]\u001b[A\n",
      " 15%|########6                                                  |  0.3650/2.5 [00:15<01:30, 42.42s/sim_s]\u001b[A\n",
      " 15%|#########                                                  |  0.3825/2.5 [00:16<01:29, 42.28s/sim_s]\u001b[A\n",
      " 16%|#########4                                                 |  0.4000/2.5 [00:17<01:27, 41.88s/sim_s]\u001b[A\n",
      " 17%|#########8                                                 |  0.4167/2.5 [00:17<01:28, 42.33s/sim_s]\u001b[A\n",
      " 17%|##########2                                                |  0.4342/2.5 [00:18<01:27, 42.46s/sim_s]\u001b[A\n",
      " 18%|##########6                                                |  0.4517/2.5 [00:19<01:27, 42.51s/sim_s]\u001b[A\n",
      " 19%|###########                                                |  0.4691/2.5 [00:20<01:26, 42.57s/sim_s]\u001b[A\n",
      " 19%|###########4                                               |  0.4866/2.5 [00:20<01:25, 42.27s/sim_s]\u001b[A\n",
      " 20%|###########8                                               |  0.5041/2.5 [00:21<01:24, 42.41s/sim_s]\u001b[A\n",
      " 21%|############2                                              |  0.5208/2.5 [00:22<01:24, 42.81s/sim_s]\u001b[A\n",
      " 22%|############7                                              |  0.5383/2.5 [00:23<01:24, 43.26s/sim_s]\u001b[A\n",
      " 22%|#############1                                             |  0.5558/2.5 [00:23<01:23, 42.71s/sim_s]\u001b[A\n",
      " 23%|#############5                                             |  0.5733/2.5 [00:24<01:22, 42.81s/sim_s]\u001b[A\n",
      " 24%|#############9                                             |  0.5908/2.5 [00:25<01:21, 42.58s/sim_s]\u001b[A\n",
      " 24%|##############3                                            |  0.6083/2.5 [00:26<01:19, 42.18s/sim_s]\u001b[A\n",
      " 25%|##############7                                            |  0.6250/2.5 [00:26<01:19, 42.33s/sim_s]\u001b[A\n",
      " 26%|###############1                                           |  0.6425/2.5 [00:27<01:19, 42.54s/sim_s]\u001b[A\n",
      " 26%|###############5                                           |  0.6600/2.5 [00:28<01:17, 42.02s/sim_s]\u001b[A\n",
      " 27%|###############9                                           |  0.6775/2.5 [00:29<01:16, 41.71s/sim_s]\u001b[A\n",
      " 28%|################4                                          |  0.6950/2.5 [00:29<01:14, 41.42s/sim_s]\u001b[A\n",
      " 28%|################8                                          |  0.7125/2.5 [00:30<01:13, 41.04s/sim_s]\u001b[A\n",
      " 29%|#################2                                         |  0.7292/2.5 [00:31<01:14, 41.91s/sim_s]\u001b[A\n",
      " 30%|#################6                                         |  0.7467/2.5 [00:31<01:14, 42.57s/sim_s]\u001b[A\n",
      " 31%|##################                                         |  0.7642/2.5 [00:32<01:13, 42.19s/sim_s]\u001b[A\n",
      " 31%|##################4                                        |  0.7816/2.5 [00:33<01:11, 41.50s/sim_s]\u001b[A\n",
      " 32%|##################8                                        |  0.7991/2.5 [00:34<01:10, 41.42s/sim_s]\u001b[A\n",
      " 33%|###################2                                       |  0.8166/2.5 [00:34<01:09, 41.20s/sim_s]\u001b[A\n",
      " 33%|###################6                                       |  0.8333/2.5 [00:35<01:09, 41.88s/sim_s]\u001b[A\n",
      " 34%|####################                                       |  0.8508/2.5 [00:36<01:09, 42.29s/sim_s]\u001b[A\n",
      " 35%|####################4                                      |  0.8683/2.5 [00:37<01:08, 42.11s/sim_s]\u001b[A\n",
      " 35%|####################9                                      |  0.8858/2.5 [00:37<01:08, 42.28s/sim_s]\u001b[A\n",
      " 36%|#####################3                                     |  0.9033/2.5 [00:38<01:07, 42.11s/sim_s]\u001b[A\n",
      " 37%|#####################7                                     |  0.9208/2.5 [00:39<01:06, 41.82s/sim_s]\u001b[A\n",
      " 38%|######################1                                    |  0.9375/2.5 [00:39<01:05, 41.92s/sim_s]\u001b[A\n",
      " 38%|######################5                                    |  0.9550/2.5 [00:40<01:05, 42.51s/sim_s]\u001b[A\n",
      " 39%|######################9                                    |  0.9725/2.5 [00:41<01:04, 42.39s/sim_s]\u001b[A\n",
      " 40%|#######################3                                   |  0.9900/2.5 [00:42<01:03, 42.34s/sim_s]\u001b[A\n",
      " 40%|#######################7                                   |  1.0075/2.5 [00:42<01:02, 41.88s/sim_s]\u001b[A\n",
      " 41%|########################1                                  |  1.0250/2.5 [00:43<01:01, 41.52s/sim_s]\u001b[A\n",
      " 42%|########################5                                  |  1.0417/2.5 [00:44<01:01, 42.15s/sim_s]\u001b[A\n",
      " 42%|########################9                                  |  1.0592/2.5 [00:45<01:01, 42.83s/sim_s]\u001b[A\n",
      " 43%|#########################4                                 |  1.0767/2.5 [00:45<01:00, 42.80s/sim_s]\u001b[A\n",
      " 44%|#########################8                                 |  1.0941/2.5 [00:46<00:58, 41.87s/sim_s]\u001b[A\n",
      " 44%|##########################2                                |  1.1116/2.5 [00:47<00:57, 41.74s/sim_s]\u001b[A\n",
      " 45%|##########################6                                |  1.1291/2.5 [00:47<00:56, 41.52s/sim_s]\u001b[A\n",
      " 46%|###########################                                |  1.1458/2.5 [00:48<00:56, 41.66s/sim_s]\u001b[A\n",
      " 47%|###########################4                               |  1.1633/2.5 [00:49<00:56, 42.24s/sim_s]\u001b[A\n",
      " 47%|###########################8                               |  1.1808/2.5 [00:50<00:55, 42.23s/sim_s]\u001b[A\n",
      " 48%|############################2                              |  1.1983/2.5 [00:50<00:54, 41.91s/sim_s]\u001b[A\n",
      " 49%|############################6                              |  1.2158/2.5 [00:51<00:53, 41.57s/sim_s]\u001b[A\n",
      " 49%|#############################1                             |  1.2333/2.5 [00:52<00:53, 41.92s/sim_s]\u001b[A\n",
      " 50%|#############################5                             |  1.2500/2.5 [00:53<00:52, 41.97s/sim_s]\u001b[A\n",
      " 51%|#############################9                             |  1.2675/2.5 [00:53<00:52, 42.63s/sim_s]\u001b[A\n",
      " 51%|##############################3                            |  1.2850/2.5 [00:54<00:50, 41.81s/sim_s]\u001b[A\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|##############################7                            |  1.3025/2.5 [00:55<00:49, 41.51s/sim_s]\u001b[A\n",
      " 53%|###############################1                           |  1.3200/2.5 [00:55<00:48, 41.36s/sim_s]\u001b[A\n",
      " 53%|###############################5                           |  1.3375/2.5 [00:56<00:47, 41.23s/sim_s]\u001b[A\n",
      " 54%|###############################9                           |  1.3542/2.5 [00:57<00:47, 41.53s/sim_s]\u001b[A\n",
      " 55%|################################3                          |  1.3717/2.5 [00:58<00:47, 42.41s/sim_s]\u001b[A\n",
      " 56%|################################7                          |  1.3892/2.5 [00:58<00:46, 41.94s/sim_s]\u001b[A\n",
      " 56%|#################################1                         |  1.4066/2.5 [00:59<00:46, 42.21s/sim_s]\u001b[A\n",
      " 57%|#################################6                         |  1.4241/2.5 [01:00<00:45, 42.45s/sim_s]\u001b[A\n",
      " 58%|##################################                         |  1.4416/2.5 [01:01<00:44, 42.05s/sim_s]\u001b[A\n",
      " 58%|##################################4                        |  1.4583/2.5 [01:01<00:44, 42.35s/sim_s]\u001b[A\n",
      " 59%|##################################8                        |  1.4758/2.5 [01:02<00:43, 42.76s/sim_s]\u001b[A\n",
      " 60%|###################################2                       |  1.4933/2.5 [01:03<00:42, 42.45s/sim_s]\u001b[A\n",
      " 60%|###################################6                       |  1.5108/2.5 [01:04<00:41, 42.20s/sim_s]\u001b[A\n",
      " 61%|####################################                       |  1.5283/2.5 [01:04<00:40, 41.99s/sim_s]\u001b[A\n",
      " 62%|####################################4                      |  1.5458/2.5 [01:05<00:39, 41.72s/sim_s]\u001b[A\n",
      " 62%|####################################8                      |  1.5625/2.5 [01:06<00:39, 41.84s/sim_s]\u001b[A\n",
      " 63%|#####################################2                     |  1.5800/2.5 [01:06<00:39, 42.69s/sim_s]\u001b[A\n",
      " 64%|#####################################7                     |  1.5975/2.5 [01:07<00:37, 42.01s/sim_s]\u001b[A\n",
      " 65%|######################################1                    |  1.6150/2.5 [01:08<00:37, 42.19s/sim_s]\u001b[A\n",
      " 65%|######################################5                    |  1.6325/2.5 [01:09<00:36, 42.09s/sim_s]\u001b[A\n",
      " 66%|######################################9                    |  1.6500/2.5 [01:09<00:35, 42.25s/sim_s]\u001b[A\n",
      " 67%|#######################################3                   |  1.6667/2.5 [01:10<00:35, 42.48s/sim_s]\u001b[A\n",
      " 67%|#######################################7                   |  1.6842/2.5 [01:11<00:35, 43.25s/sim_s]\u001b[A\n",
      " 68%|########################################1                  |  1.7017/2.5 [01:12<00:34, 42.87s/sim_s]\u001b[A\n",
      " 69%|########################################5                  |  1.7191/2.5 [01:12<00:33, 42.29s/sim_s]\u001b[A\n",
      " 69%|########################################9                  |  1.7366/2.5 [01:13<00:32, 42.22s/sim_s]\u001b[A\n",
      " 70%|#########################################3                 |  1.7541/2.5 [01:14<00:31, 41.87s/sim_s]\u001b[A\n",
      " 71%|#########################################7                 |  1.7708/2.5 [01:15<00:30, 42.51s/sim_s]\u001b[A\n",
      " 72%|##########################################2                |  1.7883/2.5 [01:15<00:30, 42.92s/sim_s]\u001b[A\n",
      " 72%|##########################################6                |  1.8058/2.5 [01:16<00:29, 42.40s/sim_s]\u001b[A\n",
      " 73%|###########################################                |  1.8233/2.5 [01:17<00:28, 41.76s/sim_s]\u001b[A\n",
      " 74%|###########################################4               |  1.8408/2.5 [01:17<00:27, 41.40s/sim_s]\u001b[A\n",
      " 74%|###########################################8               |  1.8583/2.5 [01:18<00:26, 41.13s/sim_s]\u001b[A\n",
      " 75%|############################################2              |  1.8750/2.5 [01:19<00:25, 41.46s/sim_s]\u001b[A\n",
      " 76%|############################################6              |  1.8925/2.5 [01:20<00:25, 42.53s/sim_s]\u001b[A\n",
      " 76%|#############################################              |  1.9100/2.5 [01:20<00:24, 41.81s/sim_s]\u001b[A\n",
      " 77%|#############################################4             |  1.9275/2.5 [01:21<00:23, 41.34s/sim_s]\u001b[A\n",
      " 78%|#############################################9             |  1.9450/2.5 [01:22<00:22, 41.10s/sim_s]\u001b[A\n",
      " 78%|##############################################3            |  1.9625/2.5 [01:23<00:22, 42.03s/sim_s]\u001b[A\n",
      " 79%|##############################################7            |  1.9792/2.5 [01:23<00:22, 42.56s/sim_s]\u001b[A\n",
      " 80%|###############################################1           |  1.9967/2.5 [01:24<00:21, 42.63s/sim_s]\u001b[A\n",
      " 81%|###############################################5           |  2.0142/2.5 [01:25<00:20, 42.52s/sim_s]\u001b[A\n",
      " 81%|###############################################9           |  2.0316/2.5 [01:26<00:19, 42.65s/sim_s]\u001b[A\n",
      " 82%|################################################3          |  2.0491/2.5 [01:26<00:19, 43.33s/sim_s]\u001b[A\n",
      " 83%|################################################7          |  2.0666/2.5 [01:27<00:18, 42.64s/sim_s]\u001b[A\n",
      " 83%|#################################################1         |  2.0833/2.5 [01:28<00:17, 42.62s/sim_s]\u001b[A\n",
      " 84%|#################################################5         |  2.1008/2.5 [01:29<00:17, 43.22s/sim_s]\u001b[A\n",
      " 85%|#################################################9         |  2.1183/2.5 [01:29<00:16, 42.93s/sim_s]\u001b[A\n",
      " 85%|##################################################4        |  2.1358/2.5 [01:30<00:15, 42.53s/sim_s]\u001b[A\n",
      " 86%|##################################################8        |  2.1533/2.5 [01:31<00:14, 42.68s/sim_s]\u001b[A\n",
      " 87%|###################################################2       |  2.1708/2.5 [01:31<00:14, 42.66s/sim_s]\u001b[A\n",
      " 88%|###################################################6       |  2.1875/2.5 [01:32<00:13, 42.67s/sim_s]\u001b[A\n",
      " 88%|####################################################       |  2.2050/2.5 [01:33<00:12, 43.15s/sim_s]\u001b[A\n",
      " 89%|####################################################4      |  2.2225/2.5 [01:34<00:11, 42.61s/sim_s]\u001b[A\n",
      " 90%|####################################################8      |  2.2400/2.5 [01:34<00:11, 42.87s/sim_s]\u001b[A\n",
      " 90%|#####################################################2     |  2.2575/2.5 [01:35<00:10, 42.08s/sim_s]\u001b[A\n",
      " 91%|#####################################################6     |  2.2750/2.5 [01:36<00:09, 42.06s/sim_s]\u001b[A\n",
      " 92%|######################################################     |  2.2917/2.5 [01:37<00:08, 42.92s/sim_s]\u001b[A\n",
      " 92%|######################################################4    |  2.3092/2.5 [01:37<00:08, 43.56s/sim_s]\u001b[A\n",
      " 93%|######################################################9    |  2.3267/2.5 [01:38<00:07, 42.75s/sim_s]\u001b[A\n",
      " 94%|#######################################################3   |  2.3441/2.5 [01:39<00:06, 42.20s/sim_s]\u001b[A\n",
      " 94%|#######################################################7   |  2.3616/2.5 [01:40<00:05, 42.01s/sim_s]\u001b[A\n",
      " 95%|########################################################1  |  2.3791/2.5 [01:40<00:05, 42.63s/sim_s]\u001b[A\n",
      " 96%|########################################################5  |  2.3958/2.5 [01:41<00:04, 42.98s/sim_s]\u001b[A\n",
      " 97%|########################################################9  |  2.4133/2.5 [01:42<00:03, 44.25s/sim_s]\u001b[A\n",
      " 97%|#########################################################3 |  2.4308/2.5 [01:43<00:02, 43.09s/sim_s]\u001b[A\n",
      " 98%|#########################################################7 |  2.4483/2.5 [01:43<00:02, 43.21s/sim_s]\u001b[A\n",
      " 99%|##########################################################1|  2.4658/2.5 [01:44<00:01, 43.11s/sim_s]\u001b[A\n",
      " 99%|##########################################################6|  2.4833/2.5 [01:45<00:00, 43.11s/sim_s]\u001b[A\n",
      "100%|###########################################################|  2.5000/2.5 [01:46<00:00, 42.45s/sim_s]\u001b[A"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(25, 201, 401, 101)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "time = 0.\n",
    "target_time = -2.5\n",
    "times = np.linspace(0, target_time, 25)\n",
    "initial_values = values\n",
    "all_values = hj.solve(solver_settings, dynamics, grid, times, initial_values)\n",
    "print(all_values.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "LIdzWddpjaDE",
    "outputId": "752aff0f-fc0a-4e47-d6e8-989c7011e553"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(25, 201, 401, 101)\n",
      "(201, 401, 101)\n",
      "(201, 401, 101, 3)\n"
     ]
    }
   ],
   "source": [
    "print(all_values.shape)\n",
    "Vmax = jnp.max(all_values, axis=0)\n",
    "print(Vmax.shape)\n",
    "print(grid.states.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 632
    },
    "id": "vi6UmqMfjjCo",
    "outputId": "cbbde3a5-e8a6-417c-8a8b-49b32fb734d5"
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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UmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaC0Ol7NmzeKkk06irKyMSCTCQw89FLc+Eok0uPziF79o9Jj33HNPg/ts3LixxS9IkiRJkrTj7HbeeefVy2CjRo1qdX8tDpeVlZUMHz6c2267rcH1y5cvj1v+9Kc/EYlE+OIXv9jkcYuLi+vt27Fjx5aWJ0mSJElix9kN4Pjjj4/LYI8++mir+2vf0h3GjRvHuHHjGl3ft2/fuOcPP/wwRx11FLvttluTx41EIvX2lSRJkiS1zo6yG0BRUVHSclhKr7n86KOPeOSRRzj//PN3uO26desYOHAg/fr143Of+xxz585tcvuqqioqKiriFkmSJEnKdYk5qKqqqtXHmjFjBr1792aPPfbgm9/8JitXrmz1sVo8ctkS9957L127duULX/hCk9vttdde3HPPPey7775UVFTw61//mkMOOYRXXnmFIUOGNLjPxIkTmTBhQirKliRJkpTvLgN2ynQRCdYB/4D+/fvHNY8fP55oNNriw40bN47TTz+dgQMHsmDBAq677jqOPvpoXn75ZYqKilp8vEgsFou1eK8tO0ci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fV5dXc33v/99li5dSqdOndhnn3145JFHOOGEE1JRmiRJkiTlhY8++oizzz6b5cuXU1JSwn777ce///1vjj322JT0l9IJfdJp67nRXnMpSZIktQFec5l0m6+5bKvvaUrvcylJkiRJyg+GS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaIZLSZIkSVJohktJkiRJUmiGS0mSJElSaEkPl9FolEgkErf07du3yX1mzpzJyJEj6dixI7vttht33HFHssuSJEmSpLzzu9/9jkGDBtGxY0dGjhzJM888k7K+UjJyuc8++7B8+fKty2uvvdbotgsWLOCEE07gsMMOY+7cuVx77bV897vfZfLkyakoTZIk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P378Dv8d8XPecr/73e8YNGgQHTt2ZOTIkTzzzDNNbj9z5kxGjhxJx44d2W233bjjjjvSVGn2mzhxIgceeCBdu3ald+/enHrqqbzzzjtN7tPYv/lvv/12mqrObtFotN5717dv3yb38TMeTkP/v4xEIlx00UUNbu9nvOVmzZrFSSedRFlZGZFIhIceeihufWu/f0yePJmhQ4dSVFTE0KFDmTp1aopegdIha8JldXU1p59+Ot/+9rcbXL9p0yZOPPFEKisrefbZZ5k0aRKTJ0/me9/7XpPHveyyy5g6dSqTJk3i2WefZd26dXzuc59j06ZNqXgZWWvMmDEsX748bvnGN77BrrvuygEHHNDkvscff3zcfo8++miaqs4NP/7xj+Pevx/96EdNbv/zn/+cm266idtuu43Zs2fTt29fjj32WNauXZumirPX22+/TV1dHb///e954403uPnmm7njjju49tprd7ivn/Pme/DBB7nsssv44Q9/yNy5cznssMMYN25c3C2qtrdgwQJOOOEEDjvsMObOncu1117Ld7/7Xe+F3EwzZ87koosu4oUXXmD69OnU1tYyduxYKisrd7jvO++8E/e5HjJkSBoqzg377LNP3Hv32muvNbqtn/HwZs+eHfd+T58+HYDTTz+9yf38jDdfZWUlw4cP57bbbmtwfWu+fzz//POceeaZnH322bzyyiucffbZnHHGGbz44oupehlKtViWufvuu2MlJSX12h999NFYQUFBbOnSpVvbHnjggVhRUVGsvLy8wWOtWbMmVlhYGJs0adLWtqVLl8YKCgpi//73v5Neey6prq6O9e7dO/bjH/+4ye3OPffc2CmnnJKeonLQwIEDYzfffHOzt6+rq4v17ds39rOf/Wxr28aNG2MlJSWxO+64IwUV5r6f//znsUGDBjW5jZ/zljnooINiF154YVzbXnvtFbv66qsb3P7KK6+M7bXXXnFtF1xwQWzUqFEpqzGXrVy5MgbEZs6c2eg2Tz/9dAyIrV69On2F5ZDx48fHhg8f3uzt/Ywn36WXXhobPHhwrK6ursH1fsbDAWJTp07d+ry13z/OOOOM2PHHHx/Xdtxxx8XOOuuspNes9Miakcsdef755xk2bBhlZWVb24477jiqqqp4+eWXG9zn5ZdfpqamhrFjx25tKysrY9iwYTz33HMprzmbTZs2jVWrVnHeeeftcNsZM2bQu3dv9thjD775zW+ycuXK1BeYQ2688UZ69uzJ/vvvz/XXX9/kKZoLFixgxYoVcZ/poqIijjjiCD/TrVReXk6PHj12uJ2f8+aprq7m5ZdfjvuMAowdO7bRz+jzzz9fb/vjjjuOl156iZqampTVmqvKy8sBmvW5HjFiBKWlpRxzzDE8/fTTqS4tp8yfP5+ysjIGDRrEWWedxQcffNDotn7Gk6u6upr777+fr3/960QikSa39TOeHK39/tHYZ9/vLNkrZ8LlihUr6NOnT1xb9+7d6dChAytWrGh0nw4dOtC9e/e49j59+jS6jwJ33XUXxx13HP37929yu3HjxvGXv/yFp556il/96lfMnj2bo48+mqqqqjRVmt0uvfRSJk2axNNPP83FF1/MLbfcwne+851Gt9/yuU38XfAz3Trvv/8+v/nNb7jwwgub3M7PefOtWrWKTZs2tegz2tC/73369KG2tpZVq1alrNZcFIvFuOKKKzj00EMZNmxYo9uVlpZy5513MnnyZKZMmcKee+7JMcccw6xZs9JYbfY6+OCDue+++3j88cf5wx/+wIoVKxgzZgyffPJJg9v7GU+uhx56iDVr1jT5B3A/48nV2u8fjX32/c6SvdpnsvNoNMqECROa3Gb27Nk7vKZvi4b+OhWLxXb4V6tk7JOtWvPfYMmSJTz++OP87W9/2+HxzzzzzK2Phw0bxgEHHMDAgQN55JFH+MIXvtD6wrNYS97zyy+/fGvbfvvtR/fu3TnttNO2jmY2JvHzm0+f6Ya05nO+bNkyjj/+eE4//XS+8Y1vNLmvn/OWa+lntKHtG2pX0y6++GJeffVVnn322Sa323PPPdlzzz23Ph89ejSLFy/ml7/8JYcffniqy8x648aN2/p43333ZfTo0QwePJh7772XK664osF9/Iwnz1133cW4cePizmZL5Gc8NVrz/cPvLLklo+Hy4osv5qyzzmpym1133bVZx+rbt2+9i39Xr15NTU1Nvb+IbL9PdXU1q1evjhu9XLlyJWPGjGlWv9muNf8N7r77bnr27MnJJ5/c4v5KS0sZOHAg8+fPb/G+uSLM537LDKTvvfdeg+Fyy2yEK1asoLS0dGv7ypUrG/09yActfc+XLVvGUUcdxejRo7nzzjtb3J+f88b16tWLdu3a1furdFOf0b59+za4ffv27Zv8I4viXXLJJUybNo1Zs2bRr1+/Fu8/atQo7r///hRUlvu6dOnCvvvu2+i/CX7Gk2fhwoU8+eSTTJkypcX7+hlvvdZ+/2jss5/P31myXUbDZa9evejVq1dSjjV69Giuv/56li9fvvVD/cQTT1BUVMTIkSMb3GfkyJEUFhYyffp0zjjjDACWL1/O66+/zs9//vOk1NXWtfS/QSwW4+677+acc86hsLCwxf198sknLF68OO4fnnwT5nM/d+5cgEbfv0GDBtG3b1+mT5/OiBEjgODak5kzZ3LjjTe2ruAc0JL3fOnSpRx11FGMHDmSu+++m4KCll894Oe8cR06dGDkyJFMnz6dz3/+81vbp0+fzimnnNLgPqNHj+af//xnXNsTTzzBAQcc0Kp/h/JNLBbjkksuYerUqcyYMYNBgwa16jhz5871M91KVVVVvPXWWxx22GENrvcznjx33303vXv35sQTT2zxvn7GW6+13z9Gjx7N9OnT487UeuKJJ/JmkCcnZWgioRZbuHBhbO7cubEJEybEdtppp9jcuXNjc+fOja1duzYWi8VitbW1sWHDhsWOOeaY2Jw5c2JPPvlkrF+/frGLL7546zGWLFkS23PPPWMvvvji1rYLL7ww1q9fv9iTTz4ZmzNnTuzoo4+ODR8+PFZbW5v215gNnnzyyRgQe/PNNxtcv+eee8amTJkSi8VisbVr18a+973vxZ577rnYggULYk8//XRs9OjRsV122SVWUVGRzrKz0nPPPRe76aabYnPnzo198MEHsQcffDBWVlYWO/nkk+O22/49j8VisZ/97GexkpKS2JQpU2KvvfZa7Etf+lKstLTU97wZli5dGtt9991jRx99dGzJkiWx5cuXb1225+c8nEmTJsUKCwtjd911V+zNN9+MXXbZZbEuXbrEPvzww1gsFotdffXVsbPPPnvr9h988EGsc+fOscsvvzz25ptvxu66665YYWFh7B//+EemXkJW+fa3vx0rKSmJzZgxI+4zvX79+q3bJL7nN998c2zq1Kmxd999N/b666/Hrr766hgQmzx5ciZeQtb53ve+F5sxY0bsgw8+iL3wwguxz33uc7GuXbv6GU+xTZs2xQYMGBC76qqr6q3zMx7e2rVrt37/BrZ+R1m4cGEs9v/t3b3OYUEAxvGz4fiohESCAomClkYiQqeh0ohCzg24By0XIEoXoFaoaCQalYiohHvw0T1b7L7yYt9EdnZtbP6/ZKqZk8xMJpN5TnLO6LnzR6vVuvkz+Hw+l8vlUrfb1WazUbfbldvt1mKxePn48Ge8Tbh0HEeWZT2U6XR6bbPf71WtVuX3+xUKhdRut3W5XK71u93u4Znz+ax2u61QKCS/369arabD4fDCkb2XZrOpQqHwZb1lWRoOh5Kk0+mkSqWicDgs27YVj8flOA7z+6Tlcql8Pq9AICCfz6d0Oq1Op6Pj8XjT7vOcSz9+B97pdBSJROT1elUqlbRarV7c+/c0HA5/uc/cv4djnZvr9/tKJBLyeDzK5XI312I4jqNyuXzTfjabKZvNyuPxKJlMajAYvLjH7+urNf1537if816vp1QqJZ/Pp2AwqGKxqPF4/PrOv6lGo6FoNCrbthWLxVSv17Ver6/1rPG/YzKZyLIsbbfbhzrWuLmP61vui+M4kp47f5TL5Wv7D6PRSOl0WrZtK5PJEPDf3Dfp5xfjAAAAAAD8pv/mKhIAAAAAwL9DuAQAAAAAGCNcAgAAAACMES4BAAAAAMYIlwAAAAAAY4RLAAAAAIAxwiUAAAAAwBjhEgAAAABgjHAJAAAAADBGuAQAAAAAGCNcAgAAAACMES4BAAAAAMa+AwNwEJUOGDL8AAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 1200x1000 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.jet()\n",
    "plt.figure(figsize=(12, 10))\n",
    "plt.contourf(grid.coordinate_vectors[0], grid.coordinate_vectors[1], Vmax[:, :, int((n_xd-1)/2)].T)\n",
    "plt.colorbar()\n",
    "plt.contour(grid.coordinate_vectors[0],\n",
    "            grid.coordinate_vectors[1],\n",
    "            Vmax[:, :, int((n_xd-1)/2)].T,\n",
    "            levels=0,\n",
    "            colors=\"black\",\n",
    "            linewidths=3)\n",
    "plt.plot(rObs*np.cos(np.arange(360)*math.pi/180.0), firstObs+rObs*np.sin(np.arange(360)*math.pi/180.0), 'k--')\n",
    "plt.ylim([2.5, 22.5])\n",
    "plt.savefig('HJ_Racer_BRS.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "JAXpkjN4oLuq",
    "outputId": "ccb54427-d80e-4e57-9fab-142f70e1ff84"
   },
   "outputs": [],
   "source": [
    "left_grad_values, right_grad_values = grid.upwind_grad_values(solver_settings.upwind_scheme, Vmax)\n",
    "grad_values=(left_grad_values+right_grad_values)/2."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### The Following has a long runtime as it covers many runs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "eCsSiNmVkg0P",
    "outputId": "7ce2c18f-1ca1-4818-cbf0-9e2bc89f6992"
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "-2\n",
      "-6\n",
      "1\n",
      "2\n",
      "3\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[11], line 44\u001b[0m\n\u001b[1;32m     42\u001b[0m     g1 \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mint\u001b[39m(jnp\u001b[38;5;241m.\u001b[39mround((x[\u001b[38;5;241m0\u001b[39m]\u001b[38;5;241m-\u001b[39mlo_x)\u001b[38;5;241m*\u001b[39mk_x)\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[1;32m     43\u001b[0m     g3 \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mint\u001b[39m(jnp\u001b[38;5;241m.\u001b[39mround((x[\u001b[38;5;241m2\u001b[39m]\u001b[38;5;241m-\u001b[39mlo_xdot)\u001b[38;5;241m*\u001b[39mk_xd)\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[0;32m---> 44\u001b[0m     VALUE[k,j] \u001b[38;5;241m=\u001b[39m \u001b[43mVmax\u001b[49m\u001b[43m[\u001b[49m\u001b[43mg1\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mint\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43mt\u001b[49m\u001b[38;5;241;43m+\u001b[39;49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mg3\u001b[49m\u001b[43m]\u001b[49m \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m0.15\u001b[39m\u001b[38;5;241m*\u001b[39mnp\u001b[38;5;241m.\u001b[39mabsolute(xnp[\u001b[38;5;241m0\u001b[39m])\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m2\u001b[39m \u001b[38;5;241m+\u001b[39m \u001b[38;5;241m0.05\u001b[39m\u001b[38;5;241m*\u001b[39mnp\u001b[38;5;241m.\u001b[39mlinalg\u001b[38;5;241m.\u001b[39mnorm(u)\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39m\u001b[38;5;241m2\u001b[39m\n\u001b[1;32m     45\u001b[0m \u001b[38;5;250m    \u001b[39m\u001b[38;5;124;03m'''\u001b[39;00m\n\u001b[1;32m     46\u001b[0m \u001b[38;5;124;03m    # print(x0)\u001b[39;00m\n\u001b[1;32m     47\u001b[0m \u001b[38;5;124;03m    xnp = np.array(x)\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m     56\u001b[0m \u001b[38;5;124;03m    # VALUE[k,j] += -np.linalg.norm(xnp2[:2])**2\u001b[39;00m\n\u001b[1;32m     57\u001b[0m \u001b[38;5;124;03m    '''\u001b[39;00m\n\u001b[1;32m     59\u001b[0m \u001b[38;5;66;03m# print(np.max(VALUE, axis=0).shape)\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/array.py:317\u001b[0m, in \u001b[0;36mArrayImpl.__getitem__\u001b[0;34m(self, idx)\u001b[0m\n\u001b[1;32m    315\u001b[0m   \u001b[38;5;28;01mreturn\u001b[39;00m lax_numpy\u001b[38;5;241m.\u001b[39m_rewriting_take(\u001b[38;5;28mself\u001b[39m, idx)\n\u001b[1;32m    316\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[0;32m--> 317\u001b[0m   \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mlax_numpy\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_rewriting_take\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43midx\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/numpy/lax_numpy.py:4105\u001b[0m, in \u001b[0;36m_rewriting_take\u001b[0;34m(arr, idx, indices_are_sorted, unique_indices, mode, fill_value)\u001b[0m\n\u001b[1;32m   4096\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21m_rewriting_take\u001b[39m(arr, idx, indices_are_sorted\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m, unique_indices\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m,\n\u001b[1;32m   4097\u001b[0m                     mode\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mNone\u001b[39;00m, fill_value\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mNone\u001b[39;00m):\n\u001b[1;32m   4098\u001b[0m   \u001b[38;5;66;03m# Computes arr[idx].\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m   4102\u001b[0m   \u001b[38;5;66;03m# For simplicity of generated primitives, we call lax.dynamic_slice in the\u001b[39;00m\n\u001b[1;32m   4103\u001b[0m   \u001b[38;5;66;03m# simplest cases: i.e. non-dynamic arrays indexed with integers and slices.\u001b[39;00m\n\u001b[0;32m-> 4105\u001b[0m   \u001b[38;5;28;01mif\u001b[39;00m (result \u001b[38;5;241m:=\u001b[39m \u001b[43m_attempt_rewriting_take_via_slice\u001b[49m\u001b[43m(\u001b[49m\u001b[43marr\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43midx\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmode\u001b[49m\u001b[43m)\u001b[49m) \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m \u001b[38;5;28;01mNone\u001b[39;00m:\n\u001b[1;32m   4106\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m result\n\u001b[1;32m   4108\u001b[0m   \u001b[38;5;66;03m# TODO(mattjj,dougalm): expand dynamic shape indexing support\u001b[39;00m\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/numpy/lax_numpy.py:4090\u001b[0m, in \u001b[0;36m_attempt_rewriting_take_via_slice\u001b[0;34m(arr, idx, mode)\u001b[0m\n\u001b[1;32m   4088\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mlen\u001b[39m(start_indices) \u001b[38;5;241m>\u001b[39m \u001b[38;5;241m1\u001b[39m:\n\u001b[1;32m   4089\u001b[0m   start_indices \u001b[38;5;241m=\u001b[39m util\u001b[38;5;241m.\u001b[39mpromote_dtypes(\u001b[38;5;241m*\u001b[39mstart_indices)\n\u001b[0;32m-> 4090\u001b[0m arr \u001b[38;5;241m=\u001b[39m \u001b[43mlax\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mdynamic_slice\u001b[49m\u001b[43m(\u001b[49m\u001b[43marr\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstart_indices\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mstart_indices\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mslice_sizes\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43mslice_sizes\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   4091\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m int_indices:\n\u001b[1;32m   4092\u001b[0m   arr \u001b[38;5;241m=\u001b[39m lax\u001b[38;5;241m.\u001b[39msqueeze(arr, \u001b[38;5;28mtuple\u001b[39m(int_indices))\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/lax/slicing.py:105\u001b[0m, in \u001b[0;36mdynamic_slice\u001b[0;34m(operand, start_indices, slice_sizes)\u001b[0m\n\u001b[1;32m     63\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mdynamic_slice\u001b[39m(\n\u001b[1;32m     64\u001b[0m     operand: Union[Array, np\u001b[38;5;241m.\u001b[39mndarray],\n\u001b[1;32m     65\u001b[0m     start_indices: Union[Union[Array, np\u001b[38;5;241m.\u001b[39mndarray], Sequence[ArrayLike]],\n\u001b[1;32m     66\u001b[0m     slice_sizes: Shape,\n\u001b[1;32m     67\u001b[0m ) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Array:\n\u001b[1;32m     68\u001b[0m \u001b[38;5;250m  \u001b[39m\u001b[38;5;124;03m\"\"\"Wraps XLA's `DynamicSlice\u001b[39;00m\n\u001b[1;32m     69\u001b[0m \u001b[38;5;124;03m  <https://www.tensorflow.org/xla/operation_semantics#dynamicslice>`_\u001b[39;00m\n\u001b[1;32m     70\u001b[0m \u001b[38;5;124;03m  operator.\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    103\u001b[0m \u001b[38;5;124;03m           [ 8,  9, 10, 11]], dtype=int32)\u001b[39;00m\n\u001b[1;32m    104\u001b[0m \u001b[38;5;124;03m  \"\"\"\u001b[39;00m\n\u001b[0;32m--> 105\u001b[0m   start_indices \u001b[38;5;241m=\u001b[39m \u001b[43m_dynamic_slice_indices\u001b[49m\u001b[43m(\u001b[49m\u001b[43moperand\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mstart_indices\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    106\u001b[0m   \u001b[38;5;28;01mif\u001b[39;00m jax\u001b[38;5;241m.\u001b[39mconfig\u001b[38;5;241m.\u001b[39mjax_dynamic_shapes:\n\u001b[1;32m    107\u001b[0m     dynamic_sizes, static_sizes \u001b[38;5;241m=\u001b[39m lax\u001b[38;5;241m.\u001b[39m_extract_tracers_dyn_shape(slice_sizes)\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/lax/slicing.py:2175\u001b[0m, in \u001b[0;36m_dynamic_slice_indices\u001b[0;34m(operand, start_indices)\u001b[0m\n\u001b[1;32m   2173\u001b[0m     result\u001b[38;5;241m.\u001b[39mappend(i \u001b[38;5;241m+\u001b[39m lax\u001b[38;5;241m.\u001b[39mconvert_element_type(d, _dtype(i)) \u001b[38;5;28;01mif\u001b[39;00m i \u001b[38;5;241m<\u001b[39m \u001b[38;5;241m0\u001b[39m \u001b[38;5;28;01melse\u001b[39;00m i)\n\u001b[1;32m   2174\u001b[0m     \u001b[38;5;28;01mcontinue\u001b[39;00m\n\u001b[0;32m-> 2175\u001b[0m   d_arr \u001b[38;5;241m=\u001b[39m \u001b[43mlax\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mconvert_element_type\u001b[49m\u001b[43m(\u001b[49m\u001b[43md\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43m_dtype\u001b[49m\u001b[43m(\u001b[49m\u001b[43mi\u001b[49m\u001b[43m)\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   2176\u001b[0m   result\u001b[38;5;241m.\u001b[39mappend(lax\u001b[38;5;241m.\u001b[39mselect(i \u001b[38;5;241m<\u001b[39m \u001b[38;5;241m0\u001b[39m, i \u001b[38;5;241m+\u001b[39m d_arr, i))\n\u001b[1;32m   2177\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m result\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/lax/lax.py:510\u001b[0m, in \u001b[0;36mconvert_element_type\u001b[0;34m(operand, new_dtype)\u001b[0m\n\u001b[1;32m    495\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mconvert_element_type\u001b[39m(operand: ArrayLike, new_dtype: DTypeLike) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Array:\n\u001b[1;32m    496\u001b[0m \u001b[38;5;250m  \u001b[39m\u001b[38;5;124;03m\"\"\"Elementwise cast.\u001b[39;00m\n\u001b[1;32m    497\u001b[0m \n\u001b[1;32m    498\u001b[0m \u001b[38;5;124;03m  Wraps XLA's `ConvertElementType\u001b[39;00m\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    508\u001b[0m \u001b[38;5;124;03m    An array with the same shape as `operand`, cast elementwise to `new_dtype`.\u001b[39;00m\n\u001b[1;32m    509\u001b[0m \u001b[38;5;124;03m  \"\"\"\u001b[39;00m\n\u001b[0;32m--> 510\u001b[0m   \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43m_convert_element_type\u001b[49m\u001b[43m(\u001b[49m\u001b[43moperand\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnew_dtype\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mweak_type\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[38;5;28;43;01mFalse\u001b[39;49;00m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/anaconda3/envs/clean_corl/lib/python3.9/site-packages/jax/_src/lax/lax.py:543\u001b[0m, in \u001b[0;36m_convert_element_type\u001b[0;34m(operand, new_dtype, weak_type)\u001b[0m\n\u001b[1;32m    537\u001b[0m \u001b[38;5;66;03m# Python has big integers, but convert_element_type(2 ** 100, np.float32) need\u001b[39;00m\n\u001b[1;32m    538\u001b[0m \u001b[38;5;66;03m# not be an error since the target dtype fits the value. Handle this case by\u001b[39;00m\n\u001b[1;32m    539\u001b[0m \u001b[38;5;66;03m# converting to a NumPy array before calling bind. Without this step, we'd\u001b[39;00m\n\u001b[1;32m    540\u001b[0m \u001b[38;5;66;03m# first canonicalize the input to a value of dtype int32 or int64, leading to\u001b[39;00m\n\u001b[1;32m    541\u001b[0m \u001b[38;5;66;03m# an overflow error.\u001b[39;00m\n\u001b[1;32m    542\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mtype\u001b[39m(operand) \u001b[38;5;129;01mis\u001b[39;00m \u001b[38;5;28mint\u001b[39m:\n\u001b[0;32m--> 543\u001b[0m   operand \u001b[38;5;241m=\u001b[39m \u001b[43mnp\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43masarray\u001b[49m\u001b[43m(\u001b[49m\u001b[43moperand\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mastype\u001b[49m\u001b[43m(\u001b[49m\u001b[43mnew_dtype\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    544\u001b[0m   old_weak_type \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mFalse\u001b[39;00m\n\u001b[1;32m    546\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m (old_dtype, old_weak_type) \u001b[38;5;241m==\u001b[39m (new_dtype, weak_type) \u001b[38;5;129;01mand\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(operand, Array):\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "np.random.seed(31415926)\n",
    "\n",
    "Runs = 25\n",
    "off1 = int(np.random.poisson(60, 1) - 60)\n",
    "off3 = int(np.random.poisson(35, 1) - 35)\n",
    "\n",
    "tmp = np.zeros(2)\n",
    "n_check = 20\n",
    "VALUE = np.zeros((2, 2*n_check+1))\n",
    "COST = np.zeros(Runs*(n_y-1))\n",
    "\n",
    "# Toggle adversarial and non-adversarial\n",
    "adversarialW = False\n",
    "\n",
    "print(off1)\n",
    "print(off3)\n",
    "\n",
    "g1, g3 = int((n_x-1)/2)+off1, int((n_xd-1)/2)+off3\n",
    "X = grid.states[g1, 0, g3]\n",
    "X = jnp.reshape(X,(1, 3))\n",
    "\n",
    "for r in range(1, Runs+1):\n",
    "    #if r % 5 ==0:\n",
    "    #    print(r)\n",
    "    print(r)\n",
    "    \n",
    "    x = grid.states[g1, 0, g3]\n",
    "    gv = grad_values[g1, 0, g3]\n",
    "\n",
    "    for t in range(n_y-1):\n",
    "        #u0, w = dynamics.optimal_control_and_disturbance(x, float(t)-25., gv)\n",
    "        #print(u, w)\n",
    "        #print(x)\n",
    "        x0 = x\n",
    "        for j in range(2*n_check+1):\n",
    "            for k in range(2):\n",
    "                u = jnp.array([float(0.75*((float(j)/float(n_check)) - 1.0))])\n",
    "                w = jnp.array([float(float(k)-0.5)])\n",
    "                x = x0 + dynamics.open_loop_dynamics(x0, target_time) + dynamics.control_jacobian(x0, target_time)@u + dynamics.disturbance_jacobian(x0, target_time)@w\n",
    "                xnp = np.array(x)\n",
    "\n",
    "                g1 = int(jnp.round((x[0]-lo_x)*k_x)-1)\n",
    "                g3 = int(jnp.round((x[2]-lo_xdot)*k_xd)-1)\n",
    "                VALUE[k,j] = Vmax[g1, int(t+1), g3] + 0.15*np.absolute(xnp[0])**2 + 0.05*np.linalg.norm(u)**2\n",
    "                '''\n",
    "                # print(x0)\n",
    "                xnp = np.array(x)\n",
    "                tmp[0] = (xnp[1] - firstObs) % spacerD\n",
    "                tmp[1] = spacerD - tmp[0]\n",
    "                xnp2 = xnp\n",
    "                xnp2[1] = np.min(tmp)\n",
    "\n",
    "                VALUE[k, j] = 0.5*np.absolute(xnp[0])**2 + 0.1*np.absolute(u)**2\n",
    "                if xnp2[1] <= 4.*rObs:\n",
    "                    VALUE[k,j] += 75.0*(((rObs+1.)**2/np.sqrt(1.0+(rObs+2.0)*np.linalg.norm(xnp2[:2]))) - (1.0+rObs))\n",
    "                # VALUE[k,j] += -np.linalg.norm(xnp2[:2])**2\n",
    "                '''\n",
    "\n",
    "            # print(np.max(VALUE, axis=0).shape)\n",
    "            jj = np.argmin(np.max(VALUE, axis=0))\n",
    "            \n",
    "            if adversarialW:\n",
    "                # Adversarial\n",
    "                if np.argmax(VALUE[:,jj]) == 0:\n",
    "                    w = jnp.array([-0.5])\n",
    "                elif np.argmax(VALUE[:,jj]) == 1:\n",
    "                    w = jnp.array([0.5])\n",
    "                else:\n",
    "                    print('Hmmm, argmax might be malfunctioning')\n",
    "                    w = jnp.array([0.5])\n",
    "            elif 0:\n",
    "                # Sinusoid + random else\n",
    "                # w = jnp.array([float(np.argmax(VALUE[:,jj])-0.5)])\n",
    "                if (int(t/2) % spacerD <= firstObs) and (int(t/2) % spacerD >= firstObs-2.0*rObs):\n",
    "                    w = jnp.array([0.5])\n",
    "                elif (int(t/2) % spacerD <= firstObs+3.0*rObs) and (int(t/2) % spacerD >= firstObs):\n",
    "                    w = jnp.array([-0.5])\n",
    "                else:\n",
    "                    w = jnp.array([float(np.random.rand(1)[0]-0.5)])\n",
    "            else:\n",
    "                # Stochastic\n",
    "                w = jnp.array([float(np.random.rand(1)[0]-0.5)])\n",
    "\n",
    "        # breakpoint()\n",
    "        u = jnp.array([float(0.75*((float(jj)/float(n_check)) - 1.0))])\n",
    "        if r==1 and t==0:\n",
    "            U = u\n",
    "        else:\n",
    "            U = jnp.concatenate((U, u), axis=0)\n",
    "        #print(u, w)\n",
    "        #print(x0)\n",
    "        #print(x)\n",
    "        #print()\n",
    "        # u = u0*float(jj)/float(n_check)\n",
    "        x = x0 + dynamics.open_loop_dynamics(x0, target_time) + dynamics.control_jacobian(x0, target_time)@u + dynamics.disturbance_jacobian(x0, target_time)@w\n",
    "        # print(x[0]-x0[0])\n",
    "        xnp = np.array(x)\n",
    "        COST[(r-1)*(n_y-1) + t] = 0.15*np.absolute(xnp[0])**2 + 0.05*np.linalg.norm(u)**2\n",
    "        # breakpoint()\n",
    "        g1 = int(jnp.round((x[0]-lo_x)*k_x)-1)\n",
    "        g3 = int(jnp.round((x[2]-lo_xdot)*k_xd)-1)\n",
    "        x = grid.states[g1, int(t+1), g3]\n",
    "        gv = grad_values[g1, int(t+1), g3]\n",
    "        #print(x)\n",
    "        #print()\n",
    "        X = jnp.concatenate((X, jnp.reshape(x, (1, 3))), axis=0)\n",
    "\n",
    "        # breakpoint()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 863
    },
    "id": "9eVU0Dxz9oXk",
    "outputId": "a2a60d20-3ad4-4e49-83b4-2d47d05e46bd"
   },
   "outputs": [],
   "source": [
    "CostMat = np.zeros(4*Runs)\n",
    "plt.figure(figsize=(4,15))\n",
    "for kk in range(4):\n",
    "    plt.plot(rObs*np.cos(np.arange(360)*math.pi/180.0), spacerD*kk+firstObs+rObs*np.sin(np.arange(360)*math.pi/180.0), 'k')\n",
    "# plt.plot(3.0*np.cos(np.arange(360)*math.pi/180.0), 112.5+3.0*np.sin(np.arange(360)*math.pi/180.0), 'k')\n",
    "for r in range(Runs):\n",
    "    plt.plot(X[r*(n_y-1)+1:(r+1)*(n_y-1)-1,0], X[r*(n_y-1)+1:(r+1)*(n_y-1)-1,1], 'r')\n",
    "    for k in range(4):\n",
    "        CostMat[4*r+k] = np.sum(COST[(n_y-1)*r + k*100:(n_y-1)*r + (k+1)*100])\n",
    "        \n",
    "plt.plot([0, 0], [0, 200], 'k--')\n",
    "plt.title('Centerline Trajectories', fontsize=20)\n",
    "plt.xlabel('Deviation in x (m)', fontsize=16)\n",
    "plt.ylabel('Progress in y (m)', fontsize=16)\n",
    "plt.xlim([-10.5, 10.5])\n",
    "plt.ylim([0, 200])\n",
    "plt.savefig('images/HJ_Racer_Centerline_FinalRandom.png')\n",
    "plt.show()\n",
    "\n",
    "print('Average Cost: ', np.mean(CostMat))\n",
    "print('Average Cost High: ', np.mean(np.sort(CostMat)[-5:]))\n",
    "print('Average Cost Low: ', np.mean(np.sort(CostMat)[:5]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.savetxt('StochRuns'+str(Runs)+'_CostMat.csv', CostMat, delimiter=',')\n",
    "np.savetxt('StochRuns'+str(Runs)+'_X.csv', X, delimiter=',')\n",
    "np.savetxt('StochRuns'+str(Runs)+'_COST.csv', COST, delimiter=',')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "Ufilt = \n",
    "for i in r"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(5)\n",
    "plt.plot(np.arange(1, n_y)/2., U)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(4)\n",
    "s = np.random.poisson(25, 10000)\n",
    "count, bins, ignored = plt.hist(s, 10, density=True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "1woHapmemAOc"
   },
   "outputs": [],
   "source": [
    "### Deprecated from Quickstart.ipynb\n",
    "#\n",
    "#\n",
    "#\n",
    "#\n",
    "\n",
    "target_values = hj.step(solver_settings, dynamics, grid, time, values, target_time)\n",
    "print(target_values.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 523
    },
    "id": "TPBt--AB50cX",
    "outputId": "ba90e89e-3874-48d6-a0ae-7bcd4f993a88"
   },
   "outputs": [],
   "source": [
    "plt.jet()\n",
    "plt.figure(figsize=(13, 8))\n",
    "plt.contourf(grid.coordinate_vectors[0], grid.coordinate_vectors[1], target_values[:, :, int((n_xd-1)/2)].T)\n",
    "plt.colorbar()\n",
    "plt.contour(grid.coordinate_vectors[0],\n",
    "            grid.coordinate_vectors[1],\n",
    "            target_values[:, :, int((n_xd-1)/2)].T,\n",
    "            levels=0,\n",
    "            colors=\"black\",\n",
    "            linewidths=3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "rk7kb-FM9OeE"
   },
   "outputs": [],
   "source": [
    "vmin, vmax = all_values.min(), all_values.max()\n",
    "levels = np.linspace(round(vmin), round(vmax), round(vmax) - round(vmin) + 1)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "yC9Nfek2hLDF"
   },
   "outputs": [],
   "source": [
    "def render_frame(i, colorbar=False):\n",
    "    plt.contourf(grid.coordinate_vectors[0],\n",
    "                 grid.coordinate_vectors[1],\n",
    "                 all_values[i, :, :, 30].T,\n",
    "                 vmin=vmin,\n",
    "                 vmax=vmax,\n",
    "                 levels=levels)\n",
    "    if colorbar:\n",
    "        plt.colorbar()\n",
    "    plt.contour(grid.coordinate_vectors[0],\n",
    "                grid.coordinate_vectors[1],\n",
    "                target_values[:, :, 30].T,\n",
    "                levels=0,\n",
    "                colors=\"black\",\n",
    "                linewidths=3)\n",
    "\n",
    "fig = plt.figure(figsize=(13, 8))\n",
    "render_frame(0, True)\n",
    "animation = HTML(anim.FuncAnimation(fig, render_frame, all_values.shape[0], interval=50).to_html5_video())\n",
    "plt.close(); animation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "kT4cwF25zFnS"
   },
   "outputs": [],
   "source": [
    "go.Figure(data=go.Isosurface(x=grid.states[:, :, -1, :, 0].ravel(),\n",
    "                             y=grid.states[:, :, -1, :, 1].ravel(),\n",
    "                             z=grid.states[:, :, -1, :, 3].ravel(),\n",
    "                             value=target_values[:, :, -1, :].ravel(),\n",
    "                             colorscale='jet',\n",
    "                             isomin=0,\n",
    "                             surface_count=1,\n",
    "                             isomax=0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "id": "BJlafT5qCcY0"
   },
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "colab": {
   "collapsed_sections": [],
   "provenance": []
  },
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 1
}
