{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Introduction plots in the paper demonstrating the idea of space-time separation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "c:\\PhD\\Work\\Papers\\ICLR_26\\submission\\swimpde-paper\\frozen_pinns\\examples\\burgers\\supplementary\\figures\\../../../../src\\swimpde\\Ansatz\\boundary_compliant_ansatz.py:14: SyntaxWarning: invalid escape sequence '\\p'\n",
      "  '''\n"
     ]
    }
   ],
   "source": [
    "import sys\n",
    "sys.path.append('../../../../')\n",
    "sys.path.append('../../../../src')\n",
    "from swimpde import Domain\n",
    "from swimpde import BasicAnsatz\n",
    "from swimpde import BurgersSolver\n",
    "import numpy as np\n",
    "from sklearn.metrics import mean_squared_error\n",
    "import scipy.io\n",
    "from matplotlib import ticker\n",
    "import time\n",
    "import matplotlib.pyplot as plt\n",
    "plt.rcParams['image.cmap'] = 'jet'"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Load data and reference solution"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load and visualize data\n",
    "data = scipy.io.loadmat('../../../../data/burgers_shock.mat')\n",
    "t_eval = data['t'].flatten()[:,None]\n",
    "x_eval = data['x'].flatten()[:,None]\n",
    "u_exact = np.real(data['usol']).T\n",
    "X, T = np.meshgrid(x_eval,t_eval)\n",
    "X_ = np.hstack((X.flatten()[:,None], T.flatten()[:,None]))\n",
    "\n",
    "# Set ground truth\n",
    "u_true = u_exact.flatten()[:,None]              "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Problem setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# initial condition\n",
    "def u0(x):\n",
    "    return -1 * np.sin(np.pi * x)\n",
    "\n",
    "# forcing\n",
    "def forcing(x, t):\n",
    "    return np.zeros(x.shape[0])\n",
    "\n",
    "# boundary condition\n",
    "boundary_condition = \"zero dirichlet\" # \n",
    "\n",
    "# Domain information and spacial points for the first time-block\n",
    "n_points_1d = 4000 # No. of points in space\n",
    "x_lim = [-1, 1] # Domain range\n",
    "\n",
    "# Interior points\n",
    "rng = np.random.default_rng(seed=123)\n",
    "x_space = rng.uniform(x_lim[0], x_lim[1], n_points_1d).reshape((-1, 1)) \n",
    "x_space_inner = x_space[1:-1]\n",
    "interior_points = x_space_inner\n",
    "\n",
    "# Boundary points (excluding corners)\n",
    "left = x_lim[0]\n",
    "right = x_lim[1]\n",
    "boundary_points = np.row_stack([left, right])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Train and evaluate ODE-ELM network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "rmse_elm, re_elm\n",
      "0.15175027104578626 0.15175027104578626 0.24701449545008689 0.24701449545008689\n",
      "[4.098177433013916, 0.15175027104578626, 0.0, 0.24701449545008689, 0.0]\n"
     ]
    }
   ],
   "source": [
    "# Hyper-parameters\n",
    "n_sample = 6000 # No. of sampling points (for computing gradients)\n",
    "n_col = 3000 # No. of collocation points (to be re-sampled)\n",
    "width = 2000 # Width\n",
    "reg_const = 1e-7 # Regularization constant\n",
    "svd_cutoff = 1e-10 # SVD threshold\n",
    "seeds = [1] # Seeds (to compute mean errors)\n",
    "time_blocks = 1 # Number of time-blocks for smapling\n",
    "info = [] # List to store errors and time measurements\n",
    "\n",
    "# Compute prob. distribution for (re)-sampling collocation points\n",
    "def collocation_points_probabilities(df_dx):\n",
    "    gradients = np.abs(df_dx)\n",
    "    gradients = gradients + 0.05 * np.max(gradients)\n",
    "    return gradients/np.sum(gradients)\n",
    "\n",
    "# Points where gradient of the solution at the end of a time-block is computed\n",
    "sample_test_points = np.sort(rng.uniform(x_lim[0] + 1e-4, x_lim[1] - 1e-4, n_sample)).reshape((-1, 1)) # This does not include boundary points\n",
    "\n",
    "# Domain\n",
    "domain = Domain(\n",
    "    interior_points=interior_points,\n",
    "    boundary_points=boundary_points,\n",
    "    sample_points = sample_test_points\n",
    ")\n",
    "\n",
    "# Parameter sampler for sampling of weights and biases using data-agnostic distribution\n",
    "def parameter_sampler_uniform(x, y, rng):\n",
    "    \"\"\"\n",
    "        returns: weights, biases, idx_from, idx_to\n",
    "    \"\"\"\n",
    "    n_dim = 1\n",
    "    x_left = -3\n",
    "    x_right = 3\n",
    "    # Sample weights from a normal distribution\n",
    "    weights = rng.normal(size=(n_dim, width)) #rng.uniform(x_left, x_right, size=(n_dim, n_OBF))\n",
    "    # Sample biases from a uniform distribution\n",
    "    biases = rng.uniform(x_left, x_right, size=(1, width))\n",
    "    idx_from = np.arange(width)\n",
    "    idx_to = np.arange(width)\n",
    "    return weights, biases, idx_from, idx_to\n",
    "\n",
    "# Loop over different seeds\n",
    "rel_err_elm = np.ones((len(seeds), ))\n",
    "time_elm = np.ones((len(seeds), ))\n",
    "rmse_elm = np.ones((len(seeds), ))\n",
    "j = 0\n",
    "\n",
    "# Loop over different seeds\n",
    "for seed in seeds: # Run for 3 seeds \n",
    "    # ODE-ELM network ansatz                     \n",
    "    ansatz_elm = BasicAnsatz(\n",
    "        n_neurons=width,\n",
    "        activation=\"tanh\",\n",
    "        random_state=seed,\n",
    "        regularization_scale=reg_const,\n",
    "        parameter_sampler = parameter_sampler_uniform  \n",
    "    )\n",
    "    # Burger solver\n",
    "    burgers_solver_elm = BurgersSolver(\n",
    "        domain=domain, \n",
    "        ansatz=ansatz_elm,\n",
    "        u0=u0,\n",
    "        boundary_condition=boundary_condition,\n",
    "        forcing=forcing,\n",
    "        regularization_scale=reg_const,\n",
    "        c=(0.01/np.pi)\n",
    "    )\n",
    "    # ELM fit\n",
    "    t_elm_start = time.time()\n",
    "    sol_elm, solver_status_elm = burgers_solver_elm.fit_time_blocks(t_span=[0, np.max(t_eval)], rtol=1e-8, atol=1e-8, svd_cutoff=svd_cutoff, time_blocks=time_blocks, prob_distr_resampling = collocation_points_probabilities, n_col=n_col, outer_basis=False);\n",
    "    t_elm_stop = time.time()\n",
    "    time_elm[j] = t_elm_stop - t_elm_start\n",
    "    # Evaluate ELM-ODE predictions\n",
    "    u_elm = (burgers_solver_elm.evaluate_blocks(x_eval= x_eval, t_eval = t_eval, time_blocks = time_blocks, solver_status = solver_status_elm)).T\n",
    "    # Compute metrics\n",
    "    mse_elm = mean_squared_error(u_true, u_elm.flatten()[:,None])  # mean squared error\n",
    "    rmse_elm[j] = np.sqrt(mse_elm)  # Root Mean Squared Error\n",
    "    rel_err_elm[j] = np.linalg.norm(u_true-u_elm.flatten()[:,None], 2)/np.linalg.norm(u_true,2)\n",
    "    print(\"rmse_elm, re_elm\")\n",
    "    print(rmse_elm[j], rmse_elm[j], rel_err_elm[j], rel_err_elm[j])\n",
    "    j += 1\n",
    "\n",
    "info.append(np.mean(time_elm))\n",
    "info.append(np.mean(rmse_elm))\n",
    "info.append(np.std(rmse_elm))\n",
    "info.append(np.mean(rel_err_elm))\n",
    "info.append(np.std(rel_err_elm))\n",
    "print(info)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Burgers equation: Errors and time measurements using ode-elm\n",
      "training time for ode-elm =  4.098177433013916\n",
      "rmse ode-elm =  0.15175027104578626 +- 0.0\n",
      "rel error ode-elm =  0.24701449545008689 +- 0.0\n"
     ]
    }
   ],
   "source": [
    "# Print errors and time measurements for Burgers with ODE-ELM\n",
    "res = np.vstack(info).reshape(-1)\n",
    "\n",
    "# Burgers time measurements\n",
    "print('Burgers equation: Errors and time measurements using ode-elm')\n",
    "print('training time for ode-elm = ', res[-5])\n",
    "print('rmse ode-elm = ', res[-4], '+-', res[-3])\n",
    "print('rel error ode-elm = ', res[-2], '+-', res[-1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(256, 2000)"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.shape(burgers_solver_elm.ansatz._model[0].transform(x_eval.reshape(-1,1))[:, ])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "<>:41: SyntaxWarning: invalid escape sequence '\\s'\n",
      "<>:41: SyntaxWarning: invalid escape sequence '\\s'\n",
      "C:\\Users\\chinm\\AppData\\Local\\Temp\\ipykernel_6220\\2687858901.py:41: SyntaxWarning: invalid escape sequence '\\s'\n",
      "  ax.set_ylabel(\"$\\sigma$ \" +  \"(wx$^T$+b)\", fontsize=fontsize) #u03C3\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from swimnetworks import Dense\n",
    "\n",
    "from sklearn.pipeline import Pipeline\n",
    "random_seed_tanh = 1\n",
    "\n",
    "basis_hidden_1 = Dense(\n",
    "    layer_width=10,\n",
    "    activation=np.tanh,\n",
    "    parameter_sampler=\"tanh\",\n",
    "    sample_uniformly=True,\n",
    "    random_seed=random_seed_tanh,\n",
    "    prune_duplicates=True\n",
    ")\n",
    "\n",
    "x_lim = [0, 1]\n",
    "n_points_grid = 50\n",
    "n_points_quad = 200\n",
    "grid_points = np.linspace(x_lim[0], x_lim[1], n_points_grid)\n",
    "steps = [\n",
    "    (\"hidden-1\", basis_hidden_1)\n",
    "]\n",
    "basis = Pipeline(steps=steps,verbose=False)\n",
    "basis.fit(grid_points.reshape(-1,1), np.zeros((grid_points.shape[0],)))\n",
    "\n",
    "fig, ax = plt.subplots(1,1,figsize=(2,2))\n",
    "fontsize = 22\n",
    "tanh_plot = burgers_solver_elm.ansatz._model[0].transform(x_eval.reshape(-1,1))\n",
    "#for i in range(0,100,20):\n",
    "#    if i == 0: pass\n",
    "#    else:ax.plot(x_eval, tanh_plot[:,i]) #burgers_solver_swim.ansatz.activation(x_eval) # \n",
    "\n",
    "# Use a different LaTeX font\n",
    "plt.rcParams['mathtext.fontset'] = 'cm'\n",
    "plt.rcParams['mathtext.rm'] = 'serif'\n",
    "\n",
    "ax.plot(x_eval, tanh_plot[:,0])\n",
    "ax.plot(x_eval, tanh_plot[:,1])\n",
    "ax.plot(x_eval, tanh_plot[:,3])\n",
    "ax.plot(x_eval, tanh_plot[:,9])\n",
    "#ax.set_title(\"Basis functions in space\", fontsize=fontsize)\n",
    "ax.set_ylabel(\"$\\sigma$ \" +  \"(wx$^T$+b)\", fontsize=fontsize) #u03C3\n",
    "ax.set_xlabel(r'$x$', fontsize=fontsize)\n",
    "ax.set_xticks([])\n",
    "ax.set_yticks([])\n",
    "ax.tick_params(axis='both', labelsize=fontsize)\n",
    "fig.tight_layout()\n",
    "fig.savefig(\"basis_functions.png\", dpi=600)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "29"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.shape(sol_elm.c_collection[0](t_eval.reshape(-1,)).T)[1]\n",
    "sol_elm._V_a.shape[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1,1,figsize=(2,2))\n",
    "i = 9\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[4,:])\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[5,:])\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[3,:])\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[6,:])\n",
    "plt.rcParams['mathtext.fontset'] = 'cm'\n",
    "plt.rcParams['mathtext.rm'] = 'serif'\n",
    "ax.set_ylabel(\"c(t)\", fontsize=fontsize)\n",
    "ax.set_xlabel(\"t\", fontsize=fontsize)\n",
    "ax.tick_params(axis='both', labelsize=fontsize)\n",
    "ax.set_xticks([])\n",
    "ax.set_yticks([])\n",
    "fig.tight_layout()\n",
    "fig.savefig(\"functions_coefficients.png\", dpi=600)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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z969e9m/f3+pX8WKovC///0PUH9lVHTC2G63M2PGDD744INqd6PMysqqdH5QUFC1d5ypU6fy2GOPUVBQwLx588oklXnz5gHqjnn77bdXa51VCQ4OrnR+ZZ93UlKSNt26desqt9WqVasaRle16OjoGpUpGXOxP/74g7Fjx9boh01Vn3vJZFyZ2vjOBQYGVprQamN/X79+vdYJprrS0tIIDw+v0TLlbftijzXnq85n9/bbbzNo0CAyMzNZt24d69atw2q10qNHD/r06cOgQYMYOHBguR1DqlIvkkplv0Qupuz5MjIyLnjZ868BuP766ystf/4OOGnSJO1LvWTJEl566SVt3o8//qh9CceOHYuvr2+567z33nu1A7fJZGL48OH07NmT5s2b4+Pjg8lkAiA5OZnp06cDlLpGpDxWq7XS+SUFBgZy4403smTJErZt28aBAwe0X5w7d+7UqtrXXnstzZo1q/Z6K3Mxn3fJax1KNglUxMfH54K3dTHrLFkmOzu71LzMzEwGDRqkJZvw8HBGjhxJhw4dCA0NxWKxaP9HH3/8MZ988gngus+9Nr5z9WF/h7L7fF1tu6rtVuez69q1K3v37uWJJ55gxYoV5Obmkp+fz5YtW9iyZQvPP/88oaGhPPTQQ8yaNatG/8/1IqnUlZIH67Vr11ba1u1q1113HY0aNSIjI4Nly5bx/PPPa1Xt6jR9JSYm8t577wHQrFkzNm3aVOGv4OKuwLXhn//8p9ZOPW/ePN544w1tuljxwcXdSh6sq3PR5vkX3LlCddZZsoyfn1+peW+99ZaWUCZOnMjChQsrbML46aefLiLSsjzlO3ehSu7vDzzwAK+88opbtl3Xx5piLVu2ZOHChbz77rv88ssvbNu2ja1bt7J582ZycnI4ffo0999/P3v37uXDDz+s9nrrxYn6ulKy2lidK4Aro6g96yp8nM9isWgnHJOSkvjuu+8Atcq+cuVKQN1xr7766nK3991332nt5w899FClzSrx8fEX9d4q07t3b6644gpATYb5+fmlTtC3bNmSYcOG1dr2a6Jk08X5J/HLc/ToUZfHcOTIkRqVOb+55ZtvvgHAaDTy5ptvVtom7urP3VO+cxfKlft7fdr2+by8vLjqqqv4z3/+w7p160hJSeH999/XapmLFi3it99+q/b6JKmU0L9/f216w4YNdb794l5gcK52snbtWjIzMwH1l2hF1dBTp05p08W9sCpS2+/tn//8J6BW8VesWFHqBP20adMuqsnClYovcASqdZHXBV0IVoVvv/22yjLFPzAAevXqVWpe8eceFBRE48aNK1xHQUGBy2+G50nfuQvRpUsXAgICAPWzrahjQXWV/F5XdX7J3ceaylgsFu68807uvvtu7bUff/yx2st7xt7tIbp166adA/jyyy9d3lxQlb59+2onjFevXk1OTk61e32VbMqp7Nfv0aNHWbx4sQuirdgtt9yi7azz5s3Tmr6MRmO1xzirC61atdI+77i4OO1Xf3k2btzo8p5fAIcOHeLLL7+scP7evXu1xBMWFlZmVIDizz05ObnSE+Cvv/56ld2Ia8qTvnMXonhEC4AzZ87w6quvXtT6SjZpVdWs6e5jTXWUvCmj3W6v9nKSVErQ6XQ8//zzgPpLY8yYMaV+JZYnKSmJuXPnsm/fPpfE8I9//ANQ2/jfffddvvrqK0A9sdaxY8cKlyv5q/vll18u9wBy/PhxRo0aVSvnBkry8fHREuDPP/+snaC/7rrraNq0aa1uu6ZKdt+eOnVquQfHo0ePVjn0zMW4/fbby+25lZSUxE033aSd2L7vvvu0JolixZ+7oig8+uij5a5/+fLl2r2OXMmTvnMX6pFHHtFu2zF79mxee+21SgfwzM3NZf78+VpzbkklD8JVXf/hzmPN7t27eeKJJzh58mSFZXJzc7Vzo0CNbjgmJ+rPM3LkSJ588kkef/xxzpw5w5AhQ+jXrx/Dhw8nMjISk8lERkYGf/zxBz///DPbt29HUZRqXW9QHbfddhtPPPEEiqIwe/Zs7RdCVdemxMbG0qtXL3755ReOHTtG+/btufPOO+nQoQMOh4Pt27ezdOlScnNzmTx58kWNslsdd911lzYOUjFPOUFf0uTJk1m+fDnffvstf//9N126dGHq1Kna9U47duxg4cKF5ObmMn78eO38lqua8MaOHcvq1avp1q0bkydPpk+fPhgMBnbt2sWCBQu0XkK9evXiwQcfLLP8Pffcw8KFC7Hb7bz11lvs2rWLcePG0axZM06fPs3nn3/Oxo0b8fX15brrruOzzz5zSdzged+5C9GsWTNWrFjBqFGjKCws5P777+edd97h+uuvJyYmBl9fX7Kzs4mPj2fnzp18//33FBQU8NRTT5VZV6dOnQgNDeX06dP873//Izg4mN69e5fqWTh8+HBt2l3HmszMTObOncuTTz5Jnz596NOnD+3atcPf35+MjAwOHTrE8uXLtQ4gvXv3rtY4ZhpXXZpf26MUV6YmwyNMmjRJKxsfH19hucWLFyuNGzcuNRxERQ8/P79Kx7uqqX79+pVav9FoVJKTk6tcLj4+XomKiqo01pkzZypHjx6t8v+25IixF6rk0BGtWrWqcqTY4vdQVWwl11uVqr5niqKOUjxo0KAK/88MBoPy3//+V/nggw+011atWlXltity/vf1tddeq3SU4tjY2EqH8ViwYEGlo3IHBQUpX3/9dZX/F+ePUlwd7vjO1caxYceOHUq7du2qtb8bDAblgw8+KHc9JYejKe9RHlcca2qyT2zevLla2wKUq666qlrHnpKk+asCt912G8eOHePNN9/U7q9gtVq1+0T07NmTu+66i5UrV3Lq1CmXDuFR8oQ9qL9umjRpUuVykZGR7N69m7lz53L55Zfj7e2Nt7c3rVq14tZbb2XTpk288cYbNbpS+mKUHPZk2rRpdbbdmvL19eW7775j8eLFDBo0iKCgILy8vIiMjOS2225j+/bt3HfffaWad6oaVqMm7r33XrZv387kyZOJiorCYrEQGBhI//79mT9/Plu3bq10e1OnTuWXX35h4sSJ2r0zAgMD6dKlC4899hj79u2rdAiai+Fp37kL1b17d+Li4li5ciW33nor0dHR+Pv7YzAYtHuqTJgwgffff58TJ05UOLr27bffzrfffssNN9xAixYtqnXhcF0fa/r378/+/ft59dVXGT9+PDExMdp79fHxoW3bttxyyy2sXbuWH374oVrHnpJ0iiI3BBe1o0uXLuzduxeTyURiYiKhoaHuDumi3HDDDaxatQpQr2qurLdVZRYtWqSdo/nwww8rvBmZEPWR1FRErdi2bRt79+4F1PMG9T2hJCQk8MUXXwBqsrzQhCLEpU6SinA5RVFK9Ta677773BdMNcTFxZGSklLh/BMnTnD99ddrw2MUX4cjhChLen8Jl9i/fz9///036enpfPTRR9qdFEeOHEnv3r3dHF3l1q9fz6OPPsqgQYPo27cvUVFReHl5cebMGbZv387KlSu1YVz69u3rssEwhbgUSVIRLvHKK6+UucAtJCSEd955x00R1UxRURFfffWVdl1QeQYPHszKlStdMmS/EJcq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",
      "text/plain": [
       "<Figure size 300x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1,1,figsize=(3,3))\n",
    "i = 9\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[4,:])\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[5,:])\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[3,:])\n",
    "ax.plot(t_eval, sol_elm.c_collection[0](t_eval.reshape(-1,))[6,:])\n",
    "ax.set_ylabel(\"c(t)\", fontsize=fontsize)\n",
    "ax.set_xlabel(\"t\", fontsize=fontsize)\n",
    "ax.tick_params(axis='both', labelsize=fontsize)\n",
    "ax.set_title(\"Time-varying parameters\", fontsize=fontsize)\n",
    "fig.tight_layout()\n",
    "fig.savefig(\"functions_coefficients.png\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "test2",
   "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.12.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
