{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "194ff3e4",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from matplotlib import cm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d19b6bcf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def psi(x, y, a = 1):\n",
    "    return (1/16) * a * (-1 + x**2 + y**2) * (-9 + 16 * x**2 + 16 * y**2)\n",
    "\n",
    "def F(x):\n",
    "    w1, w2 = x[0], x[1]\n",
    "    L1 = psi(w1, w2) - w2\n",
    "    L2 = psi(w1, w2) + w1\n",
    "    return np.array([L1, L2])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "13a24abd",
   "metadata": {},
   "outputs": [],
   "source": [
    "def extragradient_constant(F, x0, L, max_iter=1000):\n",
    "    \"\"\"\n",
    "    Extragradient method for solving variational inequalities.\n",
    "\n",
    "    Parameters:\n",
    "    - F: Callable, the operator function.\n",
    "    - x0: numpy.ndarray, initial guess for the solution.\n",
    "    - step_size: float, step size for the updates.\n",
    "    - max_iter: int, maximum number of iterations.\n",
    "\n",
    "    Returns:\n",
    "    - distances: list, the relative distances to the origin at each iteration.\n",
    "    \"\"\"\n",
    "    x = x0\n",
    "    init_distance = np.linalg.norm(x0)\n",
    "    distances = [1]\n",
    "    step_size = 1/L\n",
    "\n",
    "    for _ in range(max_iter):\n",
    "        # First step\n",
    "        x_tilde = x - step_size * F(x)\n",
    "\n",
    "        # Second step\n",
    "        x_next = x - 0.5 * step_size * F(x_tilde)\n",
    "\n",
    "        x = x_next\n",
    "        distances.append(np.linalg.norm(x)/init_distance)\n",
    "\n",
    "    return distances\n",
    "\n",
    "\n",
    "def extragradient_adaptive(F, x0, L0, L1, max_iter=1000):\n",
    "    \"\"\"\n",
    "    Extragradient method with adaptive step size for solving variational inequalities.\n",
    "\n",
    "    Parameters:\n",
    "    - F: Callable, the operator function.\n",
    "    - x0: numpy.ndarray, initial guess for the solution.\n",
    "    - L0: float, initial Lipschitz constant estimate.\n",
    "    - L1: float, Lipschitz constant growth factor.\n",
    "    - max_iter: int, maximum number of iterations.\n",
    "\n",
    "    Returns:\n",
    "    - distances: list, the relative distances to the origin at each iteration.\n",
    "    - step_sizes: list, the step size values at each iteration.\n",
    "    \"\"\"\n",
    "    x = x0\n",
    "    init_distance = np.linalg.norm(x0)\n",
    "    distances = [1]\n",
    "    step_sizes = []\n",
    "\n",
    "    for _ in range(max_iter):\n",
    "        step_size = 1 / (L0 + L1 * (np.linalg.norm(F(x))))\n",
    "        step_sizes.append(step_size)\n",
    "\n",
    "        # First step\n",
    "        x_tilde = x - step_size * F(x)\n",
    "\n",
    "        # Second step\n",
    "        x_next = x - 0.5 * step_size * F(x_tilde)\n",
    "\n",
    "        x = x_next\n",
    "        distances.append(np.linalg.norm(x) / init_distance)\n",
    "\n",
    "    return distances, step_sizes"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
