{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "bb033029-6c6b-43ab-a49e-cc9ec5ef747f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "from pathlib import Path\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import mannwhitneyu\n",
    "import matplotlib.gridspec as gridspec\n",
    "\n",
    "from constants import base_model_name_mapping, BASE_PATH_RESULTS, experiment_with_probe_type_order_list, experiment_order_list\n",
    "from helper import style_multimodel_heatmap, init_plotting_params, save_or_show"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "100ab5a4-5ef7-4baa-aa8b-b6e5319a17f2",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{\n",
      "  \"agg.path.chunksize\": 0,\n",
      "  \"axes.labelsize\": 13.0,\n",
      "  \"axes.titlesize\": 14.0,\n",
      "  \"axes3d.trackballsize\": 0.667,\n",
      "  \"boxplot.flierprops.markersize\": 6.0,\n",
      "  \"boxplot.meanprops.markersize\": 6.0,\n",
      "  \"errorbar.capsize\": 0.0,\n",
      "  \"figure.figsize\": [\n",
      "    6.4,\n",
      "    4.8\n",
      "  ],\n",
      "  \"figure.labelsize\": \"large\",\n",
      "  \"figure.titlesize\": \"large\",\n",
      "  \"font.cursive\": [\n",
      "    \"Apple Chancery\",\n",
      "    \"Textile\",\n",
      "    \"Zapf Chancery\",\n",
      "    \"Sand\",\n",
      "    \"Script MT\",\n",
      "    \"Felipa\",\n",
      "    \"Comic Neue\",\n",
      "    \"Comic Sans MS\",\n",
      "    \"cursive\"\n",
      "  ],\n",
      "  \"font.family\": [\n",
      "    \"sans-serif\"\n",
      "  ],\n",
      "  \"font.fantasy\": [\n",
      "    \"Chicago\",\n",
      "    \"Charcoal\",\n",
      "    \"Impact\",\n",
      "    \"Western\",\n",
      "    \"xkcd script\",\n",
      "    \"fantasy\"\n",
      "  ],\n",
      "  \"font.monospace\": [\n",
      "    \"DejaVu Sans Mono\",\n",
      "    \"Bitstream Vera Sans Mono\",\n",
      "    \"Computer Modern Typewriter\",\n",
      "    \"Andale Mono\",\n",
      "    \"Nimbus Mono L\",\n",
      "    \"Courier New\",\n",
      "    \"Courier\",\n",
      "    \"Fixed\",\n",
      "    \"Terminal\",\n",
      "    \"monospace\"\n",
      "  ],\n",
      "  \"font.sans-serif\": [\n",
      "    \"DejaVu Sans\",\n",
      "    \"Bitstream Vera Sans\",\n",
      "    \"Computer Modern Sans Serif\",\n",
      "    \"Lucida Grande\",\n",
      "    \"Verdana\",\n",
      "    \"Geneva\",\n",
      "    \"Lucid\",\n",
      "    \"Arial\",\n",
      "    \"Helvetica\",\n",
      "    \"Avant Garde\",\n",
      "    \"sans-serif\"\n",
      "  ],\n",
      "  \"font.serif\": [\n",
      "    \"DejaVu Serif\",\n",
      "    \"Bitstream Vera Serif\",\n",
      "    \"Computer Modern Roman\",\n",
      "    \"New Century Schoolbook\",\n",
      "    \"Century Schoolbook L\",\n",
      "    \"Utopia\",\n",
      "    \"ITC Bookman\",\n",
      "    \"Bookman\",\n",
      "    \"Nimbus Roman No9 L\",\n",
      "    \"Times New Roman\",\n",
      "    \"Times\",\n",
      "    \"Palatino\",\n",
      "    \"Charter\",\n",
      "    \"serif\"\n",
      "  ],\n",
      "  \"font.size\": 10.0,\n",
      "  \"font.stretch\": \"normal\",\n",
      "  \"font.style\": \"normal\",\n",
      "  \"font.variant\": \"normal\",\n",
      "  \"font.weight\": \"normal\",\n",
      "  \"legend.fontsize\": 12.0,\n",
      "  \"legend.title_fontsize\": 13.0,\n",
      "  \"lines.markersize\": 6.0,\n",
      "  \"mathtext.fontset\": \"dejavusans\",\n",
      "  \"pdf.fonttype\": 42,\n",
      "  \"pdf.use14corefonts\": false,\n",
      "  \"pgf.rcfonts\": true,\n",
      "  \"ps.fonttype\": 42,\n",
      "  \"ps.papersize\": \"letter\",\n",
      "  \"svg.fonttype\": \"path\",\n",
      "  \"xtick.labelsize\": 12.0,\n",
      "  \"xtick.major.size\": 3.5,\n",
      "  \"xtick.minor.size\": 2.0,\n",
      "  \"ytick.labelsize\": 12.0,\n",
      "  \"ytick.major.size\": 3.5,\n",
      "  \"ytick.minor.size\": 2.0\n",
      "}\n"
     ]
    }
   ],
   "source": [
    "init_plotting_params()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "f985fc8b-a3f7-4a98-bbf1-c92639b1bb5d",
   "metadata": {},
   "outputs": [],
   "source": [
    "SAVE = False\n",
    "\n",
    "base_storing_path = BASE_PATH_RESULTS / '../results_iclr_exp/plots' \n",
    "\n",
    "if SAVE:\n",
    "    base_storing_path.mkdir(parents=True, exist_ok=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "ed4100ec-6930-4df5-8229-6fb059a3a7a7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>dataset</th>\n",
       "      <th>base_model</th>\n",
       "      <th>level_2</th>\n",
       "      <th>task</th>\n",
       "      <th>mode</th>\n",
       "      <th>combiner</th>\n",
       "      <th>feature_normalization</th>\n",
       "      <th>feature_alignment</th>\n",
       "      <th>train_split</th>\n",
       "      <th>val_proportion</th>\n",
       "      <th>...</th>\n",
       "      <th>res_folder</th>\n",
       "      <th>abs_perf_gain_train_lp_acc1</th>\n",
       "      <th>abs_perf_gain_train_lp_acc5</th>\n",
       "      <th>abs_perf_gain_train_lp_bal_acc1</th>\n",
       "      <th>abs_perf_gain_train_lp_bal_acc5</th>\n",
       "      <th>abs_perf_gain_test_lp_acc1</th>\n",
       "      <th>abs_perf_gain_test_lp_acc5</th>\n",
       "      <th>abs_perf_gain_test_lp_bal_acc1</th>\n",
       "      <th>abs_perf_gain_test_lp_bal_acc5</th>\n",
       "      <th>abs_perf_gain_best_val_bal_acc1</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>imagenet-subset-50k</td>\n",
       "      <td>OpenCLIP_ViT-B-16_openai</td>\n",
       "      <td>0</td>\n",
       "      <td>attentive_probe</td>\n",
       "      <td>combined_models</td>\n",
       "      <td>stacked_zero_pad</td>\n",
       "      <td>True</td>\n",
       "      <td>[null, null]</td>\n",
       "      <td>train</td>\n",
       "      <td>0.2</td>\n",
       "      <td>...</td>\n",
       "      <td>results_iclr_exp</td>\n",
       "      <td>0.03238</td>\n",
       "      <td>0.01422</td>\n",
       "      <td>0.03238</td>\n",
       "      <td>0.01422</td>\n",
       "      <td>0.0134</td>\n",
       "      <td>0.0093</td>\n",
       "      <td>0.0134</td>\n",
       "      <td>0.0093</td>\n",
       "      <td>0.0142</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>imagenet-subset-50k</td>\n",
       "      <td>OpenCLIP_ViT-B-16_openai</td>\n",
       "      <td>189</td>\n",
       "      <td>attentive_probe</td>\n",
       "      <td>combined_models</td>\n",
       "      <td>stacked_zero_pad</td>\n",
       "      <td>True</td>\n",
       "      <td>[null, null, null, null, null, null, null, nul...</td>\n",
       "      <td>train</td>\n",
       "      <td>0.2</td>\n",
       "      <td>...</td>\n",
       "      <td>results_iclr_exp</td>\n",
       "      <td>0.14866</td>\n",
       "      <td>0.04452</td>\n",
       "      <td>0.14866</td>\n",
       "      <td>0.04452</td>\n",
       "      <td>0.0202</td>\n",
       "      <td>0.01154</td>\n",
       "      <td>0.0202</td>\n",
       "      <td>0.01154</td>\n",
       "      <td>0.0502</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>imagenet-subset-50k</td>\n",
       "      <td>OpenCLIP_ViT-B-16_openai</td>\n",
       "      <td>375</td>\n",
       "      <td>attentive_probe</td>\n",
       "      <td>combined_models</td>\n",
       "      <td>stacked_zero_pad</td>\n",
       "      <td>True</td>\n",
       "      <td>[null, null, null, null]</td>\n",
       "      <td>train</td>\n",
       "      <td>0.2</td>\n",
       "      <td>...</td>\n",
       "      <td>results_iclr_exp</td>\n",
       "      <td>0.05232</td>\n",
       "      <td>0.0212</td>\n",
       "      <td>0.05232</td>\n",
       "      <td>0.0212</td>\n",
       "      <td>0.0142</td>\n",
       "      <td>0.00882</td>\n",
       "      <td>0.0142</td>\n",
       "      <td>0.00882</td>\n",
       "      <td>0.0173</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>imagenet-subset-50k</td>\n",
       "      <td>OpenCLIP_ViT-B-16_openai</td>\n",
       "      <td>454</td>\n",
       "      <td>attentive_probe</td>\n",
       "      <td>combined_models</td>\n",
       "      <td>stacked_zero_pad</td>\n",
       "      <td>True</td>\n",
       "      <td>[null, null, null, null, null, null, null, null]</td>\n",
       "      <td>train</td>\n",
       "      <td>0.2</td>\n",
       "      <td>...</td>\n",
       "      <td>results_iclr_exp</td>\n",
       "      <td>0.10088</td>\n",
       "      <td>0.03352</td>\n",
       "      <td>0.10088</td>\n",
       "      <td>0.03352</td>\n",
       "      <td>0.01698</td>\n",
       "      <td>0.01004</td>\n",
       "      <td>0.01698</td>\n",
       "      <td>0.01004</td>\n",
       "      <td>0.0306</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>imagenet-subset-50k</td>\n",
       "      <td>OpenCLIP_ViT-B-16_openai</td>\n",
       "      <td>533</td>\n",
       "      <td>linear_probe</td>\n",
       "      <td>single_model</td>\n",
       "      <td>NaN</td>\n",
       "      <td>True</td>\n",
       "      <td>null</td>\n",
       "      <td>train</td>\n",
       "      <td>0.2</td>\n",
       "      <td>...</td>\n",
       "      <td>results_iclr_exp</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>5 rows × 82 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "               dataset                base_model  level_2             task  \\\n",
       "0  imagenet-subset-50k  OpenCLIP_ViT-B-16_openai        0  attentive_probe   \n",
       "1  imagenet-subset-50k  OpenCLIP_ViT-B-16_openai      189  attentive_probe   \n",
       "2  imagenet-subset-50k  OpenCLIP_ViT-B-16_openai      375  attentive_probe   \n",
       "3  imagenet-subset-50k  OpenCLIP_ViT-B-16_openai      454  attentive_probe   \n",
       "4  imagenet-subset-50k  OpenCLIP_ViT-B-16_openai      533     linear_probe   \n",
       "\n",
       "              mode          combiner feature_normalization  \\\n",
       "0  combined_models  stacked_zero_pad                  True   \n",
       "1  combined_models  stacked_zero_pad                  True   \n",
       "2  combined_models  stacked_zero_pad                  True   \n",
       "3  combined_models  stacked_zero_pad                  True   \n",
       "4     single_model               NaN                  True   \n",
       "\n",
       "                                   feature_alignment train_split  \\\n",
       "0                                       [null, null]       train   \n",
       "1  [null, null, null, null, null, null, null, nul...       train   \n",
       "2                           [null, null, null, null]       train   \n",
       "3   [null, null, null, null, null, null, null, null]       train   \n",
       "4                                               null       train   \n",
       "\n",
       "  val_proportion  ...        res_folder abs_perf_gain_train_lp_acc1  \\\n",
       "0            0.2  ...  results_iclr_exp                     0.03238   \n",
       "1            0.2  ...  results_iclr_exp                     0.14866   \n",
       "2            0.2  ...  results_iclr_exp                     0.05232   \n",
       "3            0.2  ...  results_iclr_exp                     0.10088   \n",
       "4            0.2  ...  results_iclr_exp                         0.0   \n",
       "\n",
       "  abs_perf_gain_train_lp_acc5 abs_perf_gain_train_lp_bal_acc1  \\\n",
       "0                     0.01422                         0.03238   \n",
       "1                     0.04452                         0.14866   \n",
       "2                      0.0212                         0.05232   \n",
       "3                     0.03352                         0.10088   \n",
       "4                         0.0                             0.0   \n",
       "\n",
       "  abs_perf_gain_train_lp_bal_acc5 abs_perf_gain_test_lp_acc1  \\\n",
       "0                         0.01422                     0.0134   \n",
       "1                         0.04452                     0.0202   \n",
       "2                          0.0212                     0.0142   \n",
       "3                         0.03352                    0.01698   \n",
       "4                             0.0                        0.0   \n",
       "\n",
       "  abs_perf_gain_test_lp_acc5 abs_perf_gain_test_lp_bal_acc1  \\\n",
       "0                     0.0093                         0.0134   \n",
       "1                    0.01154                         0.0202   \n",
       "2                    0.00882                         0.0142   \n",
       "3                    0.01004                        0.01698   \n",
       "4                        0.0                            0.0   \n",
       "\n",
       "  abs_perf_gain_test_lp_bal_acc5 abs_perf_gain_best_val_bal_acc1  \n",
       "0                         0.0093                          0.0142  \n",
       "1                        0.01154                          0.0502  \n",
       "2                        0.00882                          0.0173  \n",
       "3                        0.01004                          0.0306  \n",
       "4                            0.0                             0.0  \n",
       "\n",
       "[5 rows x 82 columns]"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_runs_path = BASE_PATH_RESULTS / '../results_iclr_exp/aggregated/all_runs_v8.pkl'\n",
    "all_runs = pd.read_pickle(all_runs_path)\n",
    "all_runs.head()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "6eb340fc-6708-4bb3-9b3b-38f63054dedc",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_runs = all_runs[~all_runs['dataset'].isin(['imagenet-subset-50k', 'wds/imagenet1k', 'wds/vtab/pcam'])].reset_index(drop=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "c424cfb4-ae4d-48fa-9db4-b73f84acea8c",
   "metadata": {},
   "outputs": [],
   "source": [
    "metrics_cols = [\n",
    "    'abs_perf_gain_train_lp_bal_acc1',\n",
    "    'abs_perf_gain_test_lp_bal_acc1',\n",
    "    'test_lp_bal_acc1'\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "87b46cb3-4f8f-47a1-a3d5-4c38153b97aa",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_runs[metrics_cols] = all_runs[metrics_cols].astype(float)\n",
    "all_runs[metrics_cols] *= 100"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "dcd49a42-969c-48f0-8175-f89e3f3c3bdd",
   "metadata": {},
   "outputs": [],
   "source": [
    "def format_if_is_bold(mean_val, std_val, is_bold):\n",
    "    formatted = f\"{mean_val:.2f} ± {std_val:.2f}\"\n",
    "    if is_bold:\n",
    "        return f\"\\\\textbf{{{formatted}}}\"\n",
    "    else:\n",
    "        return formatted\n",
    "    \n",
    "\n",
    "def format_for_latex(mean_val, std_val, row_means):\n",
    "    formatted = f\"{mean_val:.2f} ± {std_val:.2f}\"\n",
    "    \n",
    "    sorted_means = sorted(row_means, reverse=True)\n",
    "    if mean_val not in sorted_means:\n",
    "        return formatted\n",
    "        \n",
    "    rank = sorted_means.index(mean_val) + 1\n",
    "\n",
    "    if rank == 1:\n",
    "        return f\"\\\\textbf{{{formatted}}}\"\n",
    "    elif rank == 2:\n",
    "        return f\"\\\\underline{{{formatted}}}\"\n",
    "    else:\n",
    "        return formatted"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e66cfc1b-ff64-462b-ad0b-45976e8b6edd",
   "metadata": {},
   "source": [
    "## Does the training objective or model size impact the benefit of intermediate layers?"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8320cdd2-045a-4a01-8ba3-f12701197b66",
   "metadata": {},
   "source": [
    "#### Per model performance gain as a PLOT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8d7dfe4b-b507-4d7b-abf7-42554727eddc",
   "metadata": {},
   "outputs": [],
   "source": [
    "grouping_cols = [\"base_model_fmt\", \"Experiment\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "a5579bb7-6bf1-41cc-9f53-7b241791c5e1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "base_model_fmt  Experiment                               \n",
       "CLIP-B-16       All tokens last layer (attentive)            18\n",
       "                CLS last layer                               18\n",
       "                CLS+AP last layer (attentive)                18\n",
       "                CLS+AP last layer (linear)                   18\n",
       "                CLS+AP layers from all blocks (attentive)    18\n",
       "                                                             ..\n",
       "Sup-ViT-S-16    CLS last layer                               18\n",
       "                CLS+AP last layer (attentive)                18\n",
       "                CLS+AP last layer (linear)                   18\n",
       "                CLS+AP layers from all blocks (attentive)    18\n",
       "                CLS+AP layers from all blocks (linear)       18\n",
       "Name: abs_perf_gain_train_lp_bal_acc1, Length: 68, dtype: int64"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_runs.groupby(grouping_cols)[metrics_cols[0]].count()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "182bc8b2-4668-41f4-8ee7-dea1231ccbf8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array(['CLIP-B-16', 'CLIP-B-32', 'CLIP-L-14', 'DINOv2-B-14',\n",
       "       'DINOv2-L-14', 'DINOv2-S-14', 'Sup-ViT-B-16', 'Sup-ViT-L-16',\n",
       "       'Sup-ViT-S-16'], dtype=object)"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_runs['base_model_fmt'].unique()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "a1c698fa-6bcf-4710-8dfe-3e8f9c007d2f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Experiment\n",
       "CLS+AP last layer (attentive)                         162\n",
       "CLS+AP layers from all blocks (attentive)             162\n",
       "CLS last layer                                        162\n",
       "CLS+AP last layer (linear)                            162\n",
       "CLS+AP layers from all blocks (linear)                162\n",
       "All tokens last layer (attentive)                     162\n",
       "CLS+AP layers from middle & last block (attentive)     72\n",
       "CLS+AP layers from quarterly block (attentive)         72\n",
       "CLS+AP layers from middle & last block (linear)        54\n",
       "CLS+AP layers from quarterly block (linear)            54\n",
       "Name: count, dtype: int64"
      ]
     },
     "execution_count": 12,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "all_runs['Experiment'].value_counts()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "63eca5ea-51b5-46f8-8088-a7bf58340115",
   "metadata": {},
   "outputs": [],
   "source": [
    "curr_order = [\n",
    "    \"CLS last layer\",\n",
    "    'All tokens last layer (attentive)',\n",
    "    'CLS+AP layers from all blocks (linear)',\n",
    "    'CLS+AP layers from all blocks (attentive)',\n",
    "]\n",
    "subset_runs = all_runs[all_runs['Experiment'].isin(curr_order)].copy().reset_index()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "e3bad977-4894-4e9e-b18b-6fb53e55b27d",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_oder = [\n",
    "    'CLIP-B-32', \n",
    "    'CLIP-B-16',\n",
    "    'CLIP-L-14',\n",
    "    'DINOv2-S-14',\n",
    "    'DINOv2-B-14',\n",
    "    'DINOv2-L-14',\n",
    "    'Sup-ViT-S-16',\n",
    "    'Sup-ViT-B-16',\n",
    "    'Sup-ViT-L-16', \n",
    "]\n",
    "\n",
    "exp_name_mapping = {\n",
    "'CLS last layer': \"Last layer\\n(CLS, linear)\",\n",
    "'CLS+AP last layer (linear)': \"Last layer\\n(CLS+AP, linear)\",\n",
    "'CLS+AP layers from middle & last block (linear)': \"Two layers\\n(CLS+AP, linear)\",\n",
    "'CLS+AP layers from quarterly block (linear)': \"Four layers\\n(CLS+AP, linear)\",\n",
    "'CLS+AP layers from all blocks (linear)': \"All layers\\n(CLS+AP, linear)\",\n",
    "'CLS+AP last layer (attentive)': \"Last layer\\n(CLS+AP, attentive)\",\n",
    "'CLS+AP layers from middle & last block (attentive)': \"Two layers\\n(CLS+AP, attentive)\",\n",
    "'CLS+AP layers from quarterly block (attentive)': \"Four layers\\n(CLS+AP, attentive)\",\n",
    "'CLS+AP layers from all blocks (attentive)': \"All layers\\n(CLS+AP, attentive)\",\n",
    "'All tokens last layer (attentive)': \"Last layer\\n(all tokens, attentive)\",\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "e1936ba0-a714-4720-9f17-488c86e79ce8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2413980/2973412681.py:31: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(\n",
      "/tmp/ipykernel_2413980/2973412681.py:78: UserWarning: set_ticklabels() should only be used with a fixed number of ticks, i.e. after set_ticks() or using a FixedLocator.\n",
      "  ax_main.set_xticklabels(custom_labels)\n",
      "/tmp/ipykernel_2413980/2973412681.py:81: UserWarning: No artists with labels found to put in legend.  Note that artists whose label start with an underscore are ignored when legend() is called with no argument.\n",
      "  ax_raw.legend().remove()\n",
      "/tmp/ipykernel_2413980/2973412681.py:91: UserWarning: This figure includes Axes that are not compatible with tight_layout, so results might be incorrect.\n",
      "  plt.tight_layout()\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABOUAAAF4CAYAAAAbn029AAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjMsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvZiW1igAAAAlwSFlzAAAPYQAAD2EBqD+naQAA2axJREFUeJzs3XdYk9fbB/BvQBJQICAg4Ig4iqvuUetAxVn3QK1WBeu2WgUnuFAr1opaK0WcgGK1asVa68S66sZqf87WCaKAIltk5/2Dl9TICskTAvL9XFevmuec5zx3CGTcOefcIrlcLgcRERERERERERGVGD1dB0BERERERERERFTeMClHRERERERERERUwpiUIyIiIiIiIiIiKmFMyhEREREREREREZUwJuWIiIiIiIiIiIhKGJNyREREREREREREJYxJOSIiIiIiIiIiohLGpJyK5HI5EhMTIZfLdR0KEVGZxudTIiIiIiIiJuVUlpSUBKlUiqSkJF2HQkRUpvH5lIiIiIiIiEk5IiIiIiIiIiKiEsekHBERERERERERUQljUo6IiIiIiIiIiKiEMSlHRERERERERERUwpiUIyIiIiIiIiIiKmFMyhEREREREREREZUwJuWIiIiIiIiIiIhKWKlKyiUnJ2PJkiXo1asXKleuDJFIhICAgHz73rt3D7169YKxsTEqV66M0aNH49WrV3n6ZWdn47vvvkOtWrVgaGiIJk2aYPfu3Vq+J0RERERERERERAUrVUm5mJgYLFu2DPfu3UPTpk0L7BcREQEHBwc8fPgQXl5emD17Nn7//Xd0794d6enpSn0XLFiAefPmoXv37tiwYQNkMhlGjhyJPXv2aPvuEBERERERERER5auCrgN4l62tLSIjI2FjY4PQ0FC0bt06335eXl548+YNrl+/DplMBgBo06YNunfvjoCAAEycOBEA8Pz5c6xZswZfffUVfHx8AADjx49Hp06dMGfOHAwdOhT6+volc+eIiIiIiIiIiIj+X6maKSeRSGBjY1Nkv19++QV9+/ZVJOQAoFu3brC3t8fevXsVx3799VdkZGRg6tSpimMikQhTpkxBREQELl26JOwdICIiIiIiIiIiUkGpSsqp4vnz53j58iVatWqVp61Nmza4ceOG4vaNGzdQqVIlNGjQIE+/3HYiIiIiIiIiIqKSVqqWr6oiMjISQM5S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R+LFixeYN28ePD090a9fP2MUnYioQJoxYwZmzJhh7GIQEREREREVGgUyKHf+/HkAwJ49e7Bnzx61/cqgnLOzM44fP47Ro0djwoQJMDc3h5+fHxYsWMD55IiIiIiIiIiIyGgKZFDu2LFjOqetUaMGDh48mHuFISIiIiIiIiIi0pOJsQtARERERERERERU1DAoR0RERERERERElMcYlCMiIiIiIiIiIspjBXJOOSp6UlNTcefOHemx8v/M25QqVqwICwuLPCsbEREREREREZG+GJSjAuHOnTsYMmSI2vZZs2apbVu+fDmqVKmSF8Uq9BgMJSIiIiIiIsodDMpRgVCxYkUsX75c57RFRdagGWDYwBmDoURERAVXbt8nEBER0fuRCSGEsQtRECQlJcHW1haJiYmwsbExdnGIAADXr1/XGDTTRt/AmaabeW14I0+6YntKRJQ3cvs+gYiIiN4Pg3I64pdIyo/0CZoBDJxR/sD2lIgob/A+gYiIKH/j8FWiAszCwoK/aBMREZFGvE8gIiLK30yMXQAiIiIiIiIiIqKihkE5IiIiIiIiIiKiPMagHBERERERERERUR5jUI6IiIiIiIiIiCiPMShHRERERERERESUxxiUIyIiIiIiIiIiymMMyhEREREREREREeUxBuWIiIiIiIiIiIjyGINyREREREREREREeYxBOSIiIiIiIiIiojxWzNgFKCiEEACApKQkI5eEiHKLtbU1ZDKZsYtR6LE9JSIiIqLcxnt7KggYlNNRcnIyAMDZ2dnIJSGi3JKYmAgbGxtjF6PQY3tKRERERLmN9/ZUEMiEsssCZUuhUODBgweMthcRSUlJcHZ2xt27d9mQFyGs33mD7WnBxbaRKO+x3hEZB+tewcd7TSoI2FNORyYmJqhQoYKxi0F5zMbGhhdhIgNje1rwsW0kynusd0TGwbpHRLmJCz0QERERERERERHlMQbliIiIiIiIiIiI8hiDckQayOVyBAUFQS6XG7soRET5BttGorzHekdkHKx7RJQXuNADERERERERERFRHmNPOSIiIiIiIiIiojzGoBwREREREREREVEeY1COiIiIiIiIiIgojzEoR0RERERERERElMcYlCPSIjg4GDKZzNjFICLKV9g2EqmSyWQIDg6WHq9duxYymQyxsbHZPo91iej9sO4RUWHAoBwZjfLCefbs2TzJ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",
      "text/plain": [
       "<Figure size 1300x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Create figure with custom grid layout\n",
    "fig = plt.figure(figsize=(3 + len(curr_order) * 2.5, 4))\n",
    "gs = gridspec.GridSpec(\n",
    "    1, 2, \n",
    "    figure=fig,\n",
    "    width_ratios=[1.3, len(curr_order)-0.5], \n",
    "    wspace=0.2  \n",
    ")\n",
    "\n",
    "reversed_palette = []\n",
    "palette_list = list(plt.cm.tab20c.colors)\n",
    "for group_start in [8, 12]:\n",
    "    group = palette_list[group_start:group_start+3]\n",
    "    reversed_group = group[::-1]\n",
    "    reversed_palette.extend(reversed_group)\n",
    "\n",
    "palette_list = list(plt.cm.tab20b.colors)\n",
    "for group_start in [12]:\n",
    "    group = palette_list[group_start:group_start+3]\n",
    "    reversed_group = group[::-1]\n",
    "    reversed_palette.extend(reversed_group)\n",
    "\n",
    "o = 1.15\n",
    "reversed_palette = [tuple([o*c for c in rgb]) for rgb in reversed_palette]\n",
    "\n",
    "ax_raw = fig.add_subplot(gs[0, 0])\n",
    "\n",
    "leftmost_exp = curr_order[0]\n",
    "leftmost_data = subset_runs[subset_runs['Experiment'] == leftmost_exp]\n",
    "\n",
    "sns.boxplot(\n",
    "    data=leftmost_data,\n",
    "    x='base_model_fmt',\n",
    "    y='test_lp_bal_acc1',\n",
    "    order=model_oder,\n",
    "    fliersize=2,\n",
    "    palette=reversed_palette,\n",
    "    ax=ax_raw\n",
    ")\n",
    "\n",
    "ax_raw.set_title(\"\")\n",
    "ax_raw.set_xlabel(\"\")\n",
    "ax_raw.set_ylabel(\"Balanced accuracy [%]\")\n",
    "ax_raw.spines['top'].set_visible(False)\n",
    "ax_raw.spines['right'].set_visible(False) \n",
    "\n",
    "# ax_raw.yaxis.tick_right()\n",
    "# ax_raw.yaxis.set_label_position(\"right\")\n",
    "\n",
    "middle_pos = (len(model_oder) - 1) / 2\n",
    "ax_raw.set_xticks([middle_pos])\n",
    "ax_raw.set_xticklabels([exp_name_mapping[leftmost_exp]])\n",
    "ax_raw.margins(x=0.1)\n",
    "\n",
    "\n",
    "ax_main = fig.add_subplot(gs[0, 1])\n",
    "g = sns.boxplot(\n",
    "    subset_runs,\n",
    "    x='Experiment',\n",
    "    y='abs_perf_gain_test_lp_bal_acc1',\n",
    "    hue=\"base_model_fmt\",\n",
    "    hue_order=model_oder,\n",
    "    fliersize=2,\n",
    "    showfliers=False,\n",
    "    order=curr_order[1:],\n",
    "    palette=reversed_palette,\n",
    "    ax=ax_main\n",
    ")\n",
    "\n",
    "ax_main.set_xlabel(\"\")\n",
    "ax_main.set_ylabel(\"Absolute accuracy gain [pp]\")\n",
    "\n",
    "ax_main.axhline(0, ls=':', color=\"grey\", zorder=-1)\n",
    "ax_main.spines['top'].set_visible(False)\n",
    "ax_main.spines['right'].set_visible(False)\n",
    "\n",
    "custom_labels = [exp_name_mapping[val.get_text()] for val in ax_main.get_xticklabels()]\n",
    "ax_main.set_xticklabels(custom_labels)\n",
    "\n",
    "ax_main.legend().remove()\n",
    "ax_raw.legend().remove()\n",
    "\n",
    "\n",
    "handles, labels = ax_main.get_legend_handles_labels()\n",
    "# fig.legend(handles, labels, bbox_to_anchor=(0.6, 1.1), loc='upper center', \n",
    "#           ncols=3, frameon=False, fontsize=11)\n",
    "\n",
    "fig.legend(handles, labels, bbox_to_anchor=(0.9,0.75), loc='upper left', \n",
    "          ncols=1, frameon=False, fontsize=11)\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "fn = base_storing_path / \"per_model_performance_gain_dist\" / f'boxplot_gridspec_dual_v6.pdf'\n",
    "save_or_show(fig, fn, SAVE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ced13adb-1f34-4856-8ac6-b762a5eb2abb",
   "metadata": {},
   "source": [
    "##### OLD VERSION"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "86c71159-8d9c-4a04-97e9-422bbdfc7097",
   "metadata": {},
   "outputs": [],
   "source": [
    "# curr_order = [col for col in experiment_with_probe_type_order_list if (\"middle & last\" not in col) and (\"quarterly\" not in col)]\n",
    "# curr_order = curr_order[1:]\n",
    "# curr_order = [\n",
    "#     # 'CLS+AP last layer (linear)',\n",
    "#     # 'CLS+AP last layer (attentive)',\n",
    "#     'CLS+AP layers from all blocks (linear)',\n",
    "#     'CLS+AP layers from all blocks (attentive)',\n",
    "#     'All tokens last layer (attentive)',\n",
    "#     \"CLS last layer\"\n",
    "# ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "5b70098a-360a-409d-9279-227e850be904",
   "metadata": {},
   "outputs": [],
   "source": [
    "# subset_runs = all_runs[all_runs['Experiment'].isin(curr_order)].copy().reset_index()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "19530fde-f382-4176-ae97-dc59491f2e4e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# model_oder = [\n",
    "#     'CLIP-B-32', \n",
    "#     'CLIP-B-16',\n",
    "#     'CLIP-L-14',\n",
    "#     'DINOv2-S-14',\n",
    "#     'DINOv2-B-14',\n",
    "#     'DINOv2-L-14',\n",
    "#     'ViT-S-16',\n",
    "#     'ViT-B-16',\n",
    "#     'ViT-L-16', \n",
    "# ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "596f80e4-f605-49ca-86a3-9257b025cf2a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# exp_name_mapping = {\n",
    "# 'CLS last layer': \"Last layer\\n(CLS, linear)\",\n",
    "# 'CLS+AP last layer (linear)': \"Last layer\\n(CLS+AP, linear)\",\n",
    "# 'CLS+AP layers from middle & last block (linear)': \"Two layers\\n(CLS+AP, linear)\",\n",
    "# 'CLS+AP layers from quarterly block (linear)': \"Four layers\\n(CLS+AP, linear)\",\n",
    "# 'CLS+AP layers from all blocks (linear)': \"All layers\\n(CLS+AP, linear)\",\n",
    "# 'CLS+AP last layer (attentive)': \"Last layer\\n(CLS+AP, attentive)\",\n",
    "# 'CLS+AP layers from middle & last block (attentive)': \"Two layers\\n(CLS+AP, attentive)\",\n",
    "# 'CLS+AP layers from quarterly block (attentive)': \"Four layers\\n(CLS+AP, attentive)\",\n",
    "# 'CLS+AP layers from all blocks (attentive)': \"All layers\\n(CLS+AP, attentive)\",\n",
    "# 'All tokens last layer (attentive)': \"Last layer\\n(all tokens, attentive)\",\n",
    "# }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "3da18ce8-53ba-4440-98ac-d6c5353415f2",
   "metadata": {},
   "outputs": [],
   "source": [
    "# plt.figure(figsize=(10,5))\n",
    "\n",
    "# tab20c = plt.cm.tab20b.colors  \n",
    "# palette_list = list(tab20c)\n",
    "# reversed_palette = []\n",
    "# for group_start in [0, 4, 16]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]  # reverse the group\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "# opacities = [1.2175, 1.2175, 1.231]\n",
    "# reversed_palette = [tuple([o*c for o, c in zip(opacities, rgb)]) for rgb in reversed_palette]\n",
    "\n",
    "# g = sns.boxplot(\n",
    "#     subset_runs,\n",
    "#     x = 'Experiment',\n",
    "#     y = 'abs_perf_gain_test_lp_bal_acc1',\n",
    "#     hue= \"base_model_fmt\",\n",
    "#     hue_order = model_oder,\n",
    "#     # showfliers=False,\n",
    "#     fliersize=2,\n",
    "#     order=curr_order,\n",
    "#     palette=reversed_palette\n",
    "# )\n",
    "# g.set_xlabel(\"\")\n",
    "# g.set_ylabel(\"Absolute accuracy gain [pp]\")\n",
    "# # sns.move_legend(g, bbox_to_anchor=(1, 1), loc='upper left', ncols=1, frameon=False, title=\"Models\")\n",
    "# sns.move_legend(g, bbox_to_anchor=(0.01, 1.05), loc='upper left', ncols=1, frameon=False, title=\"\", fontsize=11)\n",
    "\n",
    "# custom_labels = [exp_name_mapping[val.get_text()] for val in g.get_xticklabels()]\n",
    "# g.set_xticklabels(custom_labels);\n",
    "\n",
    "# g.axhline(0, ls=':', color=\"grey\", zorder=-1)\n",
    "# g.spines['top'].set_visible(False)\n",
    "# g.spines['right'].set_visible(False)\n",
    "\n",
    "\n",
    "# fn = base_storing_path / \"per_model_performance_gain_dist\" / f'boxplot_all_models_with_fliers_v1.pdf'\n",
    "# save_or_show(plt.gcf(), fn, SAVE)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "5d54ffd0-1896-4b64-853f-9bc1ca821b27",
   "metadata": {},
   "outputs": [],
   "source": [
    "# plt.figure(figsize=(10,5))\n",
    "\n",
    "# reversed_palette = []\n",
    "# palette_list = list(plt.cm.tab20c.colors )\n",
    "# for group_start in [8, 12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "# palette_list = list(plt.cm.tab20b.colors )\n",
    "# for group_start in [16]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "\n",
    "# opacities = [1.2175, 1.2175, 1.231]\n",
    "# o=1.15\n",
    "# reversed_palette = [tuple([o*c for c in rgb]) for rgb in reversed_palette]\n",
    "\n",
    "\n",
    "# g = sns.boxplot(\n",
    "#     subset_runs,\n",
    "#     x = 'Experiment',\n",
    "#     y = 'abs_perf_gain_test_lp_bal_acc1',\n",
    "#     hue= \"base_model_fmt\",\n",
    "#     hue_order = model_oder,\n",
    "#     # showfliers=False,\n",
    "#     fliersize=2,\n",
    "#     order=curr_order,\n",
    "#     palette=reversed_palette\n",
    "# )\n",
    "# g.set_xlabel(\"\")\n",
    "# g.set_ylabel(\"Absolute accuracy gain [pp]\")\n",
    "# # sns.move_legend(g, bbox_to_anchor=(1, 1), loc='upper left', ncols=1, frameon=False, title=\"Models\")\n",
    "# sns.move_legend(g, bbox_to_anchor=(0.01, 1.05), loc='upper left', ncols=1, frameon=False, title=\"\", fontsize=11)\n",
    "\n",
    "# custom_labels = [exp_name_mapping[val.get_text()] for val in g.get_xticklabels()]\n",
    "# g.set_xticklabels(custom_labels);\n",
    "\n",
    "# g.axhline(0, ls=':', color=\"grey\", zorder=-1)\n",
    "# g.spines['top'].set_visible(False)\n",
    "# g.spines['right'].set_visible(False)\n",
    "\n",
    "\n",
    "# fn = base_storing_path / \"per_model_performance_gain_dist\" / f'boxplot_all_models_with_fliers_v2.pdf'\n",
    "# save_or_show(plt.gcf(), fn, SAVE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "7c3e70c8-9e7a-4649-8512-6b5620c9b5bb",
   "metadata": {},
   "outputs": [],
   "source": [
    "# plt.figure(figsize=(len(curr_order) * 2,5))\n",
    "\n",
    "# reversed_palette = []\n",
    "# palette_list = list(plt.cm.tab20c.colors )\n",
    "# for group_start in [8, 12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "# palette_list = list(plt.cm.tab20b.colors )\n",
    "# for group_start in [12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "\n",
    "# opacities = [1.2175, 1.2175, 1.231]\n",
    "# o=1.15\n",
    "# reversed_palette = [tuple([o*c for c in rgb]) for rgb in reversed_palette]\n",
    "\n",
    "\n",
    "# g = sns.boxplot(\n",
    "#     subset_runs,\n",
    "#     x = 'Experiment',\n",
    "#     y = 'abs_perf_gain_test_lp_bal_acc1',\n",
    "#     hue= \"base_model_fmt\",\n",
    "#     hue_order = model_oder,\n",
    "#     # showfliers=False,\n",
    "#     fliersize=2,\n",
    "#     order=curr_order,\n",
    "#     palette=reversed_palette\n",
    "# )\n",
    "# g.set_xlabel(\"\")\n",
    "# g.set_ylabel(\"Absolute accuracy gain [pp]\")\n",
    "# # sns.move_legend(g, bbox_to_anchor=(1, 1), loc='upper left', ncols=1, frameon=False, title=\"Models\")\n",
    "# sns.move_legend(g, bbox_to_anchor=(0.01, 1.05), loc='upper left', ncols=1, frameon=False, title=\"\", fontsize=11)\n",
    "\n",
    "# custom_labels = [exp_name_mapping[val.get_text()] for val in g.get_xticklabels()]\n",
    "# g.set_xticklabels(custom_labels);\n",
    "\n",
    "# g.axhline(0, ls=':', color=\"grey\", zorder=-1)\n",
    "# g.spines['top'].set_visible(False)\n",
    "# g.spines['right'].set_visible(False)\n",
    "\n",
    "# fn = base_storing_path / \"per_model_performance_gain_dist\" / f'boxplot_all_models_with_fliers_v3.pdf'\n",
    "# save_or_show(plt.gcf(), fn, SAVE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "54555c29-7f43-4cb1-8224-313c4ed1fc8e",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# plt.figure(figsize=(len(curr_order) * 2.5,4))\n",
    "\n",
    "# reversed_palette = []\n",
    "# palette_list = list(plt.cm.tab20c.colors )\n",
    "# for group_start in [8, 12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "# palette_list = list(plt.cm.tab20b.colors )\n",
    "# for group_start in [12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "\n",
    "# opacities = [1.2175, 1.2175, 1.231]\n",
    "# o=1.15\n",
    "# reversed_palette = [tuple([o*c for c in rgb]) for rgb in reversed_palette]\n",
    "\n",
    "\n",
    "# g = sns.boxplot(\n",
    "#     subset_runs,\n",
    "#     x = 'Experiment',\n",
    "#     y = 'abs_perf_gain_test_lp_bal_acc1',\n",
    "#     hue= \"base_model_fmt\",\n",
    "#     hue_order = model_oder,\n",
    "#     fliersize=2,\n",
    "#     order=curr_order[:-1],\n",
    "#     palette=reversed_palette\n",
    "# )\n",
    "# g.set_xlabel(\"\")\n",
    "# g.set_ylabel(\"Absolute accuracy gain [pp]\")\n",
    "# sns.move_legend(g, bbox_to_anchor=(0.5, 0.91), loc='lower center', ncols=3, frameon=False, title=\"\", fontsize=11)\n",
    "\n",
    "# custom_labels = [exp_name_mapping[val.get_text()] for val in g.get_xticklabels()]\n",
    "# g.set_xticklabels(custom_labels);\n",
    "\n",
    "# g.axhline(0, ls=':', color=\"grey\", zorder=-1)\n",
    "# g.spines['top'].set_visible(False)\n",
    "# g.spines['right'].set_visible(False)\n",
    "\n",
    "# fn = base_storing_path / \"per_model_performance_gain_dist\" / f'boxplot_all_models_with_fliers_v4.pdf'\n",
    "# save_or_show(plt.gcf(), fn, SAVE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "aeb21354-f5e9-4a0d-8cda-f97cd627abcd",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# import matplotlib.pyplot as plt\n",
    "# import matplotlib.gridspec as gridspec\n",
    "# import seaborn as sns\n",
    "\n",
    "# # Create figure with custom grid layout\n",
    "# fig = plt.figure(figsize=(len(curr_order) * 2.5 + 3, 4))\n",
    "# gs = gridspec.GridSpec(\n",
    "#     1, 2, \n",
    "#     figure=fig,\n",
    "#     width_ratios=[len(curr_order)-0.5, 1], \n",
    "#     wspace=0.05  \n",
    "# )\n",
    "\n",
    "\n",
    "# reversed_palette = []\n",
    "# palette_list = list(plt.cm.tab20c.colors)\n",
    "# for group_start in [8, 12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "# palette_list = list(plt.cm.tab20b.colors)\n",
    "# for group_start in [12]:\n",
    "#     group = palette_list[group_start:group_start+3]\n",
    "#     reversed_group = group[::-1]\n",
    "#     reversed_palette.extend(reversed_group)\n",
    "\n",
    "# o = 1.15\n",
    "# reversed_palette = [tuple([o*c for c in rgb]) for rgb in reversed_palette]\n",
    "\n",
    "# # Left subplot: Main boxplot with accuracy gains\n",
    "# ax_main = fig.add_subplot(gs[0, 0])\n",
    "# g = sns.boxplot(\n",
    "#     subset_runs,\n",
    "#     x='Experiment',\n",
    "#     y='abs_perf_gain_test_lp_bal_acc1',\n",
    "#     hue=\"base_model_fmt\",\n",
    "#     hue_order=model_oder,\n",
    "#     fliersize=2,\n",
    "#     order=curr_order[:-1],\n",
    "#     palette=reversed_palette,\n",
    "#     ax=ax_main\n",
    "# )\n",
    "\n",
    "# # Style the main plot\n",
    "# ax_main.set_xlabel(\"\")\n",
    "# ax_main.set_ylabel(\"Absolute accuracy gain [pp]\")\n",
    "# # ax_main.axhline(10, color=\"grey\", lw=0.5, alpha=0.5, zorder=-1)\n",
    "# # ax_main.axhline(5, color=\"grey\", lw=0.5, alpha=0.5, zorder=-1)\n",
    "# ax_main.axhline(0, ls=':', color=\"grey\", zorder=-1)\n",
    "# ax_main.spines['top'].set_visible(False)\n",
    "# ax_main.spines['right'].set_visible(False)\n",
    "\n",
    "# # Set custom labels\n",
    "# custom_labels = [exp_name_mapping[val.get_text()] for val in ax_main.get_xticklabels()]\n",
    "# ax_main.set_xticklabels(custom_labels)\n",
    "\n",
    " \n",
    "# ax_raw = fig.add_subplot(gs[0, 1])\n",
    "# rightmost_exp = curr_order[-1]\n",
    "# rightmost_data = subset_runs[subset_runs['Experiment'] == rightmost_exp]\n",
    "\n",
    "# g2 = sns.boxplot(\n",
    "#     data=rightmost_data,\n",
    "#     x='base_model_fmt',\n",
    "#     y='test_lp_bal_acc1',\n",
    "#     order=model_oder,\n",
    "#     fliersize=2,\n",
    "#     palette=reversed_palette,\n",
    "#     ax=ax_raw\n",
    "# )\n",
    "\n",
    "# ax_raw.set_title(\"\")\n",
    "# ax_raw.set_xlabel(\"\")\n",
    "# ax_raw.set_ylabel(\"Balanced accuracy [%]\", rotation=270, labelpad=15)\n",
    "# ax_raw.spines['top'].set_visible(False)\n",
    "# ax_raw.spines['left'].set_visible(False)  # Hide left spine since we want y-axis on right\n",
    "\n",
    "# ax_raw.yaxis.tick_right()\n",
    "# ax_raw.yaxis.set_label_position(\"right\")\n",
    "\n",
    "# middle_pos = (len(model_oder) - 1) / 2\n",
    "# ax_raw.set_xticks([middle_pos])\n",
    "# ax_raw.set_xticklabels([exp_name_mapping[rightmost_exp]])\n",
    "\n",
    "# ax_main.legend().remove()\n",
    "# ax_raw.legend().remove()\n",
    "\n",
    "# handles, labels = ax_main.get_legend_handles_labels()\n",
    "# fig.legend(handles, labels, bbox_to_anchor=(0.5, 1.05), loc='upper center', \n",
    "#           ncols=3, frameon=False, fontsize=11)\n",
    "\n",
    "# plt.tight_layout()\n",
    "\n",
    "# # Save the plot\n",
    "# fn = base_storing_path / \"per_model_performance_gain_dist\" / f'boxplot_gridspec_dual_v5.pdf'\n",
    "# save_or_show(fig, fn, SAVE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0edf360e-d1ea-4137-9e43-c0890aabbd29",
   "metadata": {
    "jp-MarkdownHeadingCollapsed": true
   },
   "source": [
    "#### Per model performance gain as a TABLE"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "df5794e9-ed2a-470f-b9db-5ee31f3180af",
   "metadata": {},
   "outputs": [],
   "source": [
    "# grouped_data = all_runs.groupby(grouping_cols)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "94a2df7f-45cd-47ba-a201-a4f27466476a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# for metric_col in metrics_cols:\n",
    "#     aggr_data = grouped_data[metric_col].agg([\"mean\", \"std\"]).reset_index()\n",
    "\n",
    "    \n",
    "#     # Get means for ranking\n",
    "#     mean_pivot = pd.pivot(\n",
    "#         aggr_data,\n",
    "#         index='base_model_fmt',\n",
    "#         columns=\"Experiment\", \n",
    "#         values=\"mean\"\n",
    "#     ).loc[model_oder, curr_order]\n",
    "    \n",
    "#     formatted_data = []\n",
    "#     for idx, row in aggr_data.iterrows():\n",
    "#         model = row['base_model_fmt']\n",
    "#         exp = row['Experiment']\n",
    "#         if model in mean_pivot.index and exp in mean_pivot.columns:\n",
    "#             row_means = mean_pivot.loc[model].values\n",
    "#             formatted = format_for_latex(row['mean'], row['std'], row_means)\n",
    "#             formatted_data.append({\n",
    "#                 'base_model_fmt': model,\n",
    "#                 'Experiment': exp,\n",
    "#                 'mean_std': formatted\n",
    "#             })\n",
    "    \n",
    "#     formatted_df = pd.DataFrame(formatted_data)\n",
    "#     pivoted_aggr_data = pd.pivot(\n",
    "#         formatted_df,\n",
    "#         index='base_model_fmt',\n",
    "#         columns=\"Experiment\",\n",
    "#         values=\"mean_std\"\n",
    "#     )\n",
    "#     pivoted_aggr_data = pivoted_aggr_data.loc[model_oder, curr_order]\n",
    "#     pivoted_aggr_data.index.name = None\n",
    "#     pivoted_aggr_data.columns.name = None\n",
    "#     pivoted_aggr_data = pivoted_aggr_data.reset_index(names='Models')\n",
    "    \n",
    "#     latex_version_pivoted_aggr_data = pivoted_aggr_data.to_latex(escape=False, index=False)\n",
    "\n",
    "#     if SAVE:\n",
    "#         filename = base_storing_path / 'tables_for_per_model_size_n_dataset_perf_gain' / f\"{metric_col}_aggr_over_ds_table.tex\"  # Creates filename based on metric_col\n",
    "#         with open(filename, 'w', encoding='utf-8') as f:\n",
    "#             f.write(latex_version_pivoted_aggr_data)\n",
    "#         print(f\"Stored latex table at {filename=}\")\n",
    "#     else:\n",
    "#         print(metric_col)\n",
    "#         print()\n",
    "#         print(latex_version_pivoted_aggr_data)\n",
    "#         print()\n",
    "#         print()\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9e30c988-3bfd-4a17-a4f0-3e6deb092d51",
   "metadata": {},
   "source": [
    "## When Do Intermediate Layers Improve Task Performance?"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1df2aaf8-2174-46cf-b623-0e2a112aa677",
   "metadata": {
    "jp-MarkdownHeadingCollapsed": true
   },
   "source": [
    "### V1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "080db865-61d4-467c-a0dc-29abc17eaa14",
   "metadata": {},
   "outputs": [],
   "source": [
    "metrics_cols = [\n",
    "    'train_lp_bal_acc1',\n",
    "    'test_lp_bal_acc1',\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "d2cef36e-c780-491d-929e-669e7cd36d83",
   "metadata": {},
   "outputs": [],
   "source": [
    "grouping_cols = [\"dataset\", \"Experiment\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "5751fb0f-a226-40cb-b4d9-4a5fe6d76b7a",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_runs[metrics_cols] = all_runs[metrics_cols] * 100"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "d2a0dd7d-2d1a-4e56-babf-d21d8054f8f3",
   "metadata": {},
   "outputs": [],
   "source": [
    "grouped_data = all_runs.groupby(grouping_cols)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "425440ac-b4bc-47ea-88ea-3b7beebec17d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['CLS last layer',\n",
       " 'CLS+AP last layer (linear)',\n",
       " 'CLS+AP layers from all blocks (linear)',\n",
       " 'CLS+AP last layer (attentive)',\n",
       " 'CLS+AP layers from all blocks (attentive)',\n",
       " 'All tokens last layer (attentive)']"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "curr_order = [col for col in experiment_with_probe_type_order_list if (\"middle & last\" not in col) and (\"quarterly\" not in col)]\n",
    "curr_order"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "4e47b893-dec6-4a08-9e18-cf5f8d66a94f",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "ds_mapping = all_runs[['dataset', 'dataset_fmt', 'dataset_domain']].value_counts().reset_index().set_index('dataset')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "7b9526bb-d7da-4dc5-8df6-abf953fdcde9",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "train_lp_bal_acc1\n",
      "\n",
      "\\begin{tabular}{llllllll}\n",
      "\\toprule\n",
      "Category & Dataset & CLS last layer & CLS+AP last layer (linear) & CLS+AP layers from all blocks (linear) & CLS+AP last layer (attentive) & CLS+AP layers from all blocks (attentive) & All tokens last layer (attentive) \\\\\n",
      "\\midrule\n",
      "Natural (multi-domain) & Country-211 & 30.58 ± 4.94 & 36.16 ± 6.43 & 55.42 ± 12.99 & 39.19 ± 8.69 & \\underline{78.60 ± 7.55} & \\textbf{96.89 ± 6.32} \\\\\n",
      "Natural (multi-domain) & CIFAR-100 & 83.98 ± 9.57 & 86.49 ± 8.30 & 93.70 ± 3.63 & 86.77 ± 10.33 & \\underline{95.88 ± 3.04} & \\textbf{99.95 ± 0.04} \\\\\n",
      "Natural (multi-domain) & CIFAR-10 & 96.00 ± 2.88 & 96.72 ± 2.43 & 98.29 ± 1.27 & 96.71 ± 2.47 & \\underline{98.67 ± 1.19} & \\textbf{99.88 ± 0.20} \\\\\n",
      "Natural (multi-domain) & PASCAL VOC 2007 & 90.26 ± 4.35 & 93.20 ± 3.83 & 97.47 ± 1.19 & 93.81 ± 4.67 & \\underline{99.29 ± 0.63} & \\textbf{99.67 ± 0.70} \\\\\n",
      "Natural (multi-domain) & STL-10 & 99.43 ± 0.45 & 99.54 ± 0.67 & \\underline{99.98 ± 0.04} & 99.64 ± 0.41 & 99.95 ± 0.10 & \\textbf{100.00 ± 0.00} \\\\\n",
      "Natural (multi-domain) & Caltech-101 & 98.82 ± 0.97 & 99.54 ± 0.44 & \\textbf{100.00 ± 0.00} & 99.12 ± 1.04 & \\textbf{100.00 ± 0.00} & \\textbf{100.00 ± 0.00} \\\\\n",
      "Natural (single-domain) & SVHN & 43.97 ± 4.08 & 51.26 ± 4.36 & 73.20 ± 4.81 & 50.54 ± 5.94 & \\underline{76.27 ± 5.33} & \\textbf{91.74 ± 5.22} \\\\\n",
      "Natural (single-domain) & FGVC Aircraft & 74.27 ± 8.57 & 78.02 ± 8.17 & \\underline{97.85 ± 1.74} & 77.69 ± 5.95 & 96.08 ± 2.50 & \\textbf{99.68 ± 0.83} \\\\\n",
      "Natural (single-domain) & Stanford Cars & 86.20 ± 5.44 & 90.23 ± 4.29 & \\underline{99.61 ± 0.20} & 88.32 ± 4.23 & 98.81 ± 0.57 & \\textbf{99.85 ± 0.03} \\\\\n",
      "Natural (single-domain) & GTSRB & 84.22 ± 3.13 & 89.96 ± 3.10 & \\underline{99.42 ± 0.33} & 90.05 ± 3.17 & 99.32 ± 0.40 & \\textbf{99.98 ± 0.07} \\\\\n",
      "Natural (single-domain) & Pets & 96.87 ± 3.55 & 97.75 ± 2.95 & \\underline{99.97 ± 0.04} & 97.49 ± 2.99 & 99.79 ± 0.35 & \\textbf{100.00 ± 0.01} \\\\\n",
      "Natural (single-domain) & Flowers & 99.40 ± 0.98 & 99.85 ± 0.25 & \\textbf{100.00 ± 0.00} & 99.15 ± 1.38 & \\textbf{100.00 ± 0.00} & \\textbf{100.00 ± 0.00} \\\\\n",
      "Specialized & Diabetic Retinopathy & 44.17 ± 2.98 & 46.56 ± 3.03 & 55.22 ± 1.52 & 45.91 ± 3.98 & \\underline{56.19 ± 2.36} & \\textbf{72.95 ± 6.89} \\\\\n",
      "Specialized & RESISC45 & 87.89 ± 2.11 & 91.26 ± 1.95 & \\underline{98.89 ± 0.42} & 91.13 ± 2.21 & 98.86 ± 0.51 & \\textbf{100.00 ± 0.00} \\\\\n",
      "Specialized & EuroSAT & 89.90 ± 5.42 & 93.34 ± 3.94 & 99.22 ± 0.41 & 93.04 ± 4.43 & \\underline{99.51 ± 0.39} & \\textbf{100.00 ± 0.01} \\\\\n",
      "Structured & Dmlab & 42.63 ± 3.19 & 44.87 ± 3.79 & 54.20 ± 2.90 & 44.70 ± 3.27 & \\underline{60.03 ± 3.68} & \\textbf{74.13 ± 4.70} \\\\\n",
      "Structured & FER2013 & 55.89 ± 3.05 & 59.29 ± 3.40 & 67.25 ± 4.07 & 59.63 ± 3.24 & \\underline{72.35 ± 3.77} & \\textbf{79.19 ± 7.34} \\\\\n",
      "Structured & DTD & 87.45 ± 5.61 & 91.29 ± 4.92 & \\underline{99.89 ± 0.14} & 95.67 ± 4.77 & 99.73 ± 0.30 & \\textbf{99.94 ± 0.18} \\\\\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\n",
      "\n",
      "\n",
      "test_lp_bal_acc1\n",
      "\n",
      "\\begin{tabular}{llllllll}\n",
      "\\toprule\n",
      "Category & Dataset & CLS last layer & CLS+AP last layer (linear) & CLS+AP layers from all blocks (linear) & CLS+AP last layer (attentive) & CLS+AP layers from all blocks (attentive) & All tokens last layer (attentive) \\\\\n",
      "\\midrule\n",
      "Natural (multi-domain) & Country-211 & 2148.45 ± 635.06 & 2266.51 ± 642.40 & \\underline{2474.51 ± 676.39} & 2283.41 ± 669.90 & \\textbf{2643.97 ± 736.22} & 2065.24 ± 663.83 \\\\\n",
      "Natural (multi-domain) & CIFAR-100 & 8544.78 ± 570.64 & 8606.11 ± 559.55 & \\underline{8820.56 ± 377.06} & 8631.44 ± 564.77 & \\textbf{8878.00 ± 338.86} & 8718.00 ± 470.05 \\\\\n",
      "Natural (multi-domain) & PASCAL VOC 2007 & 8781.88 ± 230.75 & \\underline{8920.30 ± 239.81} & \\textbf{8927.46 ± 200.49} & 8900.96 ± 246.27 & 8906.23 ± 226.05 & 8759.64 ± 292.14 \\\\\n",
      "Natural (multi-domain) & Caltech-101 & 9557.04 ± 139.94 & \\underline{9599.74 ± 142.55} & 9592.89 ± 161.18 & 9566.00 ± 143.13 & \\textbf{9645.40 ± 115.41} & 9579.90 ± 141.10 \\\\\n",
      "Natural (multi-domain) & CIFAR-10 & 9690.56 ± 193.32 & 9698.89 ± 187.41 & \\underline{9752.00 ± 128.89} & 9709.44 ± 178.30 & \\textbf{9767.56 ± 123.56} & 9732.89 ± 150.14 \\\\\n",
      "Natural (multi-domain) & STL-10 & 9928.89 ± 51.11 & 9928.19 ± 55.71 & 9931.94 ± 47.26 & \\underline{9932.08 ± 48.56} & \\textbf{9932.78 ± 48.76} & 9929.86 ± 53.00 \\\\\n",
      "Natural (single-domain) & FGVC Aircraft & 5568.81 ± 1218.01 & 5472.38 ± 1105.68 & 5406.88 ± 766.45 & 5752.92 ± 1126.05 & \\underline{6211.84 ± 1065.06} & \\textbf{6495.44 ± 1185.26} \\\\\n",
      "Natural (single-domain) & SVHN & 5606.00 ± 591.25 & 6299.69 ± 522.47 & 8045.73 ± 432.12 & 6345.29 ± 660.54 & \\underline{8330.56 ± 399.03} & \\textbf{8636.85 ± 374.52} \\\\\n",
      "Natural (single-domain) & Stanford Cars & 7780.66 ± 1065.19 & 7830.47 ± 1010.33 & 7694.32 ± 786.54 & 7977.83 ± 972.04 & \\underline{8415.81 ± 783.38} & \\textbf{8677.24 ± 690.54} \\\\\n",
      "Natural (single-domain) & GTSRB & 7151.42 ± 745.97 & 7574.17 ± 596.22 & 8027.35 ± 360.69 & 7620.04 ± 618.48 & \\underline{8498.04 ± 378.24} & \\textbf{8953.30 ± 423.72} \\\\\n",
      "Natural (single-domain) & Pets & 9398.29 ± 235.59 & 9392.90 ± 226.72 & 9197.47 ± 276.05 & \\underline{9410.03 ± 191.90} & \\textbf{9426.90 ± 178.34} & 9375.54 ± 191.95 \\\\\n",
      "Natural (single-domain) & Flowers & 9802.66 ± 259.88 & 9843.03 ± 197.79 & 9777.40 ± 218.41 & 9808.61 ± 261.09 & \\textbf{9848.19 ± 169.23} & \\underline{9844.14 ± 191.88} \\\\\n",
      "Specialized & Diabetic Retinopathy & 4580.05 ± 246.07 & 4734.65 ± 220.52 & \\underline{5171.96 ± 129.08} & 4765.63 ± 284.40 & \\textbf{5265.56 ± 149.39} & 4773.61 ± 262.26 \\\\\n",
      "Specialized & RESISC45 & 9045.22 ± 169.50 & 9177.07 ± 179.56 & \\underline{9497.80 ± 90.51} & 9226.76 ± 164.76 & \\textbf{9568.70 ± 83.62} & 9451.83 ± 136.79 \\\\\n",
      "Specialized & EuroSAT & 9388.79 ± 252.12 & 9553.47 ± 148.02 & \\underline{9796.87 ± 30.08} & 9570.97 ± 148.07 & \\textbf{9825.36 ± 23.84} & 9726.54 ± 52.72 \\\\\n",
      "Structured & Dmlab & 4491.43 ± 348.86 & 4672.04 ± 370.57 & 5283.39 ± 298.76 & 4752.39 ± 369.01 & \\underline{5558.94 ± 371.93} & \\textbf{5860.39 ± 448.93} \\\\\n",
      "Structured & FER2013 & 5908.33 ± 460.60 & 6125.99 ± 475.39 & 6533.25 ± 402.14 & 6269.60 ± 477.11 & \\textbf{6912.98 ± 308.98} & \\underline{6682.34 ± 334.89} \\\\\n",
      "Structured & DTD & 7599.29 ± 347.14 & 7716.90 ± 272.95 & \\underline{8003.55 ± 198.03} & 7852.25 ± 292.04 & \\textbf{8004.14 ± 217.74} & 7740.54 ± 494.36 \\\\\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\n",
      "\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for metric_col in metrics_cols:\n",
    "    aggr_data = grouped_data[metric_col].agg([\"mean\", \"std\"]).reset_index()\n",
    "    \n",
    "    # Get means for ranking\n",
    "    mean_pivot = pd.pivot(\n",
    "        aggr_data,\n",
    "        index='dataset',\n",
    "        columns=\"Experiment\", \n",
    "        values=\"mean\"\n",
    "    ).loc[:, curr_order]\n",
    "    \n",
    "    formatted_data = []\n",
    "    for idx, row in aggr_data.iterrows():\n",
    "        model = row['dataset']\n",
    "        exp = row['Experiment']\n",
    "        if model in mean_pivot.index and exp in mean_pivot.columns:\n",
    "            row_means = mean_pivot.loc[model].values\n",
    "            formatted = format_for_latex(row['mean'], row['std'], row_means)\n",
    "            formatted_data.append({\n",
    "                'dataset': model,\n",
    "                'Experiment': exp,\n",
    "                'mean_std': formatted\n",
    "            })\n",
    "    \n",
    "    formatted_df = pd.DataFrame(formatted_data)\n",
    "    pivoted_aggr_data = pd.pivot(\n",
    "        formatted_df,\n",
    "        index='dataset',\n",
    "        columns=\"Experiment\",\n",
    "        values=\"mean_std\"\n",
    "    )\n",
    "    pivoted_aggr_data = pivoted_aggr_data.loc[:, curr_order]\n",
    "    pivoted_aggr_data = pivoted_aggr_data.sort_values(curr_order[-2], key=lambda x: x.apply(lambda val: float(val.split('±')[0].split('{')[-1].strip())))\n",
    "    \n",
    "    pivoted_aggr_data.index.name = None\n",
    "    pivoted_aggr_data.columns.name = None\n",
    "    pivoted_aggr_data = pivoted_aggr_data.reset_index(names='Dataset')\n",
    "    pivoted_aggr_data.insert(0, 'Category', ds_mapping.loc[pivoted_aggr_data[\"Dataset\"].tolist(), 'dataset_domain'].reset_index(drop=True))\n",
    "    pivoted_aggr_data['Dataset'] = ds_mapping.loc[pivoted_aggr_data[\"Dataset\"].tolist(), 'dataset_fmt'].reset_index(drop=True)\n",
    "\n",
    "    pivoted_aggr_data = pivoted_aggr_data.sort_values('Category', kind='stable')\n",
    "\n",
    "    latex_version_pivoted_aggr_data = pivoted_aggr_data.to_latex(escape=False, index=False)\n",
    "\n",
    "    if SAVE:\n",
    "        filename = base_storing_path / 'tables_for_per_model_size_n_dataset_perf_gain' / f\"{metric_col}_aggr_over_models_table_v1.tex\"  # Creates filename based on metric_col\n",
    "        with open(filename, 'w', encoding='utf-8') as f:\n",
    "            f.write(latex_version_pivoted_aggr_data)\n",
    "        print(f\"Stored latex table at {filename=}\")\n",
    "\n",
    "    else:\n",
    "        print(metric_col)\n",
    "        print()\n",
    "        print(latex_version_pivoted_aggr_data)\n",
    "        print()\n",
    "        print()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "69e6f509-6b64-4394-ab4e-988d7d16a15d",
   "metadata": {},
   "source": [
    "### V2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "id": "3c4369d2-35a4-4d97-a183-3d97764c711b",
   "metadata": {},
   "outputs": [],
   "source": [
    "grouping_cols = [\"dataset\", \"Experiment\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 79,
   "id": "87df830f-2027-4426-867e-059e044a8ff2",
   "metadata": {},
   "outputs": [],
   "source": [
    "metrics_cols = [\n",
    "    'abs_perf_gain_train_lp_bal_acc1_mod',\n",
    "    'abs_perf_gain_test_lp_bal_acc1_mod',\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "id": "2907ba2f-9acd-4134-9584-e69c7473e622",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_runs['abs_perf_gain_train_lp_bal_acc1_mod'] = all_runs['abs_perf_gain_train_lp_bal_acc1'].copy().astype(float)\n",
    "all_runs['abs_perf_gain_test_lp_bal_acc1_mod'] = all_runs['abs_perf_gain_test_lp_bal_acc1'].copy().astype(float)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "id": "c1643a64-8d38-4093-ac36-ba5700d7abfe",
   "metadata": {},
   "outputs": [],
   "source": [
    "all_runs.loc[all_runs['Experiment'] == 'CLS last layer', 'abs_perf_gain_train_lp_bal_acc1_mod'] = all_runs.loc[all_runs['Experiment'] == 'CLS last layer', 'train_lp_bal_acc1'].astype(float)\n",
    "all_runs.loc[all_runs['Experiment'] == 'CLS last layer', 'abs_perf_gain_test_lp_bal_acc1_mod'] = all_runs.loc[all_runs['Experiment'] == 'CLS last layer', 'test_lp_bal_acc1'].astype(float)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "id": "b26a4842-a1de-4f11-b437-84bba7071424",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['CLS last layer', 'All tokens last layer (attentive)', 'CLS+AP last layer (linear)', 'CLS+AP layers from all blocks (linear)', 'CLS+AP last layer (attentive)', 'CLS+AP layers from all blocks (attentive)']\n"
     ]
    }
   ],
   "source": [
    "curr_order = [col for col in experiment_with_probe_type_order_list if (\"middle & last\" not in col) and (\"quarterly\" not in col)]\n",
    "print(curr_order)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "id": "015f3ba4-a3e9-49b6-a6ab-7c67fb288fdb",
   "metadata": {},
   "outputs": [],
   "source": [
    "subset_runs = all_runs[all_runs['Experiment'].isin(curr_order)].copy().reset_index(drop=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "id": "42040943-ad1f-4324-806a-aec4274c4c3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "grouped_data = subset_runs.groupby(grouping_cols)\n",
    "ds_mapping = subset_runs[['dataset', 'dataset_fmt', 'dataset_domain']].value_counts().reset_index().set_index('dataset')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "id": "252a683d-971f-4b1b-a291-f732b38ae023",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# for metric_col in metrics_cols:\n",
    "#     aggr_data = grouped_data[metric_col].agg([\"mean\", \"std\"]).reset_index()\n",
    "    \n",
    "#     # Get means for ranking\n",
    "#     mean_pivot = pd.pivot(\n",
    "#         aggr_data,\n",
    "#         index='dataset',\n",
    "#         columns=\"Experiment\", \n",
    "#         values=\"mean\"\n",
    "#     ).loc[:, curr_order]\n",
    "    \n",
    "#     formatted_data = []\n",
    "#     for idx, row in aggr_data.iterrows():\n",
    "#         model = row['dataset']\n",
    "#         exp = row['Experiment']\n",
    "#         if model in mean_pivot.index and exp in mean_pivot.columns:\n",
    "#             row_means = mean_pivot.loc[model].values\n",
    "#             formatted = format_for_latex(row['mean'], row['std'], row_means[1:])\n",
    "#             formatted_data.append({\n",
    "#                 'dataset': model,\n",
    "#                 'Experiment': exp,\n",
    "#                 'mean_std': formatted\n",
    "#             })\n",
    "    \n",
    "#     formatted_df = pd.DataFrame(formatted_data)\n",
    "#     pivoted_aggr_data = pd.pivot(\n",
    "#         formatted_df,\n",
    "#         index='dataset',\n",
    "#         columns=\"Experiment\",\n",
    "#         values=\"mean_std\"\n",
    "#     )\n",
    "#     pivoted_aggr_data = pivoted_aggr_data.loc[:, curr_order]\n",
    "#     pivoted_aggr_data = pivoted_aggr_data.sort_values(curr_order[-1], key=lambda x: x.apply(lambda val: float(val.split('±')[0].split('{')[-1].strip())))\n",
    "    \n",
    "#     pivoted_aggr_data.index.name = None\n",
    "#     pivoted_aggr_data.columns.name = None\n",
    "#     pivoted_aggr_data = pivoted_aggr_data.reset_index(names='Dataset')\n",
    "#     pivoted_aggr_data.insert(0, 'Category', ds_mapping.loc[pivoted_aggr_data[\"Dataset\"].tolist(), 'dataset_domain'].reset_index(drop=True))\n",
    "#     pivoted_aggr_data['Dataset'] = ds_mapping.loc[pivoted_aggr_data[\"Dataset\"].tolist(), 'dataset_fmt'].reset_index(drop=True)\n",
    "\n",
    "#     pivoted_aggr_data = pivoted_aggr_data.sort_values('Category', kind='stable')\n",
    "\n",
    "#     latex_version_pivoted_aggr_data = pivoted_aggr_data.to_latex(escape=False, index=False)\n",
    "\n",
    "#     if SAVE:\n",
    "#         filename = base_storing_path / 'tables_for_per_model_size_n_dataset_perf_gain' / f\"{metric_col}_aggr_over_models_table_v2.tex\"  # Creates filename based on metric_col\n",
    "#         with open(filename, 'w', encoding='utf-8') as f:\n",
    "#             f.write(latex_version_pivoted_aggr_data)\n",
    "#         print(f\"Stored latex table at {filename=}\")\n",
    "\n",
    "#     else:\n",
    "#         print(metric_col)\n",
    "#         print()\n",
    "#         print(latex_version_pivoted_aggr_data)\n",
    "#         print()\n",
    "#         print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "id": "a328d0cc-371e-4641-bc6e-115bc8a039fd",
   "metadata": {},
   "outputs": [],
   "source": [
    "# all_runs['dataset'].unique()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "bce5c239-5b37-430a-b5fb-e40bbb5f74fb",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "# tmp = all_runs[all_runs['dataset']=='wds/fgvc_aircraft'][['Experiment','base_model_fmt','test_lp_bal_acc1']]\n",
    "\n",
    "# for exp, exp_data in tmp.groupby('Experiment'):\n",
    "#     print(exp)\n",
    "#     display(exp_data.sort_values('base_model_fmt'))\n",
    "#     print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "id": "59eaf9d7-a3b1-4b72-96ad-70cfbb44e70e",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_p_values_wrt_to_highest_mean(ds_data, alpha=0.05):\n",
    "    best_mean_exp_per_ds = ds_data.groupby(\"Experiment\")[metric_col].mean().idxmax()\n",
    "\n",
    "    pvalues = {best_mean_exp_per_ds: np.nan}\n",
    "    statistics = {best_mean_exp_per_ds:np.nan}\n",
    "    \n",
    "    print(ds_data['dataset'].unique(), best_mean_exp_per_ds)\n",
    "    exp1_data = pd.to_numeric(ds_data[ds_data['Experiment'] == best_mean_exp_per_ds].sort_values('base_model_fmt')[metric_col]).values\n",
    "\n",
    "    for exp in sorted(ds_data[\"Experiment\"].unique()):\n",
    "        if exp == best_mean_exp_per_ds:\n",
    "            continue\n",
    "        \n",
    "        exp2_data = pd.to_numeric(ds_data[ds_data['Experiment'] == exp].sort_values('base_model_fmt')[metric_col]).values\n",
    "        if len(exp1_data) != len(exp2_data):\n",
    "            continue\n",
    "\n",
    "        assert len(exp1_data) == len(exp2_data)\n",
    "        \n",
    "        statistic, pval = mannwhitneyu(exp1_data, exp2_data, alternative='greater')\n",
    "    \n",
    "        pvalues[exp] = pval\n",
    "        statistics[exp] = statistic\n",
    "\n",
    "    rejected = {exp: pval < alpha for exp, pval in pvalues.items()}\n",
    "    print()\n",
    "    return dict(pvalues=pvalues, rejected=rejected, statistics=statistics)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "id": "6d7805b9-1561-458b-b69e-a8f38e396fa2",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['wds/cars'] All tokens last layer (attentive)\n",
      "\n",
      "['wds/country211'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/fer2013'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/fgvc_aircraft'] All tokens last layer (attentive)\n",
      "\n",
      "['wds/gtsrb'] All tokens last layer (attentive)\n",
      "\n",
      "['wds/stl10'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/voc2007'] CLS+AP layers from all blocks (linear)\n",
      "\n",
      "['wds/vtab/caltech101'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/cifar10'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/cifar100'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/diabetic_retinopathy'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/dmlab'] All tokens last layer (attentive)\n",
      "\n",
      "['wds/vtab/dtd'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/eurosat'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/flowers'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/pets'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/resisc45'] CLS+AP layers from all blocks (attentive)\n",
      "\n",
      "['wds/vtab/svhn'] All tokens last layer (attentive)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2413980/3585573844.py:4: FutureWarning: DataFrameGroupBy.apply operated on the grouping columns. This behavior is deprecated, and in a future version of pandas the grouping columns will be excluded from the operation. Either pass `include_groups=False` to exclude the groupings or explicitly select the grouping columns after groupby to silence this warning.\n",
      "  res = grouped_data.apply(get_p_values_wrt_to_highest_mean)\n"
     ]
    }
   ],
   "source": [
    "metric_col = 'test_lp_bal_acc1'\n",
    "grouped_data = subset_runs.groupby('dataset')\n",
    "\n",
    "res = grouped_data.apply(get_p_values_wrt_to_highest_mean)\n",
    "res_df = pd.DataFrame(res.tolist(), index=res.index)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "id": "ceb0d8e4-af47-4f38-a2f7-c2540a84ada8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['CLS last layer', 'All tokens last layer (attentive)', 'CLS+AP last layer (linear)', 'CLS+AP layers from all blocks (linear)', 'CLS+AP last layer (attentive)', 'CLS+AP layers from all blocks (attentive)']\n"
     ]
    }
   ],
   "source": [
    "grouped_data = subset_runs.groupby(grouping_cols)\n",
    "curr_order = [col for col in experiment_with_probe_type_order_list if (\"middle & last\" not in col) and (\"quarterly\" not in col)]\n",
    "print(curr_order)\n",
    "ds_mapping = subset_runs[['dataset', 'dataset_fmt', 'dataset_domain']].value_counts().reset_index().set_index('dataset')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "2de78b38-1bcb-4588-b322-700655af6038",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "abs_perf_gain_train_lp_bal_acc1_mod\n",
      "\n",
      "\\begin{tabular}{llllllll}\n",
      "\\toprule\n",
      "Category & Dataset & CLS last layer & All tokens last layer (attentive) & CLS+AP last layer (linear) & CLS+AP layers from all blocks (linear) & CLS+AP last layer (attentive) & CLS+AP layers from all blocks (attentive) \\\\\n",
      "\\midrule\n",
      "Natural (multi-domain) & STL-10 & \\textbf{0.99 ± 0.00} & \\textbf{0.57 ± 0.45} & \\textbf{0.11 ± 0.34} & \\textbf{0.56 ± 0.42} & \\textbf{0.21 ± 0.22} & \\textbf{0.52 ± 0.44} \\\\\n",
      "Natural (multi-domain) & Caltech-101 & \\textbf{0.99 ± 0.01} & \\textbf{1.18 ± 0.97} & \\textbf{0.72 ± 0.72} & \\textbf{1.18 ± 0.97} & \\textbf{0.30 ± 0.50} & \\textbf{1.18 ± 0.97} \\\\\n",
      "Natural (multi-domain) & CIFAR-10 & \\textbf{0.96 ± 0.03} & \\textbf{3.88 ± 2.81} & \\textbf{0.73 ± 0.64} & \\textbf{2.29 ± 1.70} & \\textbf{0.71 ± 0.76} & \\textbf{2.67 ± 1.81} \\\\\n",
      "Natural (multi-domain) & PASCAL VOC 2007 & \\textbf{0.90 ± 0.04} & \\textbf{9.40 ± 4.06} & \\textbf{2.93 ± 1.51} & \\textbf{7.21 ± 4.17} & \\textbf{3.55 ± 0.96} & \\textbf{9.03 ± 3.76} \\\\\n",
      "Natural (multi-domain) & CIFAR-100 & \\textbf{0.84 ± 0.10} & \\textbf{15.97 ± 9.55} & \\textbf{2.51 ± 1.45} & \\textbf{9.72 ± 6.91} & \\textbf{2.80 ± 1.56} & \\textbf{11.90 ± 6.69} \\\\\n",
      "Natural (multi-domain) & Country-211 & \\textbf{0.31 ± 0.05} & \\textbf{66.31 ± 9.81} & \\textbf{5.59 ± 2.81} & \\textbf{24.84 ± 8.62} & \\textbf{8.62 ± 6.98} & \\textbf{48.02 ± 6.04} \\\\\n",
      "Natural (single-domain) & Flowers & \\textbf{0.99 ± 0.01} & \\textbf{0.60 ± 0.98} & \\textbf{0.45 ± 0.77} & \\textbf{0.60 ± 0.98} & \\textbf{-0.25 ± 0.94} & \\textbf{0.60 ± 0.98} \\\\\n",
      "Natural (single-domain) & Pets & \\textbf{0.97 ± 0.04} & \\textbf{3.12 ± 3.55} & \\textbf{0.88 ± 0.92} & 3.09 ± 3.55 & \\textbf{0.61 ± 0.89} & \\textbf{2.92 ± 3.24} \\\\\n",
      "Natural (single-domain) & Stanford Cars & 0.86 ± 0.05 & \\textbf{13.65 ± 5.44} & 4.03 ± 2.22 & 13.41 ± 5.37 & \\textbf{2.12 ± 2.82} & \\textbf{12.62 ± 5.27} \\\\\n",
      "Natural (single-domain) & GTSRB & 0.84 ± 0.03 & \\textbf{15.76 ± 3.13} & 5.74 ± 2.03 & 15.20 ± 2.97 & 5.83 ± 2.20 & 15.10 ± 2.87 \\\\\n",
      "Natural (single-domain) & FGVC Aircraft & \\textbf{0.74 ± 0.09} & \\textbf{25.42 ± 8.32} & \\textbf{3.75 ± 3.04} & 23.58 ± 7.18 & \\textbf{3.42 ± 3.98} & \\textbf{21.81 ± 8.30} \\\\\n",
      "Natural (single-domain) & SVHN & 0.44 ± 0.04 & \\textbf{47.77 ± 5.29} & 7.29 ± 2.38 & 29.23 ± 3.83 & 6.57 ± 3.70 & \\textbf{32.30 ± 4.33} \\\\\n",
      "Specialized & EuroSAT & 0.90 ± 0.05 & 10.10 ± 5.42 & 3.44 ± 1.87 & 9.32 ± 5.14 & 3.14 ± 1.24 & \\textbf{9.61 ± 5.07} \\\\\n",
      "Specialized & RESISC45 & 0.88 ± 0.02 & 12.11 ± 2.11 & 3.37 ± 0.95 & \\textbf{10.99 ± 1.87} & 3.24 ± 1.12 & \\textbf{10.96 ± 1.88} \\\\\n",
      "Specialized & Diabetic Retinopathy & 0.44 ± 0.03 & 28.78 ± 4.98 & 2.39 ± 0.70 & \\textbf{11.05 ± 2.85} & 1.74 ± 1.26 & \\textbf{12.02 ± 3.38} \\\\\n",
      "Structured & DTD & 0.87 ± 0.06 & \\textbf{12.49 ± 5.54} & 3.84 ± 7.04 & \\textbf{12.44 ± 5.52} & \\textbf{8.23 ± 3.91} & \\textbf{12.28 ± 5.47} \\\\\n",
      "Structured & FER2013 & 0.56 ± 0.03 & \\textbf{23.30 ± 6.85} & 3.40 ± 1.12 & 11.37 ± 1.59 & 3.74 ± 1.07 & \\textbf{16.47 ± 2.07} \\\\\n",
      "Structured & Dmlab & 0.43 ± 0.03 & \\textbf{31.50 ± 3.45} & 2.25 ± 0.72 & 11.58 ± 2.24 & 2.07 ± 1.85 & \\textbf{17.40 ± 2.45} \\\\\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\n",
      "\n",
      "\n",
      "abs_perf_gain_test_lp_bal_acc1_mod\n",
      "\n",
      "\\begin{tabular}{llllllll}\n",
      "\\toprule\n",
      "Category & Dataset & CLS last layer & All tokens last layer (attentive) & CLS+AP last layer (linear) & CLS+AP layers from all blocks (linear) & CLS+AP last layer (attentive) & CLS+AP layers from all blocks (attentive) \\\\\n",
      "\\midrule\n",
      "Natural (multi-domain) & STL-10 & \\textbf{99.29 ± 0.51} & \\textbf{0.01 ± 0.16} & \\textbf{-0.01 ± 0.12} & \\textbf{0.03 ± 0.10} & \\textbf{0.03 ± 0.08} & \\textbf{0.04 ± 0.17} \\\\\n",
      "Natural (multi-domain) & CIFAR-10 & \\textbf{96.91 ± 1.93} & \\textbf{0.42 ± 0.58} & \\textbf{0.08 ± 0.11} & \\textbf{0.61 ± 0.71} & \\textbf{0.19 ± 0.29} & \\textbf{0.77 ± 0.79} \\\\\n",
      "Natural (multi-domain) & Caltech-101 & \\textbf{95.57 ± 1.40} & \\textbf{0.23 ± 0.52} & \\textbf{0.43 ± 0.41} & \\textbf{0.36 ± 0.63} & \\textbf{0.09 ± 0.42} & \\textbf{0.88 ± 0.77} \\\\\n",
      "Natural (multi-domain) & PASCAL VOC 2007 & \\textbf{87.82 ± 2.31} & \\textbf{-0.22 ± 1.24} & \\textbf{1.38 ± 0.49} & \\textbf{1.46 ± 0.99} & \\textbf{1.19 ± 0.88} & \\textbf{1.24 ± 0.89} \\\\\n",
      "Natural (multi-domain) & CIFAR-100 & \\textbf{85.45 ± 5.71} & \\textbf{1.73 ± 1.33} & \\textbf{0.61 ± 0.21} & \\textbf{2.76 ± 2.48} & \\textbf{0.87 ± 0.56} & \\textbf{3.33 ± 2.75} \\\\\n",
      "Natural (multi-domain) & Country-211 & \\textbf{21.48 ± 6.35} & \\textbf{-0.83 ± 1.66} & \\textbf{1.18 ± 0.54} & \\textbf{3.26 ± 1.05} & \\textbf{1.35 ± 0.65} & \\textbf{4.96 ± 1.37} \\\\\n",
      "Natural (single-domain) & Pets & \\textbf{93.98 ± 2.36} & \\textbf{-0.23 ± 0.83} & \\textbf{-0.05 ± 0.41} & -2.01 ± 1.04 & \\textbf{0.12 ± 0.53} & \\textbf{0.29 ± 0.76} \\\\\n",
      "Natural (single-domain) & Flowers & \\textbf{98.03 ± 2.60} & \\textbf{0.41 ± 0.93} & \\textbf{0.40 ± 0.75} & \\textbf{-0.25 ± 0.57} & \\textbf{0.06 ± 0.76} & \\textbf{0.46 ± 0.97} \\\\\n",
      "Natural (single-domain) & Stanford Cars & 77.81 ± 10.65 & \\textbf{8.97 ± 5.22} & 0.50 ± 1.07 & -0.86 ± 3.76 & \\textbf{1.97 ± 1.95} & \\textbf{6.35 ± 3.71} \\\\\n",
      "Natural (single-domain) & FGVC Aircraft & \\textbf{55.69 ± 12.18} & \\textbf{9.27 ± 4.37} & \\textbf{-0.96 ± 2.22} & -1.62 ± 5.01 & \\textbf{1.84 ± 2.09} & \\textbf{6.43 ± 3.25} \\\\\n",
      "Natural (single-domain) & GTSRB & 71.51 ± 7.46 & \\textbf{18.02 ± 6.37} & 4.23 ± 2.60 & 8.76 ± 4.20 & 4.69 ± 2.41 & 13.47 ± 4.92 \\\\\n",
      "Natural (single-domain) & SVHN & 56.06 ± 5.91 & \\textbf{30.31 ± 5.08} & 6.94 ± 2.59 & 24.40 ± 4.41 & 7.39 ± 3.70 & \\textbf{27.25 ± 4.24} \\\\\n",
      "Specialized & EuroSAT & 93.89 ± 2.52 & 3.38 ± 2.18 & 1.65 ± 1.17 & 4.08 ± 2.48 & 1.82 ± 1.22 & \\textbf{4.37 ± 2.41} \\\\\n",
      "Specialized & RESISC45 & 90.45 ± 1.69 & 4.07 ± 1.05 & 1.32 ± 0.74 & \\textbf{4.53 ± 0.99} & 1.82 ± 0.59 & \\textbf{5.23 ± 1.10} \\\\\n",
      "Specialized & Diabetic Retinopathy & 45.80 ± 2.46 & 1.94 ± 1.90 & 1.55 ± 0.44 & \\textbf{5.92 ± 2.03} & 1.86 ± 0.77 & \\textbf{6.86 ± 2.00} \\\\\n",
      "Structured & DTD & 75.99 ± 3.47 & \\textbf{1.41 ± 2.19} & 1.18 ± 1.76 & \\textbf{4.04 ± 2.19} & \\textbf{2.53 ± 1.67} & \\textbf{4.05 ± 1.92} \\\\\n",
      "Structured & FER2013 & 59.08 ± 4.61 & \\textbf{7.74 ± 2.15} & 2.18 ± 1.05 & 6.25 ± 1.19 & 3.61 ± 1.13 & \\textbf{10.05 ± 1.76} \\\\\n",
      "Structured & Dmlab & 44.91 ± 3.49 & \\textbf{13.69 ± 2.77} & 1.81 ± 0.45 & 7.92 ± 1.95 & 2.61 ± 1.65 & \\textbf{10.68 ± 2.78} \\\\\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\n",
      "\n",
      "\n"
     ]
    }
   ],
   "source": [
    "for metric_col in metrics_cols:\n",
    "    aggr_data = grouped_data[metric_col].agg([\"mean\", \"std\"]).reset_index()\n",
    "    \n",
    "    mean_pivot = pd.pivot(\n",
    "        aggr_data,\n",
    "        index='dataset',\n",
    "        columns=\"Experiment\", \n",
    "        values=\"mean\"\n",
    "    ).loc[:, curr_order]\n",
    "    \n",
    "    formatted_data = []\n",
    "    for idx, row in aggr_data.iterrows():\n",
    "        dataset = row['dataset']\n",
    "        exp = row['Experiment']\n",
    "        if dataset in mean_pivot.index and exp in mean_pivot.columns:\n",
    "            not_bold = res_df.loc[dataset, 'rejected'][exp]\n",
    "            formatted = format_if_is_bold(row['mean'], row['std'], not not_bold)\n",
    "            formatted_data.append({\n",
    "                'dataset': dataset,\n",
    "                'Experiment': exp,\n",
    "                'mean_std': formatted\n",
    "            }) \n",
    "    \n",
    "    formatted_df = pd.DataFrame(formatted_data)\n",
    "    pivoted_aggr_data = pd.pivot(\n",
    "        formatted_df,\n",
    "        index='dataset',\n",
    "        columns=\"Experiment\",\n",
    "        values=\"mean_std\"\n",
    "    )\n",
    "    pivoted_aggr_data = pivoted_aggr_data.loc[:, curr_order]\n",
    "    pivoted_aggr_data = pivoted_aggr_data.sort_values(curr_order[-1], key=lambda x: x.apply(lambda val: float(val.split('±')[0].split('{')[-1].strip())))\n",
    "    \n",
    "    pivoted_aggr_data.index.name = None\n",
    "    pivoted_aggr_data.columns.name = None\n",
    "    pivoted_aggr_data = pivoted_aggr_data.reset_index(names='Dataset')\n",
    "    pivoted_aggr_data.insert(0, 'Category', ds_mapping.loc[pivoted_aggr_data[\"Dataset\"].tolist(), 'dataset_domain'].reset_index(drop=True))\n",
    "    pivoted_aggr_data['Dataset'] = ds_mapping.loc[pivoted_aggr_data[\"Dataset\"].tolist(), 'dataset_fmt'].reset_index(drop=True)\n",
    "\n",
    "    pivoted_aggr_data = pivoted_aggr_data.sort_values('Category', kind='stable')\n",
    "\n",
    "    latex_version_pivoted_aggr_data = pivoted_aggr_data.to_latex(escape=False, index=False)\n",
    "\n",
    "    if SAVE:\n",
    "        filename = base_storing_path / 'tables_for_per_model_size_n_dataset_perf_gain' / f\"{metric_col}_aggr_over_models_table_v2.tex\"  # Creates filename based on metric_col\n",
    "        with open(filename, 'w', encoding='utf-8') as f:\n",
    "            f.write(latex_version_pivoted_aggr_data)\n",
    "        print(f\"Stored latex table at {filename=}\")\n",
    "\n",
    "    else:\n",
    "        print(metric_col)\n",
    "        print()\n",
    "        print(latex_version_pivoted_aggr_data)\n",
    "        print()\n",
    "        print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "caed254f-c02f-48de-a8af-f15221eeec25",
   "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.10.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
