{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b2fca8a5",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import scienceplots  # noqa\n",
    "import seaborn as sns\n",
    "from matplotlib.patches import Patch\n",
    "\n",
    "plt.style.use(['science', 'grid'])\n",
    "plt.rcParams.update({'font.size': 16})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "40a384db",
   "metadata": {},
   "outputs": [],
   "source": [
    "from dataclasses import dataclass\n",
    "\n",
    "\n",
    "@dataclass\n",
    "class Bar:\n",
    "    value: float\n",
    "    model: str\n",
    "    loss_term: str  # E, E+ρ, E+ρ+∇, E+ρ+∇+H\n",
    "\n",
    "\n",
    "scf_acceleration_data = [\n",
    "    Bar(\n",
    "        0.637004, model='SCAN', loss_term='SCAN'\n",
    "    ),  # evaluations/1000/SCF_acceleration_baseline/summary.txt\n",
    "    Bar(\n",
    "        0.685672, model='NNmGGA', loss_term=r'$E + \\rho$'\n",
    "    ),  # evaluations/1000/NNmGGA_b3lyp_qm7/baseline/summary.txt\n",
    "    Bar(\n",
    "        0.454744, model='NNmGGA', loss_term=r'$+ \\nabla$'\n",
    "    ),  # evaluations/1000/NNmGGA_b3lyp_qm7/grad/NNmGGA2_b3lyp_qm7_58/summary.txt\n",
    "    Bar(\n",
    "        0.465199,\n",
    "        model='NNmGGA',\n",
    "        loss_term=r'$+ H$',  # TODO: redo this with _correction_9  (same grad weight as above)\n",
    "    ),  # evaluations/1000_correction/NNmGGA2_b3lyp_qm7_correction/NNmGGA2_b3lyp_qm7_correction_11/summary.txt\n",
    "    # Bar(\n",
    "    #     0.5962962963,\n",
    "    #     model='NNmGGA (fused)',\n",
    "    #     loss_term=r'$E + \\rho$'\n",
    "    # ),\n",
    "    # Bar(\n",
    "    #     0.4037037037,\n",
    "    #     model='NNmGGA (fused)',\n",
    "    #     loss_term=r'$+ \\nabla$'\n",
    "    # ),\n",
    "    Bar(0.681888, model='Skala-mGGA', loss_term=r'$E + \\rho$'),\n",
    "    Bar(0.492539, model='Skala-mGGA', loss_term=r'$+ \\nabla$'),\n",
    "    Bar(0.486281, model='Skala-mGGA', loss_term=r'$+ H$'),\n",
    "    # EG-XC\n",
    "    Bar(0.686102, model='EG-XC', loss_term=r'$E + \\rho$'),\n",
    "    Bar(0.503412, model='EG-XC', loss_term=r'$+ \\nabla$'),\n",
    "    Bar(0.517405, model='EG-XC', loss_term=r'$+ H$'),\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "fc8ed132",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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g9bW1qTJpt/vR3Nzs/ZakiTtkv/g4GDKSe2JiIiIaO3I+qX5pe5wFwRh2TvgZVlGVVWV7tK+IAhobW1FfX09JElK2otK69+Xv/g4lMuXct2MFRG23YWv/Nev5boZRLRO5Ov5kOfCjSdn84iulCiKSRNNl8vFntANTrl8CV/6k3tz3YwV+eq//jLXTaAMYQJAa0G+ng95Ltx48i4RJSJay5gAEBEZl/UxokREREREABNRIiIiIsoRJqJERERElBNMRImIiIgoJ5iIEhEREVFOMBElIiIiopxgIkpEREREOcFElIiIiIhygokoEREREeUEE1EiIiIiygkmokRERESUE0xEiYiIiCgnmIgSERERUU4U5boB2TQ7O4vh4WHdMqvViu3bt+Ott97Cz3/+84T33H///QCAX/ziF7hx44aurLy8HLfeeiuuXr2KS5cu6co2b96Me+65B7OzsxgdHU2o97777kNxcTF+/etfY2JiQlf2jne8A9u2bcP169chy7Ku7Oabb8bdd98NABgZGUFRkf4Qvve978WmTZswPj6Oa9eu6cq2bt2Kd77znZicnMSvfvUrXVlRURF27doFADh//jxmZmZ05e95z3uwZcsW/Pa3v8WVK1d0ZbfeeivKy8tx48YN/OIXv9CVWSwW2O12AMDPf/5zvPXWW7pyURRRVlaGy5cv43e/+52urLS0FHfffTemp6fxyiuvYLGKigoUFhbi1VdfxRtvvIHrE2/glUuvAwDutN6CWzffjNibb+O319/QvW9TSRHevVUAAG193d+6zYqbigtx6doklBtTurKtpZuwtXQTJt+axvjr+uNWUlSAe+4om/9bf3cNs3Oqrnz77aW45aZi/D72Jq69od8Pb771NgAsuw9/9rOf4e2339bXu307rFYrXnvtNfz+97/Xld1yyy3YtGkTpqamku5Dm82GgoIC/PKXv8Sbb76pK3vXu96F2267Da+//jp+85vfJNR77733Ym5uDtFoNKHenTt3oqSkBBcvXoSiKLqyO++8E3fccQdisRjGxsZ0ZTfddBN27NgBYD6+VVW/D+Px/Zvf/Aavv64/drfffjvuuusuvPHGG3j11Vd1ZYWFhaioqAAAXLhwAVNT+uN69913o7S0FL///e/x2muv6crSPUcsjMN3lm2G9ZabcO2Nt/D7mH7/3nJTEbbfLmB2bg4//931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PHgQwPw0U/ExopIkIRwOr6otRERERJQdafV/x3sgR0dHAQAVFRUZebqSIAioqanBoUOH0NbWhlgshtOnT2tjQReSZRkTExOw2WwJZYt7VYmIiIho7VqyR3RychLPPfccnnvuOW2ZKIqoqalBTU0NRFHE+fPnddM5pRIIBOB0OjEwMIDjx4/D6XTqpmdqa2sDAFRXV8Pj8aC1tTVpsun3+3U9sJIkafOTDg4OwuVyLf9XExEREVHOpewRnZiYQF1dndbLGL9rfsuWLQCA8fFxtLe3IxQKoaqqatkNNTQ0LDl2UxCEZe+Al2UZVVVVummeBEFAa2sr6uvrIUkS3G73sm0hIiIiotxL2SP67LPPoqKiAi+//DJefvll7Ny5E11dXZicnITX64XT6YSiKOjt7V3VHKLpEEUxaaLpcrlw7tw5PmueiIiIKI+k7BE9e/as7jGfLS0t2LNnDwKBACRJQm9vb8qnJBERERERLSdlIirLMsrLy7XX8ZuSzpw5wwSUiIiIiFYt5aX50tLShGUWi4VJKBERERFlRMpENB8nZiUiIiKi/JHy0nwsFsOHP/xh3TJVVROWxb300kuZbRkRERERrWspE9HVPi2JiIiIiGgpKRPRxx57LJvtICIiIqINxvCz5omIiIiIMomJKBERERHlBBNRIiIiIsoJJqJERERElBNMRImIiIgoJ5iIEhEREVFOMBElIiIiopxgIkpEREREOcFElIiIiIhygokoEREREeUEE1EiIiIiygkmokRERESUE0xEiYiIiCgnmIgSERERUU4wESUiIiKinMibRDQajcLpdMLpdEKW5YRyr9ebg1YRERER0UqtqUTU6/XC4XDA6XQiHA7ryjweD3w+HxoaGhKSzlAoBEmSstlUIiIiIlqlNZOINjU1AQAikQi6u7vh8XgQjUYBAIqiQJZlSJKE2trahCS1r68PLpcr620mIiIiopVbM4nowMAAmpubAQCiKKKxsRHBYHDZ9/n9frS0tJjdPCIiIiLKMIuqqmquGxEOh+H1enHu3LmUy+rq6tDY2IhYLIZQKITu7m7IsoxTp04ZSkR//OMfQ1VVlJSUmPZ3LPbaa69h06ZNGa934SGzWCwZr//GjRu44447Ml6v2a689nuUbSrOdTNW5HexG9hUers5lasqYEKcAMDbyhVsveUWU+o2kzIzja133mlK3fka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",
      "text/plain": [
       "<Figure size 700x350 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib.ticker import PercentFormatter\n",
    "\n",
    "sns.set_theme(style='whitegrid')\n",
    "palette = sns.color_palette('deep')\n",
    "\n",
    "# Define loss term colors (matching tabel_visualization.ipynb style)\n",
    "# loss_colors = [colors[2], colors[1], colors[0], colors[3]]  # E, E+ρ, E+ρ+∇, E+ρ+∇+H\n",
    "loss_colors = {\n",
    "    'SCAN': 'gray',\n",
    "    r'$E + \\rho$': palette[1],  # Orange\n",
    "    r'$+ \\nabla$': palette[0],  # Blue\n",
    "    r'$+ H$': palette[3],  # Red\n",
    "}\n",
    "\n",
    "# Group data by model\n",
    "models = ['SCAN', 'NNmGGA', 'Skala-mGGA', 'EG-XC']\n",
    "model_data = {model: [] for model in models}\n",
    "\n",
    "for bar in scf_acceleration_data:\n",
    "    model_data[bar.model].append(bar)\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(7, 3.5))\n",
    "\n",
    "# Bar width and positions\n",
    "bar_width = 0.2\n",
    "# SCAN (1 bar), NNmGGA (3 bars span 0.6), Skala (3 bars span 0.6), EG-XC (3 bars span 0.6)\n",
    "# Need spacing that accounts for bar group widths\n",
    "# Even spacing: gap of 0.3 between groups\n",
    "# SCAN bar at 0 spans -0.1 to 0.1\n",
    "# NNmGGA bars centered at 0.6 span 0.3 to 0.9\n",
    "# Skala bars centered at 1.3 span 1.0 to 1.6\n",
    "# EG-XC bars centered at 2.0 span 1.7 to 2.3\n",
    "model_positions = [0, 0.6, 1.3, 2.0]  # Even spacing without overlap\n",
    "\n",
    "# Plot bars for each model group\n",
    "for i, model in enumerate(models):\n",
    "    bars = model_data[model]\n",
    "\n",
    "    if model == 'SCAN':\n",
    "        # SCAN has only one bar, centered at its position\n",
    "        ax.bar(\n",
    "            model_positions[i],\n",
    "            bars[0].value,\n",
    "            bar_width,\n",
    "            color=loss_colors['SCAN'],\n",
    "            edgecolor='black',\n",
    "            linewidth=0.5,\n",
    "            alpha=0.8,\n",
    "        )\n",
    "    else:\n",
    "        # Learnable models have multiple bars grouped together\n",
    "        n_bars = len(bars)\n",
    "        # Center the bars around the model position\n",
    "        offsets = [(j - (n_bars - 1) / 2) * bar_width for j in range(n_bars)]\n",
    "\n",
    "        for j, bar in enumerate(bars):\n",
    "            color = loss_colors[bar.loss_term]\n",
    "            ax.bar(\n",
    "                model_positions[i] + offsets[j],\n",
    "                bar.value,\n",
    "                bar_width,\n",
    "                color=color,\n",
    "                edgecolor='black',\n",
    "                linewidth=0.5,\n",
    "                alpha=0.8,\n",
    "            )\n",
    "\n",
    "baseline = scf_acceleration_data[0].value  # SCAN baseline\n",
    "\n",
    "ax.axhline(y=baseline, color='k', linestyle='--', linewidth=1)\n",
    "# ax.text(\n",
    "#     0.05,\n",
    "#     baseline * 1.02,\n",
    "#     'slowdown',\n",
    "#     va='bottom',\n",
    "#     ha='left',\n",
    "#     color='k',\n",
    "#     fontsize=14,\n",
    "#     transform=ax.get_yaxis_transform(),\n",
    "# )\n",
    "# ax.text(\n",
    "#     0.05,\n",
    "#     baseline * 0.98,\n",
    "#     'speedup',\n",
    "#     va='top',\n",
    "#     ha='left',\n",
    "#     color='k',\n",
    "#     fontsize=14,\n",
    "#     transform=ax.get_yaxis_transform(),\n",
    "# )\n",
    "\n",
    "ax.set_ylabel('Relative Iteration Count (RIC)', labelpad=3)\n",
    "ax.set_xticks(model_positions)\n",
    "ax.set_xticklabels(models)\n",
    "ax.set_ylim(0, 0.71)\n",
    "ax.tick_params(axis='y', pad=-2)\n",
    "\n",
    "# Format y-axis as percentage\n",
    "ax.yaxis.set_major_formatter(PercentFormatter(1.0))\n",
    "\n",
    "# Create legend for loss terms\n",
    "legend_elements = [\n",
    "    Patch(facecolor=loss_colors['SCAN'], edgecolor='black', label='Baseline'),\n",
    "    Patch(facecolor=loss_colors[r'$E + \\rho$'], edgecolor='black', label=r'$E + \\rho$'),\n",
    "    Patch(\n",
    "        facecolor=loss_colors[r'$+ \\nabla$'],\n",
    "        edgecolor='black',\n",
    "        label=r'$+ \\nabla$ (ours)',\n",
    "    ),\n",
    "    Patch(facecolor=loss_colors[r'$+ H$'], edgecolor='black', label=r'$+ H$ (ours)'),\n",
    "]\n",
    "ax.legend(\n",
    "    handles=legend_elements,\n",
    "    ncol=4,\n",
    "    loc='upper center',\n",
    "    bbox_to_anchor=(0.5, 1.17),\n",
    "    columnspacing=1.5,\n",
    "    handletextpad=0.5,\n",
    "    frameon=False,\n",
    ")\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig('scf_acceleration.pdf')\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
}
