{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e6c6951b-14c0-45bc-a225-6882fb6f3c50",
   "metadata": {},
   "outputs": [],
   "source": [
    "from joblib import Parallel, delayed  # for parallelization\n",
    "from tqdm import tqdm  # for progress bars\n",
    "from utils import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d845fa22-6a8c-4a86-8740-c441495b6e99",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "id": "d9b7c3b1-10a0-4acc-bbd1-966f887e1a3a",
   "metadata": {},
   "source": [
    "## Code reproducibility instructions\n",
    "\n",
    "Run all the chunks below in order."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8c013bbf-777d-4c9a-9557-ac34a57c9659",
   "metadata": {},
   "outputs": [],
   "source": [
    "## RUN SIMULATION FOR N ##\n",
    "\n",
    "N = 4\n",
    "# Define experiment parameters\n",
    "T_values = [100, 300, 600, 1200]\n",
    "num_runs = 35  # number of repetitions per T\n",
    "rng_master = np.random.default_rng(123)  # master RNG for reproducibility\n",
    "\n",
    "run_parallel = True\n",
    "\n",
    "# Boundaries for p (for each coordinate)\n",
    "p_m_arr = [0] * N\n",
    "p_M_arr = [3] * N\n",
    "\n",
    "# Define the \"true\" psi_i functions (for data generation & regret measurement)\n",
    "psi = []\n",
    "\n",
    "for i in range(N):\n",
    "    psi.append(lambda w, i=i: norm.cdf(2*w/(i+1)))  # using default argument to fix i\n",
    "\n",
    "# Define the true theta for each component\n",
    "if N ==2:\n",
    "    beta = 0.9\n",
    "\n",
    "if N ==4:\n",
    "    beta = 0.9\n",
    "\n",
    "if N ==6:\n",
    "    beta = 0.92\n",
    "\n",
    "s_concave = [-0.20, -0.10, 0, 0.10, 0.20]\n",
    "\n",
    "for s_concave_i in s_concave:\n",
    "    mean_theta_diffs = []\n",
    "    ci_theta_diffs = []\n",
    "    mean_p_diffs = []\n",
    "    ci_p_diffs = []\n",
    "    mean_regrets = []\n",
    "    ci_regrets = []\n",
    "    mean_l2_errors = []\n",
    "    ci_l2_errors = []\n",
    "\n",
    "    print(f\"\\nRunning experiments for N = {N} s = {s_concave_i}\")\n",
    "    theta_true = []\n",
    "    for i in range(N):\n",
    "        theta = np.ones(N)\n",
    "        theta[i] = -beta\n",
    "        # For j != i, set them to sqrt((1-beta^2)/(N-1))\n",
    "        for j in range(N):\n",
    "            if j != i:\n",
    "                theta[j] = np.sqrt((1-beta**2)/(N-1))\n",
    "        theta_true.append(theta)\n",
    "    \n",
    "    \n",
    "    def check_sum_abs(vector, i):\n",
    "        gamma = np.delete(vector, i)\n",
    "        sum_gamma = np.sum(np.abs(gamma))\n",
    "        beta = - vector[i]\n",
    "        return sum_gamma < beta\n",
    "    \n",
    "    #for i in range(N):\n",
    "    #    print(check_sum_abs(theta_true[i], i))\n",
    "    \n",
    "    # Compute p* (the optimal p vector) via a fixed-point iteration\n",
    "    p_initial = [p_M_arr[i] / 2 for i in range(N)]\n",
    "    error = 100\n",
    "    while error > 0.0001:\n",
    "        p_t = []\n",
    "        for i in range(N):\n",
    "            b_i = theta_true[i][i]\n",
    "            gamma_i = np.delete(theta_true[i], i)\n",
    "            p_minus_i = np.delete(p_initial, i)\n",
    "            q_i = p_minus_i @ gamma_i\n",
    "            p_t.append(p_opt(psi[i], b_i, q_i, p_m_arr[i], p_M_arr[i]))\n",
    "        error = np.linalg.norm(np.array(p_initial) - np.array(p_t), 2)\n",
    "        #print(\"Fixed-point error:\", error)\n",
    "        p_initial = p_t\n",
    "    \n",
    "    p_star = p_initial  # optimal initial p vector\n",
    "    print(\"p_star = \",p_star)\n",
    "    #p_star = [-1/theta_true[i][i] for n in range(N)]\n",
    "    \n",
    "    \n",
    "    for T in T_values:\n",
    "        print(f\"\\nRunning experiments for T = {T}, for {num_runs} independent episodes\")\n",
    "        seeds = rng_master.integers(1_000_000, size=num_runs)\n",
    "        if run_parallel == True:\n",
    "            results = Parallel(n_jobs=-1)(\n",
    "                delayed(single_run_experiment)(T, N, seed, p_m_arr, p_M_arr, psi, theta_true, p_star, p_initial, s_concave_i)\n",
    "                for seed in tqdm(seeds, desc=f\"Repetitions for T={T}\", leave=False)\n",
    "            )\n",
    "        else:\n",
    "            results = []\n",
    "            for seed in seeds:\n",
    "                results.append(single_run_experiment(T, seed))\n",
    "        \n",
    "        run_theta_diffs = np.array([r[0] for r in results])\n",
    "        run_l2_errors   = np.array([r[1] for r in results])\n",
    "        run_p_diffs     = np.array([r[2] for r in results])\n",
    "        run_total_regret= np.array([r[3] for r in results])\n",
    "        \n",
    "        def mean_ci(a):\n",
    "            m = np.nanmean(a)\n",
    "            s = np.nanstd(a, ddof=1)\n",
    "            c = 1.96 * s / np.sqrt(num_runs)\n",
    "            return m, c\n",
    "    \n",
    "        mean_theta, ci_theta = mean_ci(run_theta_diffs)\n",
    "        mean_p,     ci_p     = mean_ci(run_p_diffs)\n",
    "        mean_r,     ci_r     = mean_ci(run_total_regret)\n",
    "        mean_l2,    ci_l2    = mean_ci(run_l2_errors)\n",
    "        \n",
    "        mean_theta_diffs.append(mean_theta)\n",
    "        ci_theta_diffs.append(ci_theta)\n",
    "        mean_p_diffs.append(mean_p)\n",
    "        ci_p_diffs.append(ci_p)\n",
    "        mean_regrets.append(mean_r)\n",
    "        ci_regrets.append(ci_r)\n",
    "        mean_l2_errors.append(mean_l2)\n",
    "        ci_l2_errors.append(ci_l2)\n",
    "        \n",
    "        print(f\"\\n[T={T}] Results over {num_runs} runs:\")\n",
    "        print(f\"   Sum(||theta_hat - theta_true||): {mean_theta:.3f} ± {ci_theta:.3f}\")\n",
    "        print(f\"   L2 Error (Isotonic via LS): {mean_l2:.5f} ± {ci_l2:.5f}\")\n",
    "        print(f\"   ||p_T - p^*||: {mean_p:.3f} ± {ci_p:.3f}\")\n",
    "        print(f\"   Total Regret: {mean_r:.3f} ± {ci_r:.3f}\")\n",
    "\n",
    "    #SAVE DATA\n",
    "    data_folder = \"DATA\"\n",
    "    os.makedirs(data_folder, exist_ok=True)\n",
    "    \n",
    "    # Save data as a CSV file\n",
    "    if N == 2:\n",
    "        data_file = os.path.join(data_folder, f\"data_N_{N}_{s_concave_i}.csv\")\n",
    "    \n",
    "    if N == 4:\n",
    "        data_file = os.path.join(data_folder, f\"data_N_{N}_{s_concave_i}.csv\")\n",
    "    \n",
    "    if N == 6:\n",
    "        data_file = os.path.join(data_folder, f\"data_N_{N}_{s_concave_i}.csv\")\n",
    "    \n",
    "    \n",
    "    # Create a DataFrame with all the relevant data\n",
    "    data_df = pd.DataFrame({\n",
    "        \"T_values\": T_values,\n",
    "        \"mean_theta_diffs\": mean_theta_diffs,\n",
    "        \"ci_theta_diffs\": ci_theta_diffs,\n",
    "        \"mean_l2_errors\": mean_l2_errors,\n",
    "        \"ci_l2_errors\": ci_l2_errors,\n",
    "        \"mean_p_diffs\": mean_p_diffs,\n",
    "        \"ci_p_diffs\": ci_p_diffs,\n",
    "        \"mean_regrets\": mean_regrets,\n",
    "        \"ci_regrets\": ci_regrets\n",
    "    })\n",
    "    \n",
    "    # Save the data\n",
    "    data_df.to_csv(data_file, index=False)\n",
    "    print(f\"Data saved successfully to {data_file}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "712d0e9f-5f2a-462e-9119-d00b1f6e8321",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib as mpl\n",
    "\n",
    "def save_plot(N, s_values=None, data_folder=\"DATA\", plot_folder=\"PLOTS\"):\n",
    "    \"\"\"\n",
    "    Plot metrics for a fixed N over multiple s-values.\n",
    "    \n",
    "    Parameters\n",
    "    ----------\n",
    "    N : int\n",
    "        Number of sellers.\n",
    "    s_values : list of float, optional\n",
    "        Values of s used in the simulations.\n",
    "        If None, defaults to [-0.5, -0.25, 0, 0.25, 0.5].\n",
    "    \"\"\"\n",
    "    if s_values is None:\n",
    "        s_values = [-0.5, -0.25, 0, 0.25, 0.5]\n",
    "\n",
    "    os.makedirs(plot_folder, exist_ok=True)\n",
    "\n",
    "    # --- Matplotlib style ---\n",
    "    plt.style.use('seaborn-v0_8-whitegrid')\n",
    "    mpl.rcParams['text.usetex'] = True\n",
    "    mpl.rcParams['font.family'] = 'serif'\n",
    "    mpl.rcParams['font.size'] = 16\n",
    "    mpl.rcParams['axes.labelsize'] = 18\n",
    "    mpl.rcParams['axes.titlesize'] = 20\n",
    "    mpl.rcParams['legend.fontsize'] = 13\n",
    "    mpl.rcParams['xtick.labelsize'] = 16\n",
    "    mpl.rcParams['ytick.labelsize'] = 14\n",
    "\n",
    "    # Color map (works for arbitrary length of s_values)\n",
    "    num_s = len(s_values)\n",
    "    cmap = plt.get_cmap(\"tab10\" if num_s <= 10 else \"tab20\")\n",
    "    colors = {\n",
    "        s: cmap(i / max(num_s - 1, 1))\n",
    "        for i, s in enumerate(s_values)\n",
    "    }\n",
    "\n",
    "    # --- Create 4 subplots ---\n",
    "    fig_main, axes = plt.subplots(1, 4, figsize=(20, 5))\n",
    "    ax1, ax2, ax3, ax4 = axes.ravel()\n",
    "\n",
    "    for s in s_values:\n",
    "        # use EXACT same pattern as in your data generation\n",
    "        filename = os.path.join(data_folder, f\"data_N_{N}_{s}.csv\")\n",
    "        if not os.path.exists(filename):\n",
    "            print(f\"[WARN] File not found, skipping: {filename}\")\n",
    "            continue\n",
    "\n",
    "        df = pd.read_csv(filename)\n",
    "\n",
    "        T_values         = df[\"T_values\"].to_numpy()\n",
    "        mean_theta_diffs = df[\"mean_theta_diffs\"].to_numpy()\n",
    "        ci_theta_diffs   = df[\"ci_theta_diffs\"].to_numpy()\n",
    "        mean_l2_errors   = df[\"mean_l2_errors\"].to_numpy()\n",
    "        ci_l2_errors     = df[\"ci_l2_errors\"].to_numpy()\n",
    "        mean_p_diffs     = df[\"mean_p_diffs\"].to_numpy()\n",
    "        ci_p_diffs       = df[\"ci_p_diffs\"].to_numpy()\n",
    "        mean_regrets     = df[\"mean_regrets\"].to_numpy()\n",
    "        ci_regrets       = df[\"ci_regrets\"].to_numpy()\n",
    "\n",
    "        c = colors[s]\n",
    "        # label nicely, without forcing extra decimals\n",
    "        label = rf\"$s = {s:g}$\"\n",
    "\n",
    "        # (A) Theta error\n",
    "        ax1.plot(T_values, mean_theta_diffs, '-o', linewidth=2, alpha=0.8,\n",
    "                 color=c, label=label)\n",
    "        ax1.fill_between(T_values,\n",
    "                         mean_theta_diffs - ci_theta_diffs,\n",
    "                         mean_theta_diffs + ci_theta_diffs,\n",
    "                         color=c, alpha=0.2)\n",
    "\n",
    "        # (B) Psi error\n",
    "        ax2.plot(T_values, mean_l2_errors, '-o', linewidth=2, alpha=0.8,\n",
    "                 color=c, label=label)\n",
    "        ax2.fill_between(T_values,\n",
    "                         mean_l2_errors - ci_l2_errors,\n",
    "                         mean_l2_errors + ci_l2_errors,\n",
    "                         color=c, alpha=0.2)\n",
    "\n",
    "        # (C) p difference\n",
    "        ax3.plot(T_values, mean_p_diffs, '-o', linewidth=2, alpha=0.8,\n",
    "                 color=c, label=label)\n",
    "        ax3.fill_between(T_values,\n",
    "                         mean_p_diffs - ci_p_diffs,\n",
    "                         mean_p_diffs + ci_p_diffs,\n",
    "                         color=c, alpha=0.2)\n",
    "\n",
    "        # (D) Regret\n",
    "        ax4.plot(T_values, mean_regrets, '-o', linewidth=2, alpha=0.8,\n",
    "                 color=c, label=label)\n",
    "        ax4.fill_between(T_values,\n",
    "                         mean_regrets - ci_regrets,\n",
    "                         mean_regrets + ci_regrets,\n",
    "                         color=c, alpha=0.2)\n",
    "\n",
    "    # --- Titles and labels ---\n",
    "    ax1.set_title(r'$\\sum_{i=1}^{' + str(N) + r'} \\|\\widehat{\\theta}_i^{(T)} - \\theta_i\\|_2$',\n",
    "                  fontsize=20, fontweight='bold')\n",
    "    ax1.set_xlabel(\"T\")\n",
    "    ax1.set_ylabel(f\"N={N}\")\n",
    "    ax1.grid(True, linestyle=\"--\", alpha=0.5)\n",
    "\n",
    "    ax2.set_title(r'$\\sum_{i=1}^{' + str(N) + r'} \\|\\widehat{\\psi}_i^{(T)} - \\psi_i\\|_2$',\n",
    "\n",
    "                  fontsize=20, fontweight='bold')\n",
    "    ax2.set_xlabel(\"T\")\n",
    "    ax2.grid(True, linestyle=\"--\", alpha=0.5)\n",
    "\n",
    "    ax3.set_title(r'$\\|\\mathbf{p}^{(T)} - \\mathbf{p}^\\star\\|_2$',\n",
    "                  fontsize=20, fontweight='bold')\n",
    "    ax3.set_xlabel(\"T\")\n",
    "    ax3.grid(True, linestyle=\"--\", alpha=0.5)\n",
    "\n",
    "    ax4.set_title(\"Total Expected Regret\", fontsize=20, fontweight='bold')\n",
    "    ax4.set_xlabel(\"T\")\n",
    "    ax4.grid(True, linestyle=\"--\", alpha=0.5)\n",
    "\n",
    "    # --- Legends ---\n",
    "    for ax in [ax1, ax2, ax3, ax4]:\n",
    "        ax.legend()\n",
    "\n",
    "    # ---------------- Save + Show ----------------\n",
    "    plt.subplots_adjust(right=0.90)    # MORE SPACE ON THE RIGHT FOR LEGENDS\n",
    "\n",
    "    outfile = os.path.join(plot_folder, f\"stacked_overdraw_N_{N}.pdf\")\n",
    "    plt.savefig(outfile, dpi=300, bbox_inches=\"tight\")  # <-- FIX\n",
    "    plt.show()\n",
    "\n",
    "    print(f\"Saved to {outfile}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9b48d1c1-8f1d-4479-bfb1-0844fe2dd7c3",
   "metadata": {},
   "outputs": [],
   "source": [
    "save_plot(N,s_concave)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3aaa8f1e-1409-49de-9651-84292dc6e6cc",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib as mpl\n",
    "\n",
    "def save_plot_log_log(N, s_values=None, data_folder=\"DATA\", plot_folder=\"PLOTS\"):\n",
    "    \"\"\"\n",
    "    1×4 window.\n",
    "    Two legends on the RIGHT side:\n",
    "        Legend 1: NE convergence slopes (p-error)\n",
    "        Legend 2: Regret slopes\n",
    "    \"\"\"\n",
    "\n",
    "    if s_values is None:\n",
    "        s_values = [-0.5, -0.25, 0, 0.25, 0.5]\n",
    "\n",
    "    os.makedirs(plot_folder, exist_ok=True)\n",
    "\n",
    "    # --- Matplotlib style ---\n",
    "    plt.style.use('seaborn-v0_8-whitegrid')\n",
    "    mpl.rcParams['text.usetex'] = True\n",
    "    mpl.rcParams['font.family'] = 'serif'\n",
    "    mpl.rcParams['font.size'] = 20\n",
    "    mpl.rcParams['axes.labelsize'] = 22\n",
    "    mpl.rcParams['axes.titlesize'] = 22\n",
    "    mpl.rcParams['legend.fontsize'] = 17\n",
    "\n",
    "    # --- Log–log fit helper ---\n",
    "    def loglog_fit(x, y):\n",
    "        mask = (x > 0) & (y > 0) & np.isfinite(x) & np.isfinite(y)\n",
    "        if mask.sum() < 2:\n",
    "            return np.nan, np.nan\n",
    "        logx, logy = np.log(x[mask]), np.log(y[mask])\n",
    "        m, b = np.polyfit(logx, logy, 1)\n",
    "        return m, b\n",
    "\n",
    "    EPS = 1e-300\n",
    "\n",
    "    num_s = len(s_values)\n",
    "    cmap = plt.get_cmap(\"tab10\" if num_s <= 10 else \"tab20\")\n",
    "    colors = {s: cmap(i / max(num_s - 1, 1)) for i, s in enumerate(s_values)}\n",
    "\n",
    "    # --- Layout: 1 x 4 ---\n",
    "    fig, axes = plt.subplots(1, 4, figsize=(26, 6))\n",
    "    ax1, ax2, ax3, ax4 = axes\n",
    "\n",
    "    legend_NE = []     # handles for NE slopes\n",
    "    legend_REG = []    # handles for Regret slopes\n",
    "\n",
    "    # --- Loop over s-values ---\n",
    "    for s in s_values:\n",
    "        fname = os.path.join(data_folder, f\"data_N_{N}_{s}.csv\")\n",
    "        if not os.path.exists(fname):\n",
    "            print(f\"[WARN] missing {fname}\")\n",
    "            continue\n",
    "\n",
    "        df = pd.read_csv(fname)\n",
    "\n",
    "        T = df[\"T_values\"].to_numpy()\n",
    "\n",
    "        th     = df[\"mean_theta_diffs\"].to_numpy()\n",
    "        th_ci  = df[\"ci_theta_diffs\"].to_numpy()\n",
    "\n",
    "        psi    = df[\"mean_l2_errors\"].to_numpy()\n",
    "        psi_ci = df[\"ci_l2_errors\"].to_numpy()\n",
    "\n",
    "        p      = df[\"mean_p_diffs\"].to_numpy()\n",
    "        p_ci   = df[\"ci_p_diffs\"].to_numpy()\n",
    "\n",
    "        reg    = df[\"mean_regrets\"].to_numpy()\n",
    "        reg_ci = df[\"ci_regrets\"].to_numpy()\n",
    "\n",
    "        c = colors[s]\n",
    "\n",
    "        # === Panel 1: Theta error ===\n",
    "        ax1.plot(T, th, \"-o\", lw=2, color=c)\n",
    "        ax1.fill_between(T, th - th_ci, th + th_ci, color=c, alpha=0.2)\n",
    "\n",
    "        # === Panel 2: Psi error ===\n",
    "        ax2.plot(T, psi, \"-o\", lw=2, color=c)\n",
    "        ax2.fill_between(T, psi - psi_ci, psi + psi_ci, color=c, alpha=0.2)\n",
    "\n",
    "        # === Panel 3: p-error (log–log) ===\n",
    "        safe_p = np.maximum(p, EPS)\n",
    "        low_p  = np.maximum(p - p_ci, EPS)\n",
    "        up_p   = np.maximum(p + p_ci, EPS)\n",
    "\n",
    "        m_p, b_p = loglog_fit(T, safe_p)\n",
    "        label_NE = rf\"$s={s:g},\\ m={m_p:.2f}$\" if np.isfinite(m_p) else rf\"$s={s:g}$\"\n",
    "\n",
    "        logT = np.log(T)\n",
    "        h_NE = ax3.plot(logT, np.log(safe_p), \"-o\", lw=2, color=c, label=label_NE)[0]\n",
    "        ax3.fill_between(logT, np.log(low_p), np.log(up_p), color=c, alpha=0.2)\n",
    "\n",
    "        if np.isfinite(m_p):\n",
    "            xfit = np.linspace(logT.min(), logT.max(), 200)\n",
    "            ax3.plot(xfit, b_p + m_p * xfit, \"--\", lw=2, color=c)\n",
    "\n",
    "        legend_NE.append(h_NE)\n",
    "\n",
    "        # === Panel 4: Regret (log–log) ===\n",
    "        safe_r = np.maximum(reg, EPS)\n",
    "        low_r  = np.maximum(reg - reg_ci, EPS)\n",
    "        up_r   = np.maximum(reg + reg_ci, EPS)\n",
    "\n",
    "        m_r, b_r = loglog_fit(T, safe_r)\n",
    "        label_REG = rf\"$s={s:g},\\ m={m_r:.2f}$\" if np.isfinite(m_r) else rf\"$s={s:g}$\"\n",
    "\n",
    "        h_REG = ax4.plot(logT, np.log(safe_r), \"-o\", lw=2, color=c, label=label_REG)[0]\n",
    "        ax4.fill_between(logT, np.log(low_r), np.log(up_r), color=c, alpha=0.2)\n",
    "\n",
    "        if np.isfinite(m_r):\n",
    "            xfit = np.linspace(logT.min(), logT.max(), 200)\n",
    "            ax4.plot(xfit, b_r + m_r * xfit, \"--\", lw=2, color=c)\n",
    "\n",
    "        legend_REG.append(h_REG)\n",
    "\n",
    "    # ---------------- Titles / Labels ----------------\n",
    "    ax1.set_title(r\"$\\sum_{i=1}^N \\|\\widehat{\\theta}_i^{(T)} - \\theta_i\\|_2$\")\n",
    "\n",
    "    ax2.set_title(r\"$\\sum_{i=1}^N \\|\\widehat{\\psi}_i^{(T)} - \\psi_i\\|_2$\")\n",
    "    ax3.set_title(\"NE Convergence (log–log)\")\n",
    "    ax4.set_title(\"Regret (log–log)\")\n",
    "\n",
    "    ax1.set_xlabel(\"T\")\n",
    "    ax1.set_ylabel(f\"N={N}\")\n",
    "    ax2.set_xlabel(\"T\")\n",
    "    ax3.set_xlabel(\"log T\")\n",
    "    ax4.set_xlabel(\"log T\")\n",
    "\n",
    "    for ax in axes:\n",
    "        ax.grid(True, linestyle=\"--\", alpha=0.5)\n",
    "\n",
    "    # ---------------- TWO LEGENDS ON THE RIGHT ----------------\n",
    "\n",
    "    # Legend 1 (NE convergence slopes)\n",
    "    fig.legend(handles=legend_NE,\n",
    "               loc=\"center right\",\n",
    "               bbox_to_anchor=(1.06, 0.70),\n",
    "               title=\"NE Convergence Slopes\",\n",
    "               title_fontsize=18,\n",
    "               fontsize=16)\n",
    "\n",
    "    # Legend 2 (Regret slopes)\n",
    "    fig.legend(handles=legend_REG,\n",
    "               loc=\"center right\",\n",
    "               bbox_to_anchor=(1.06, 0.30),\n",
    "               title=\"Regret Slopes\",\n",
    "               title_fontsize=18,\n",
    "               fontsize=16)\n",
    "\n",
    "    # ---------------- Save + Show ----------------\n",
    "    plt.subplots_adjust(right=0.90)    # MORE SPACE ON THE RIGHT FOR LEGENDS\n",
    "\n",
    "    outfile = os.path.join(plot_folder, f\"stacked_overdraw_N_log_log{N}.pdf\")\n",
    "    plt.savefig(outfile, dpi=300, bbox_inches=\"tight\")  # <-- FIX\n",
    "    plt.show()\n",
    "\n",
    "    print(f\"Saved to {outfile}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "641276bd-5101-41ad-ae67-cb0940cdb006",
   "metadata": {},
   "outputs": [],
   "source": [
    "save_plot_log_log(N, s_concave)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "381a1bf0-c980-43c4-9ea9-7045b1fb2656",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4c0bc3d2-8f81-4925-a09b-973b0eff8867",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6185dffd-412b-45e8-bf1b-ab8e9c4fc240",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "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.11.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
