{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 252,
   "id": "8510fc3e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import scienceplots  # noqa\n",
    "import seaborn as sns\n",
    "\n",
    "plt.style.use(['science', 'grid'])\n",
    "plt.rcParams.update({'font.size': 16})\n",
    "\n",
    "# Set seaborn colorblind palette\n",
    "sns.set_theme(style='whitegrid')\n",
    "colors = sns.color_palette('deep')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 253,
   "id": "70a7a90a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "B3LYP checkpoint configuration:\n",
      "\n",
      "EGXC_b3lyp_qm7:\n",
      "  baseline: 58\n",
      "  gradient: 61\n",
      "  grad_and_hess: 76\n",
      "\n",
      "NNmGGA_b3lyp_qm7:\n",
      "  baseline: 64\n",
      "  grad: 10\n",
      "  hessian: 3\n",
      "  grad_and_hessian: 11\n",
      "\n",
      "Skala_mGGA_b3lyp_qm7:\n",
      "  baseline: 6\n",
      "  gradient: 18\n",
      "  grad_and_hessian: 21\n",
      "  hessian: 19\n"
     ]
    }
   ],
   "source": [
    "# ===== B3LYP Model Results =====\n",
    "# Configuration flags\n",
    "\n",
    "# Configuration: Specify checkpoint integers for each model and loss term\n",
    "# Modify these checkpoint IDs as needed\n",
    "\n",
    "b3lyp_checkpoints = {\n",
    "    'EGXC_b3lyp_qm7': {\n",
    "        'baseline': 58,\n",
    "        'gradient': 61,\n",
    "        'grad_and_hess': 76,\n",
    "    },\n",
    "    'NNmGGA_b3lyp_qm7': {\n",
    "        'baseline': 64,\n",
    "        'grad': 10,\n",
    "        'hessian': 3,\n",
    "        'grad_and_hessian': 11,\n",
    "    },\n",
    "    'Skala_mGGA_b3lyp_qm7': {\n",
    "        'baseline': 6,\n",
    "        'gradient': 18,\n",
    "        'grad_and_hessian': 21,\n",
    "        'hessian': 19,\n",
    "    },\n",
    "}\n",
    "\n",
    "# # Model-specific checkpoint configuration (for overlay plots)\n",
    "# # Uncomment and modify these checkpoint IDs for model-specific evaluations\n",
    "# b3lyp_checkpoints_model_specific = {\n",
    "#     'EGXC_b3lyp_qm7': {\n",
    "#         'gradient': 59,\n",
    "#         'grad_and_hess': 74,\n",
    "#     },\n",
    "#     'NNmGGA_b3lyp_qm7': {\n",
    "#         'grad': 58,\n",
    "#         'grad_and_hessian': 10,\n",
    "#         'hessian': 2,\n",
    "#     },\n",
    "#     'Skala_mGGA_b3lyp_qm7': {\n",
    "#         'baseline': 6,\n",
    "#         'gradient': 17,\n",
    "#         'grad_and_hessian': 21,\n",
    "#         'hessian': 19,\n",
    "#     },\n",
    "# }\n",
    "\n",
    "print('B3LYP checkpoint configuration:')\n",
    "for model, loss_terms in b3lyp_checkpoints.items():\n",
    "    print(f'\\n{model}:')\n",
    "    for loss_term, checkpoint_id in loss_terms.items():\n",
    "        print(f'  {loss_term}: {checkpoint_id}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 254,
   "id": "0b2a155f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Processing EGXC_b3lyp_qm7/baseline ($E + \\rho$), checkpoint 58\n",
      "Processing EGXC_b3lyp_qm7/gradient ($+ \\nabla$ (ours)), checkpoint 61\n",
      "Processing EGXC_b3lyp_qm7/grad_and_hess ($+ \\nabla + H$ (ours)), checkpoint 76\n",
      "Processing NNmGGA_b3lyp_qm7/baseline ($E + \\rho$), checkpoint 64\n",
      "Processing NNmGGA_b3lyp_qm7/grad ($+ \\nabla$ (ours)), checkpoint 10\n",
      "Processing NNmGGA_b3lyp_qm7/hessian ($+ H$ (ours)), checkpoint 3\n",
      "Processing NNmGGA_b3lyp_qm7/grad_and_hessian ($+ \\nabla + H$ (ours)), checkpoint 11\n",
      "Processing Skala_mGGA_b3lyp_qm7/baseline ($E + \\rho$), checkpoint 6\n",
      "Processing Skala_mGGA_b3lyp_qm7/gradient ($+ \\nabla$ (ours)), checkpoint 18\n",
      "Processing Skala_mGGA_b3lyp_qm7/grad_and_hessian ($+ \\nabla + H$ (ours)), checkpoint 21\n",
      "Processing Skala_mGGA_b3lyp_qm7/hessian ($+ H$ (ours)), checkpoint 19\n",
      "=== All Available B3LYP Metrics ===\n",
      "\n",
      "Available metrics: ['idx', 'cycles_to_convergence', 'relative_cycle_count', 'total_energy_mae_mEh', 'xc_energy_mae_mEh', 'mean_field_mae_mEh', 'coulomb_energy_mae_mEh', 'dipole_difference_au', 'homo_lumo_gap_mae_mEh', 'density_L1', 'density_L2', 'overlap_based_mae_surrogate', 'overlap_based_mse_surrogate', 'ref_total_energy_on_model_dm_mEh', 'ref_xc_energy_on_model_dm_mEh', 'ref_cycles_cold', 'ref_cycles_warm', 'warm_start_ric', 'ref_energy_after_1_cycle_mEh']\n",
      "\n",
      "\n",
      "coulomb_energy_mae_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           179.16±0.00    25.25±0.00     5.95±0.00      \n",
      "NNmGGA         179.28±0.00    9.41±0.00      23.21±0.00     7.66±0.00      \n",
      "Skala_mGGA     127.33±0.00    9.81±0.00      8.41±0.00      77.67±0.00     \n",
      "\n",
      "cycles_to_convergence:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           9.17±0.00      9.28±0.00      8.81±0.00      \n",
      "NNmGGA         9.19±0.00      9.22±0.00      8.52±0.00      8.98±0.00      \n",
      "Skala_mGGA     8.77±0.00      9.09±0.00      8.90±0.00      8.54±0.00      \n",
      "\n",
      "density_L1:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           0.2314±0.0000  0.0449±0.0000  0.0559±0.0000  \n",
      "NNmGGA         0.2359±0.0000  0.0428±0.0000  0.1423±0.0000  0.0424±0.0000  \n",
      "Skala_mGGA     0.1267±0.0000  0.0481±0.0000  0.0473±0.0000  0.1698±0.0000  \n",
      "\n",
      "density_L2:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           3.97e-04±0.00e+001.67e-05±0.00e+001.65e-05±0.00e+00\n",
      "NNmGGA         4.20e-04±0.00e+001.36e-05±0.00e+008.76e-05±0.00e+001.34e-05±0.00e+00\n",
      "Skala_mGGA     3.50e-04±0.00e+001.78e-05±0.00e+001.81e-05±0.00e+001.51e-04±0.00e+00\n",
      "\n",
      "dipole_difference_au:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           0.0292±0.0000  8.81e-03±0.00e+000.0103±0.0000  \n",
      "NNmGGA         0.0302±0.0000  7.92e-03±0.00e+000.0259±0.0000  7.45e-03±0.00e+00\n",
      "Skala_mGGA     0.0288±0.0000  0.0111±0.0000  9.58e-03±0.00e+000.0216±0.0000  \n",
      "\n",
      "homo_lumo_gap_mae_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           53.68±0.00     50.13±0.00     48.29±0.00     \n",
      "NNmGGA         53.95±0.00     49.36±0.00     51.32±0.00     47.47±0.00     \n",
      "Skala_mGGA     57.98±0.00     49.88±0.00     48.14±0.00     51.61±0.00     \n",
      "\n",
      "idx:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           74160.42±0.00  74160.42±0.00  74160.42±0.00  \n",
      "NNmGGA         74160.42±0.00  74160.42±0.00  74160.42±0.00  74160.42±0.00  \n",
      "Skala_mGGA     74160.42±0.00  74160.42±0.00  74160.42±0.00  74160.42±0.00  \n",
      "\n",
      "mean_field_mae_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           1.38±0.00      0.5148±0.0000  0.4623±0.0000  \n",
      "NNmGGA         1.46±0.00      0.8721±0.0000  1.35±0.00      0.6521±0.0000  \n",
      "Skala_mGGA     0.6199±0.0000  0.7351±0.0000  0.6794±0.0000  1.28±0.00      \n",
      "\n",
      "overlap_based_mae_surrogate:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           3.23±0.00      0.9141±0.0000  1.36±0.00      \n",
      "NNmGGA         3.26±0.00      1.02±0.00      3.29±0.00      1.07±0.00      \n",
      "Skala_mGGA     2.00±0.00      0.9986±0.0000  1.06±0.00      3.27±0.00      \n",
      "\n",
      "overlap_based_mse_surrogate:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           8.45e-03±0.00e+001.03e-03±0.00e+001.48e-03±0.00e+00\n",
      "NNmGGA         8.88e-03±0.00e+001.03e-03±0.00e+005.44e-03±0.00e+001.10e-03±0.00e+00\n",
      "Skala_mGGA     0.0102±0.0000  1.14e-03±0.00e+001.18e-03±0.00e+006.68e-03±0.00e+00\n",
      "\n",
      "ref_cycles_cold:\n",
      "Model          Loss_0         \n",
      "------------------------------\n",
      "EGXC           8.10±0.00      \n",
      "NNmGGA         8.10±0.00      8.10±0.00      8.10±0.00      \n",
      "Skala_mGGA     8.10±0.00      8.10±0.00      8.10±0.00      \n",
      "\n",
      "ref_cycles_warm:\n",
      "Model          Loss_0         \n",
      "------------------------------\n",
      "EGXC           4.17±0.00      \n",
      "NNmGGA         5.53±0.00      3.81±0.00      3.77±0.00      \n",
      "Skala_mGGA     5.50±0.00      3.98±0.00      3.93±0.00      \n",
      "\n",
      "ref_energy_after_1_cycle_mEh:\n",
      "Model          Loss_0         \n",
      "------------------------------\n",
      "EGXC           9.62e-03±0.00e+00\n",
      "NNmGGA         0.2226±0.0000  6.94e-03±0.00e+005.14e-03±0.00e+00\n",
      "Skala_mGGA     0.1662±0.0000  6.70e-03±0.00e+005.64e-03±0.00e+00\n",
      "\n",
      "ref_total_energy_on_model_dm_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           2.76±0.00      0.2640±0.0000  0.3803±0.0000  \n",
      "NNmGGA         2.93±0.00      0.2890±0.0000  1.27±0.00      0.3015±0.0000  \n",
      "Skala_mGGA     2.41±0.00      0.3203±0.0000  0.3251±0.0000  1.47±0.00      \n",
      "\n",
      "ref_xc_energy_on_model_dm_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           3.57±0.00      0.5055±0.0000  0.5472±0.0000  \n",
      "NNmGGA         3.79±0.00      0.8688±0.0000  2.58±0.00      0.7000±0.0000  \n",
      "Skala_mGGA     2.17±0.00      0.8024±0.0000  0.8429±0.0000  2.72±0.00      \n",
      "\n",
      "relative_cycle_count:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           1.14±0.00      1.15±0.00      1.09±0.00      \n",
      "NNmGGA         1.14±0.00      1.14±0.00      1.05±0.00      1.11±0.00      \n",
      "Skala_mGGA     1.09±0.00      1.13±0.00      1.10±0.00      1.06±0.00      \n",
      "\n",
      "total_energy_mae_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           1.17±0.00      0.4483±0.0000  0.5411±0.0000  \n",
      "NNmGGA         2.02±0.00      1.57±0.00      1.27±0.00      1.44±0.00      \n",
      "Skala_mGGA     0.8932±0.0000  2.02±0.00      1.69±0.00      1.51±0.00      \n",
      "\n",
      "warm_start_ric:\n",
      "Model          Loss_0         \n",
      "------------------------------\n",
      "EGXC           0.5174±0.0000  \n",
      "NNmGGA         0.6857±0.0000  0.4707±0.0000  0.4652±0.0000  \n",
      "Skala_mGGA     0.6819±0.0000  0.4925±0.0000  0.4863±0.0000  \n",
      "\n",
      "xc_energy_mae_mEh:\n",
      "Model          Loss_0         Loss_1         Loss_2         \n",
      "------------------------------------------------------------\n",
      "EGXC           1.81±0.00      0.7251±0.0000  0.6322±0.0000  \n",
      "NNmGGA         1.67±0.00      1.66±0.00      1.06±0.00      1.33±0.00      \n",
      "Skala_mGGA     1.10±0.00      1.85±0.00      1.54±0.00      1.07±0.00      \n"
     ]
    }
   ],
   "source": [
    "# Aggregate B3LYP metrics\n",
    "from aggregate_metrics import aggregate_b3lyp_model_loss_combinations\n",
    "\n",
    "# Build model batches from configuration\n",
    "# Structure: (model_name, loss_term, label, checkpoint_id)\n",
    "b3lyp_model_batches = []\n",
    "\n",
    "# Map loss terms to display labels\n",
    "loss_term_labels = {\n",
    "    'baseline': r'$E + \\rho$',\n",
    "    'gradient': r'$+ \\nabla$ (ours)',\n",
    "    'grad': r'$+ \\nabla$ (ours)',\n",
    "    'grad_and_hessian': r'$+ \\nabla + H$ (ours)',\n",
    "    'grad_and_hess': r'$+ \\nabla + H$ (ours)',\n",
    "    'hessian': r'$+ H$ (ours)',\n",
    "}\n",
    "\n",
    "for model_name, loss_terms in b3lyp_checkpoints.items():\n",
    "    batch = []\n",
    "    for loss_term, checkpoint_id in loss_terms.items():\n",
    "        label = loss_term_labels.get(loss_term, loss_term)\n",
    "        batch.append((model_name, loss_term, label, checkpoint_id))\n",
    "    b3lyp_model_batches.append(batch)\n",
    "\n",
    "# Aggregate all metrics (use CSV for better precision)\n",
    "all_means_b3lyp, all_stds_b3lyp = aggregate_b3lyp_model_loss_combinations(\n",
    "    checkpoint_root='../evaluations/1000',\n",
    "    model_batches=b3lyp_model_batches,\n",
    "    use_csv=True,  # Use results.csv for better precision\n",
    ")\n",
    "\n",
    "# Display ALL available metrics\n",
    "print('=== All Available B3LYP Metrics ===\\n')\n",
    "print(f'Available metrics: {list(all_means_b3lyp.keys())}\\n')\n",
    "\n",
    "# Print all metrics in a formatted way\n",
    "for metric_name in sorted(all_means_b3lyp.keys()):\n",
    "    print(f'\\n{metric_name}:')\n",
    "    # Get model names and number of loss terms\n",
    "    model_names = list(all_means_b3lyp[metric_name].keys())\n",
    "    num_losses = len(all_means_b3lyp[metric_name][model_names[0]]) if model_names else 0\n",
    "\n",
    "    # Print header\n",
    "    header = f'{\"Model\":<15}'\n",
    "    for i in range(num_losses):\n",
    "        header += f'{\"Loss_\" + str(i):<15}'\n",
    "    print(header)\n",
    "    print('-' * (15 + 15 * num_losses))\n",
    "\n",
    "    # Print data for each model\n",
    "    for model in model_names:\n",
    "        values = all_means_b3lyp[metric_name][model]\n",
    "        stds = all_stds_b3lyp[metric_name][model]\n",
    "\n",
    "        # Format with appropriate precision based on magnitude\n",
    "        formatted_vals = []\n",
    "        for v, s in zip(values, stds):\n",
    "            if abs(v) < 0.01:\n",
    "                formatted_vals.append(f'{v:.2e}±{s:.2e}')\n",
    "            elif abs(v) < 1:\n",
    "                formatted_vals.append(f'{v:.4f}±{s:.4f}')\n",
    "            else:\n",
    "                formatted_vals.append(f'{v:.2f}±{s:.2f}')\n",
    "\n",
    "        row = f'{model:<15}'\n",
    "        for val in formatted_vals:\n",
    "            row += f'{val:<15}'\n",
    "        print(row)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 255,
   "id": "6bf6f252",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "B3LYP data formatted for plotting!\n",
      "\n",
      "Available plot keys: ['E_tot', 'E_rho', 'E_xc', 'Delta_eps_HL', 'mu_rho', 'E_C', 'RIC', 'E_b3lyp']\n",
      "Models: ['NNmGGA', 'Skala-mGGA', 'EG-XC']\n",
      "Loss configurations: ['$E + \\\\rho$', '$+ \\\\nabla$ (ours)', '$+ \\\\nabla + H$ (ours)']\n",
      "\n",
      "Row 1 metrics: ['E_tot', 'E_rho', 'E_xc', 'Delta_eps_HL']\n",
      "Row 2 metrics: ['mu_rho', 'E_C', 'RIC', 'E_b3lyp']\n",
      "\n",
      "Example - E_tot data:\n",
      "  NNmGGA: [2.0242228513669716, 1.5748950978571656, 1.4367517331262434]\n",
      "  Skala-mGGA: [0.8931897276024756, 2.0210517659900233, 1.6944043812693508]\n",
      "  EG-XC: [1.1730132721992277, 0.4482920675027344, 0.5410738658744094]\n"
     ]
    }
   ],
   "source": [
    "# Format B3LYP data for plotting\n",
    "# Extract model display names from aggregated results and reorder: NNmGGA, Skala-mGGA, EG-XC\n",
    "# Map internal model names to display names\n",
    "model_display_name_map = {\n",
    "    'Skala_mGGA': 'Skala-mGGA',\n",
    "    'NNmGGA': 'NNmGGA',\n",
    "    'EGXC': 'EG-XC',\n",
    "}\n",
    "\n",
    "if all_means_b3lyp:\n",
    "    available_models = list(all_means_b3lyp[list(all_means_b3lyp.keys())[0]].keys())\n",
    "    # Reorder to: NNmGGA, Skala-mGGA, EG-XC\n",
    "    models_b3lyp_internal = ['NNmGGA', 'Skala_mGGA', 'EGXC']\n",
    "    # Ensure all models exist in the data\n",
    "    models_b3lyp_internal = [m for m in models_b3lyp_internal if m in available_models]\n",
    "    # Convert to display names\n",
    "    models_b3lyp = [model_display_name_map.get(m, m) for m in models_b3lyp_internal]\n",
    "else:\n",
    "    models_b3lyp = ['NNmGGA', 'Skala-mGGA', 'EG-XC']\n",
    "\n",
    "# Extract loss labels from all batches (collect all unique labels in specific order)\n",
    "losses_b3lyp = []\n",
    "if b3lyp_model_batches:\n",
    "    # Define the desired order for loss labels: baseline, +∇, +∇+H\n",
    "    desired_order = [\n",
    "        r'$E_\\mathrm{tot}$',\n",
    "        r'$E + \\rho$',\n",
    "        r'$+ \\nabla$ (ours)',\n",
    "        r'$+ \\nabla + H$ (ours)',\n",
    "    ]\n",
    "\n",
    "    # Collect all unique labels from all batches\n",
    "    all_labels = set()\n",
    "    for batch in b3lyp_model_batches:\n",
    "        for item in batch:\n",
    "            all_labels.add(item[2])  # label is at index 2\n",
    "\n",
    "    # Build losses_b3lyp in the desired order, only including labels that exist\n",
    "    for label in desired_order:\n",
    "        if label in all_labels:\n",
    "            losses_b3lyp.append(label)\n",
    "\n",
    "# Map summary.txt metric names to plot keys\n",
    "metric_name_map = {\n",
    "    'total_energy_mae_mEh': 'E_tot',\n",
    "    'mean_field_mae_mEh': 'E_rho',\n",
    "    'xc_energy_mae_mEh': 'E_xc',\n",
    "    'homo_lumo_gap_mae_mEh': 'Delta_eps_HL',\n",
    "    'dipole_difference_au': 'mu_rho',\n",
    "    'coulomb_energy_mae_mEh': 'E_C',\n",
    "    'relative_cycle_count': 'RIC',  # Relative iterations till convergence of learned functional\n",
    "    'ref_total_energy_on_model_dm_mEh': 'E_b3lyp',\n",
    "}\n",
    "\n",
    "# Create data and errors dicts in the format expected by plotting code\n",
    "data_b3lyp = {}\n",
    "errors_b3lyp = {}\n",
    "\n",
    "# Build mapping from model internal name to its batch labels (in original order)\n",
    "model_batch_labels = {}\n",
    "for batch_idx, batch in enumerate(b3lyp_model_batches):\n",
    "    model_name = batch[0][0]\n",
    "    if '_b3lyp' in model_name:\n",
    "        internal_name = model_name.split('_b3lyp')[0]\n",
    "    else:\n",
    "        internal_name = model_name.split('_')[0]\n",
    "    # Store labels in the order they appear in the batch (using internal name)\n",
    "    model_batch_labels[internal_name] = [item[2] for item in batch]\n",
    "\n",
    "# Reorder data arrays to match losses_b3lyp order\n",
    "# Also store which label corresponds to each position in reordered data for each model\n",
    "model_data_labels = {}  # model_display_name -> [label_for_idx_0, label_for_idx_1, ...]\n",
    "\n",
    "for summary_metric, plot_key in metric_name_map.items():\n",
    "    if summary_metric in all_means_b3lyp:\n",
    "        data_b3lyp[plot_key] = {}\n",
    "        errors_b3lyp[plot_key] = {}\n",
    "\n",
    "        # The aggregated results use internal model names as keys (e.g., 'EGXC', 'NNmGGA', 'Skala_mGGA')\n",
    "        # Convert display names back to internal names for data lookup\n",
    "        reverse_display_map = {v: k for k, v in model_display_name_map.items()}\n",
    "\n",
    "        for model_display_name in models_b3lyp:\n",
    "            # Get internal name for data lookup\n",
    "            model_internal_name = reverse_display_map.get(\n",
    "                model_display_name, model_display_name\n",
    "            )\n",
    "\n",
    "            if model_internal_name in all_means_b3lyp[summary_metric]:\n",
    "                original_values = all_means_b3lyp[summary_metric][model_internal_name]\n",
    "                original_stds = all_stds_b3lyp[summary_metric][model_internal_name]\n",
    "\n",
    "                # Get the original order of labels for this model (use internal name)\n",
    "                original_labels = model_batch_labels.get(model_internal_name, [])\n",
    "\n",
    "                # Create mapping: desired_label -> index in original array\n",
    "                label_to_idx = {label: idx for idx, label in enumerate(original_labels)}\n",
    "\n",
    "                # Reorder values and stds to match losses_b3lyp order\n",
    "                reordered_values = []\n",
    "                reordered_stds = []\n",
    "                reordered_labels = []  # Track which label each value corresponds to\n",
    "                for desired_label in losses_b3lyp:\n",
    "                    if desired_label in label_to_idx:\n",
    "                        idx = label_to_idx[desired_label]\n",
    "                        if idx < len(original_values):\n",
    "                            reordered_values.append(original_values[idx])\n",
    "                            reordered_stds.append(original_stds[idx])\n",
    "                            reordered_labels.append(desired_label)\n",
    "\n",
    "                data_b3lyp[plot_key][model_display_name] = reordered_values\n",
    "                errors_b3lyp[plot_key][model_display_name] = reordered_stds\n",
    "\n",
    "                # Store labels for this model (only need to do once, so check if already stored)\n",
    "                if model_display_name not in model_data_labels:\n",
    "                    model_data_labels[model_display_name] = reordered_labels\n",
    "\n",
    "# Add metric labels and units for plotting\n",
    "metric_labels = {\n",
    "    'E_tot': r'$E_\\mathrm{tot}$',\n",
    "    'E_rho': r'$E_\\rho$',\n",
    "    'E_xc': r'$E_\\mathrm{xc}$',\n",
    "    'Delta_eps_HL': r'$\\Delta \\varepsilon_\\mathrm{HL}$',\n",
    "    'mu_rho': r'Dipole $\\mu_\\rho$',\n",
    "    'E_C': r'$E_\\mathrm{C}$',\n",
    "    'RIC': r'Rel. Iter. Count (RIC)',\n",
    "    'E_b3lyp': r'$E_\\mathrm{B3LYP}$',\n",
    "}\n",
    "\n",
    "metric_units = {\n",
    "    'E_tot': '(MAE mEh)',\n",
    "    'E_rho': '(MAE mEh)',\n",
    "    'E_xc': '(MAE mEh)',\n",
    "    'Delta_eps_HL': '(MAE mEh)',\n",
    "    'mu_rho': '(MAE)',\n",
    "    'E_C': '(MAE mEh)',\n",
    "    'RIC': '',\n",
    "    'E_b3lyp': '(MAE mEh)',\n",
    "}\n",
    "\n",
    "# Define the layout\n",
    "row1_metrics = ['E_tot', 'E_rho', 'E_xc', 'Delta_eps_HL']\n",
    "row2_metrics = ['mu_rho', 'E_C', 'RIC', 'E_b3lyp']\n",
    "all_metrics_b3lyp = row1_metrics + row2_metrics\n",
    "\n",
    "print('B3LYP data formatted for plotting!')\n",
    "print(f'\\nAvailable plot keys: {list(data_b3lyp.keys())}')\n",
    "print(f'Models: {models_b3lyp}')\n",
    "print(f'Loss configurations: {losses_b3lyp}')\n",
    "print(f'\\nRow 1 metrics: {row1_metrics}')\n",
    "print(f'Row 2 metrics: {row2_metrics}')\n",
    "print('\\nExample - E_tot data:')\n",
    "for model in models_b3lyp:\n",
    "    if model in data_b3lyp.get('E_tot', {}):\n",
    "        print(f'  {model}: {data_b3lyp[\"E_tot\"][model]}')\n",
    "\n",
    "# Loss colors matching convention: colors[2], colors[1], colors[0], colors[3] for E, E+ρ (baseline), E+ρ+∇, E+ρ+∇+H\n",
    "# Map loss labels to colors based on convention from tabel_visualization.ipynb\n",
    "loss_color_map = {\n",
    "    r'$E_\\mathrm{tot}$': colors[1],  # E+ρ (baseline)\n",
    "    r'$E + \\rho$': colors[1],  # E+ρ (baseline)\n",
    "    r'$+ \\nabla$ (ours)': colors[0],  # E+ρ+∇\n",
    "    r'$+ \\nabla + H$ (ours)': colors[3],  # E+ρ+∇+H\n",
    "    r'$+ H$ (ours)': colors[\n",
    "        4\n",
    "    ],  # +H (hessian only, kept for compatibility but not plotted)\n",
    "}\n",
    "# Assign colors based on loss labels\n",
    "loss_colors_b3lyp = [\n",
    "    loss_color_map.get(label, colors[i % len(colors)])\n",
    "    for i, label in enumerate(losses_b3lyp)\n",
    "]\n",
    "\n",
    "# Map labels for legend display\n",
    "legend_label_map = {\n",
    "    r'$+ \\nabla + H$ (ours)': r'$+ H$ (ours)',\n",
    "    r'$+ \\nabla$ (ours)': r'$+ \\nabla$ (ours)',\n",
    "    r'$E + \\rho$': r'$E + \\rho$',\n",
    "    r'$E_\\mathrm{tot}$': r'$E_\\mathrm{tot}$',\n",
    "}\n",
    "loss_labels_b3lyp = [legend_label_map.get(label, label) for label in losses_b3lyp]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 256,
   "id": "2902960b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1400x450 with 8 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ===== Configuration for General vs Model Specific Overlay =====\n",
    "# Set USE_OVERLAY to True to overlay \"general\" and \"model specific\" bars\n",
    "USE_OVERLAY = False  # Set to True to enable overlay\n",
    "\n",
    "# If USE_OVERLAY is True, specify which data should be \"general\" vs \"model specific\"\n",
    "# Uncomment and modify these to point to your actual data sources\n",
    "if USE_OVERLAY:\n",
    "    import copy\n",
    "\n",
    "    # Example: Create deep copies of the current data structure\n",
    "    # Replace these with your actual \"general\" and \"model specific\" data sources\n",
    "    data_general_b3lyp = copy.deepcopy(\n",
    "        data_b3lyp\n",
    "    )  # Modify this to point to your general data\n",
    "    data_model_specific_b3lyp = copy.deepcopy(\n",
    "        data_b3lyp\n",
    "    )  # Modify this to point to your model-specific data\n",
    "    errors_general_b3lyp = copy.deepcopy(\n",
    "        errors_b3lyp\n",
    "    )  # Modify this to point to your general error data\n",
    "    errors_model_specific_b3lyp = copy.deepcopy(\n",
    "        errors_b3lyp\n",
    "    )  # Modify this to point to your model-specific error data\n",
    "else:\n",
    "    # Placeholder variables when overlay is disabled (to avoid errors)\n",
    "    data_general_b3lyp = {}\n",
    "    data_model_specific_b3lyp = {}\n",
    "    errors_general_b3lyp = {}\n",
    "    errors_model_specific_b3lyp = {}\n",
    "\n",
    "# ===== B3LYP Combined Plot =====\n",
    "fig_b3lyp, axes_b3lyp = plt.subplots(\n",
    "    2, 4, figsize=(14, 4.5), gridspec_kw={'hspace': 0.06, 'wspace': 0.3}\n",
    ")\n",
    "x_b3lyp = np.arange(len(models_b3lyp))\n",
    "width_b3lyp = 0.20\n",
    "num_losses_b3lyp = len(losses_b3lyp)\n",
    "offsets_b3lyp = [\n",
    "    width_b3lyp * (i - (num_losses_b3lyp - 1) / 2) for i in range(num_losses_b3lyp)\n",
    "]\n",
    "\n",
    "ALPHA = 1.0\n",
    "\n",
    "for idx, metric in enumerate(all_metrics_b3lyp):\n",
    "    row = idx // 4\n",
    "    col = idx % 4\n",
    "    ax = axes_b3lyp[row, col]\n",
    "\n",
    "    if USE_OVERLAY:\n",
    "        # First plot \"general\" bars in background with white background and hatching\n",
    "        for loss_idx in range(num_losses_b3lyp):\n",
    "            desired_label = losses_b3lyp[loss_idx]\n",
    "\n",
    "            for model_idx, model in enumerate(models_b3lyp):\n",
    "                if model in data_general_b3lyp.get(metric, {}):\n",
    "                    vals = data_general_b3lyp[metric][model]\n",
    "                    errs = errors_general_b3lyp[metric][model]\n",
    "\n",
    "                    model_labels = model_data_labels.get(model, [])\n",
    "\n",
    "                    if desired_label in model_labels:\n",
    "                        data_idx = model_labels.index(desired_label)\n",
    "                        if data_idx < len(vals):\n",
    "                            val = vals[data_idx]\n",
    "                            color = loss_color_map.get(\n",
    "                                desired_label, colors[loss_idx % len(colors)]\n",
    "                            )\n",
    "\n",
    "                            ax.bar(\n",
    "                                model_idx + offsets_b3lyp[loss_idx],\n",
    "                                val,\n",
    "                                width_b3lyp,\n",
    "                                color='white',\n",
    "                                hatch='///',  # White tilted stripes\n",
    "                                alpha=ALPHA,\n",
    "                                edgecolor=color,\n",
    "                                linewidth=0.5,\n",
    "                                zorder=1,\n",
    "                            )\n",
    "\n",
    "        # Then plot \"model specific\" bars on top with solid colors\n",
    "        for loss_idx in range(num_losses_b3lyp):\n",
    "            desired_label = losses_b3lyp[loss_idx]\n",
    "\n",
    "            for model_idx, model in enumerate(models_b3lyp):\n",
    "                if model in data_model_specific_b3lyp.get(metric, {}):\n",
    "                    vals = data_model_specific_b3lyp[metric][model]\n",
    "                    errs = errors_model_specific_b3lyp[metric][model]\n",
    "\n",
    "                    model_labels = model_data_labels.get(model, [])\n",
    "\n",
    "                    if desired_label in model_labels:\n",
    "                        data_idx = model_labels.index(desired_label)\n",
    "                        if data_idx < len(vals):\n",
    "                            val = vals[data_idx]\n",
    "                            err = errs[data_idx]\n",
    "                            color = loss_color_map.get(\n",
    "                                desired_label, colors[loss_idx % len(colors)]\n",
    "                            )\n",
    "\n",
    "                            ax.bar(\n",
    "                                model_idx + offsets_b3lyp[loss_idx],\n",
    "                                val,\n",
    "                                width_b3lyp,\n",
    "                                label=loss_labels_b3lyp[loss_idx]\n",
    "                                if (idx == 0 and model_idx == 0)\n",
    "                                else None,\n",
    "                                color=color,\n",
    "                                edgecolor='black',\n",
    "                                linewidth=0.5,\n",
    "                                zorder=3,\n",
    "                            )\n",
    "\n",
    "                            ax.errorbar(\n",
    "                                model_idx + offsets_b3lyp[loss_idx],\n",
    "                                val,\n",
    "                                yerr=err,\n",
    "                                fmt='none',\n",
    "                                color='0.2',\n",
    "                                capsize=1.5,\n",
    "                                capthick=0.6,\n",
    "                                linewidth=1,\n",
    "                                zorder=4,\n",
    "                            )\n",
    "    else:\n",
    "        # Original single-bar plotting (no overlay)\n",
    "        for loss_idx in range(num_losses_b3lyp):\n",
    "            desired_label = losses_b3lyp[loss_idx]\n",
    "\n",
    "            for model_idx, model in enumerate(models_b3lyp):\n",
    "                if model in data_b3lyp[metric]:\n",
    "                    vals = data_b3lyp[metric][model]\n",
    "                    errs = errors_b3lyp[metric][model]\n",
    "\n",
    "                    model_labels = model_data_labels.get(model, [])\n",
    "\n",
    "                    if desired_label in model_labels:\n",
    "                        data_idx = model_labels.index(desired_label)\n",
    "                        if data_idx < len(vals):\n",
    "                            val = vals[data_idx]\n",
    "                            err = errs[data_idx]\n",
    "                            color = loss_color_map.get(\n",
    "                                desired_label, colors[loss_idx % len(colors)]\n",
    "                            )\n",
    "\n",
    "                            ax.bar(\n",
    "                                model_idx + offsets_b3lyp[loss_idx],\n",
    "                                val,\n",
    "                                width_b3lyp,\n",
    "                                label=loss_labels_b3lyp[loss_idx]\n",
    "                                if (idx == 0 and model_idx == 0)\n",
    "                                else None,\n",
    "                                color=color,\n",
    "                                edgecolor='black',\n",
    "                                linewidth=0.5,\n",
    "                                zorder=3,\n",
    "                            )\n",
    "\n",
    "                            ax.errorbar(\n",
    "                                model_idx + offsets_b3lyp[loss_idx],\n",
    "                                val,\n",
    "                                yerr=err,\n",
    "                                fmt='none',\n",
    "                                color='0.2',\n",
    "                                capsize=1.5,\n",
    "                                capthick=0.6,\n",
    "                                linewidth=1,\n",
    "                                zorder=4,\n",
    "                            )\n",
    "\n",
    "    ax.set_ylabel(f'{metric_labels[metric]} {metric_units[metric]}')\n",
    "    ax.set_xticks(x_b3lyp)\n",
    "\n",
    "    # Only show x labels on bottom row (model names)\n",
    "    if row == 0:\n",
    "        ax.set_xticklabels([])\n",
    "    else:\n",
    "        ax.set_xticklabels(models_b3lyp)\n",
    "\n",
    "    # Use linear scale for all metrics\n",
    "    ax.set_yscale('linear')\n",
    "\n",
    "    # Set y-axis limit for RIC to start at 1\n",
    "    if metric == 'RIC':\n",
    "        ax.set_ylim(bottom=1.0, top=1.16)\n",
    "\n",
    "    ax.grid(True, axis='y', alpha=0.8, linewidth=0.5, zorder=0)\n",
    "    ax.grid(False, axis='x')\n",
    "    ax.set_axisbelow(True)\n",
    "\n",
    "    ax.yaxis.set_tick_params(pad=-1)\n",
    "    for label in ax.get_yticklabels():\n",
    "        label.set_ha('right')\n",
    "\n",
    "# Create legend\n",
    "from matplotlib.patches import Patch\n",
    "\n",
    "if USE_OVERLAY:\n",
    "    # Create legend with \"general\" and \"model specific\" labels\n",
    "    # Get handles and labels from the first axis (loss terms)\n",
    "    handles_b3lyp, labels_b3lyp_legend = axes_b3lyp[0, 0].get_legend_handles_labels()\n",
    "\n",
    "    # Create patches for \"general\" and \"model specific\"\n",
    "    general_patch = Patch(\n",
    "        facecolor='white', edgecolor='black', hatch='///', label='General'\n",
    "    )\n",
    "    model_specific_patch = Patch(\n",
    "        facecolor='gray', edgecolor='black', label='Model specific'\n",
    "    )\n",
    "\n",
    "    # Combine all legend elements\n",
    "    all_handles = [general_patch, model_specific_patch] + handles_b3lyp\n",
    "    all_labels = ['General', 'Model specific'] + labels_b3lyp_legend\n",
    "\n",
    "    fig_b3lyp.legend(\n",
    "        all_handles,\n",
    "        all_labels,\n",
    "        loc='upper center',\n",
    "        bbox_to_anchor=(0.5, 0.96),\n",
    "        ncol=len(all_handles),\n",
    "        frameon=False,\n",
    "        columnspacing=1.5,\n",
    "        handletextpad=0.4,\n",
    "    )\n",
    "else:\n",
    "    # Original legend (loss terms only)\n",
    "    handles_b3lyp, labels_b3lyp_legend = axes_b3lyp[0, 0].get_legend_handles_labels()\n",
    "\n",
    "    fig_b3lyp.legend(\n",
    "        handles_b3lyp,\n",
    "        labels_b3lyp_legend,\n",
    "        loc='upper center',\n",
    "        bbox_to_anchor=(0.5, 0.96),\n",
    "        ncol=num_losses_b3lyp,\n",
    "        frameon=False,\n",
    "        columnspacing=1.5,\n",
    "        handletextpad=0.4,\n",
    "    )\n",
    "\n",
    "plt.savefig('b3lyp_qm7_comparison.pdf', bbox_inches='tight')\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
