{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Grad-Cam Application on ShapEval\n",
    "Init"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/lib/python3.10/site-packages/tqdm/auto.py:21: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n",
      "/tmp/ipykernel_6507/3041713977.py:50: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  state_dict = torch.load(model_path)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "SimpleCNN(\n",
       "  (conv1): Conv2d(3, 16, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1))\n",
       "  (relu): ReLU()\n",
       "  (pool): MaxPool2d(kernel_size=2, stride=2, padding=0, dilation=1, ceil_mode=False)\n",
       "  (conv2): Conv2d(16, 32, kernel_size=(3, 3), stride=(1, 1), padding=(1, 1))\n",
       "  (fc1): Linear(in_features=10240, out_features=128, bias=True)\n",
       "  (fc2): Linear(in_features=128, out_features=2, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import shap\n",
    "import numpy as np\n",
    "from torchvision import transforms\n",
    "import cv2\n",
    "from PIL import Image\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from model import *\n",
    "\n",
    "    \n",
    "\n",
    "# Paths\n",
    "model_path = \"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results/T_model.pth\"\n",
    "image_path = \"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/data/grayscale_image.png\"\n",
    "\n",
    "# Define preprocess function for a single image\n",
    "def preprocess_image(image_path):\n",
    "    \"\"\"\n",
    "    Preprocesses a single image for model input.\n",
    "    \"\"\"\n",
    "    image = Image.open(image_path).convert(\"RGB\")  # Ensure 3 channels (RGB)\n",
    "    transform = transforms.Compose([\n",
    "        #transforms.Resize((20, 256)),  # Resize\n",
    "        transforms.ToTensor(),\n",
    "    ])\n",
    "    image_tensor = transform(image).unsqueeze(0)  # Add batch dimension\n",
    "    return image_tensor\n",
    "\n",
    "\n",
    "# Prediction function\n",
    "def predict(processed_image):\n",
    "    \"\"\"\n",
    "    Function to predict probabilities for SHAP.\n",
    "    \"\"\"\n",
    "    if isinstance(processed_image, np.ndarray):\n",
    "        # Convert numpy array to PyTorch tensor\n",
    "        processed_image = torch.from_numpy(processed_image).float()\n",
    "\n",
    "    # Perform inference\n",
    "    with torch.no_grad():\n",
    "        output = model(processed_image)  # Output from the model\n",
    "        probabilities = nn.Softmax(dim=1)(output)  # Apply softmax to get probabilities\n",
    "    \n",
    "    return probabilities.numpy()  # Convert to numpy array for SHAP\n",
    "\n",
    "# Instantiate and load the model\n",
    "model = SimpleCNN()\n",
    "state_dict = torch.load(model_path)\n",
    "model.load_state_dict(state_dict)\n",
    "model.eval()\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 1. Grad-CAM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/lib/python3.10/site-packages/torch/nn/modules/module.py:1827: FutureWarning: Using a non-full backward hook when the forward contains multiple autograd Nodes is deprecated and will be removed in future versions. This hook will be missing some grad_input. Please use register_full_backward_hook to get the documented behavior.\n",
      "  self._maybe_warn_non_full_backward_hook(args, result, grad_fn)\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAgMAAABRCAYAAABPAqWCAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAFcJJREFUeJzt3X1wFOUdB/Dv7l3eLpHEQJoGBSwkGBKYRB2hYzCEEkGbClYCCpWGYGEIaEtrKUWmkrRmqC3MyKu1RWyKCVRe0rEw7RCRjE6lLVBAC4VIG4QpEQGTAAaa5O7XP+52b3dvL1wCY8T9fmYed/d53ztyz2/3XlREREBERESOpfb2BIiIiKh3MRggIiJyOAYDREREDsdggIiIyOEYDBARETkcgwEiIiKHYzBARETkcAwGiIiIHI7BABERkcMxGCDqoZkzZ+KOO+7o7WnccBs3bkRmZiaioqKQlJTU29MxURQF5eXlvT0Noi8cBgN002lsbMRTTz2FoUOHwuPxwOPxICsrC/Pnz8d7773X29ML6+LFi6ioqEBOTg4SEhIQFxeH4cOHY9GiRThz5oxtm6lTp0JRFCxatMi2vL6+HoqiQFEUvPbaa7Z18vLyoCgKhg8ffs05Hjt2DDNnzsSQIUPwm9/8Br/+9a8jP8HrcOjQITzxxBMYMGAAYmJikJycjMLCQrz66qvwer2fyRyuR0tLC+bMmYOUlBTEx8dj7Nix+Mc//tHb0yKKmLu3J0DUHTt27MBjjz0Gt9uNb33rW8jJyYGqqjh27Bi2b9+Ol156CY2NjRg0aFBvT9XkP//5DwoLC3Hq1ClMmTIFc+bMQXR0NN577z288sorqK2tRUNDg6nNxYsX8cc//hF33HEHNm3ahJ///OdQFMW2/9jYWNTU1OCJJ54w5Z88eRLvvvsuYmNjI5pnfX09fD4fVq5cifT09J6dbDetX78ec+fORWpqKmbMmIGMjAxcunQJu3fvxpNPPommpiY8++yzn8lcesLn86GoqAiHDx/GwoUL0a9fP6xbtw4FBQU4cOAAMjIyenuKRNcmRDeJEydOSHx8vAwbNkzOnDkTUt7R0SErV66UU6dOddnP5cuXb8h8SkpKZNCgQdes19HRITk5OeLxeOSdd94JKW9tbZVnn302JH/Dhg0SFRUlb731lgCQ+vr6kDp79uwRAPLoo4+K2+2Wc+fOmcorKyslNTVVRo8eLdnZ2deca0VFhQAI6ed6fPrpp2HL9u7dKy6XS0aPHi0XL14MKd+3b5+8+uqr+jEAWbp06Q2b243w+9//XgDIli1b9LyPP/5YkpKSZNq0ab04M6LIMRigm8acOXMEgPz1r3+NuE1JSYnEx8fLiRMn5KGHHpKEhASZNGmSiIi8/fbbUlxcLAMGDJDo6Gi5/fbbZcGCBdLW1hbST21trWRnZ0tMTIxkZ2fL9u3bIw4GNm/eLACksrIy4nmLiIwbN06+/vWvi4jIsGHDZPbs2SF1tGCgqqpK4uPjZd26daby7Oxsefrpp2XMmDHXDAYGDRokAEzJuPCuXbtWsrKyJDo6WtLS0mTevHnS3Nxs6kMbZ//+/XL//fdLXFycfO973ws75oMPPihut1s+/PDDrh+MAOucTp48KWVlZTJ06FCJjY2V5ORkKS4ulsbGRlO79vZ2KS8vl/T0dImJiZHk5GTJy8uTXbt26XWamppk5syZctttt0l0dLR8+ctflokTJ4b0ZTVlyhRJTU0Vr9dryp8zZ454PB65evVqROdG1Jv4mQG6aezYsQPp6ekYNWpUt9p1dnZiwoQJ+NKXvoTly5dj8uTJAIAtW7agra0NZWVlWL16NSZMmIDVq1fj29/+tqn9rl27MHnyZCiKgmXLluGRRx5BaWkp9u/fH9H4b7zxBgBgxowZEc/5zJkz2LNnD6ZNmwYAmDZtGrZu3Yr29nbb+h6PB5MmTcKmTZv0vMOHD+PIkSOYPn16RGO++OKL+OY3vwkAeOmll7Bx40Y8+uijAIDy8nLMnz8f/fv3x4oVKzB58mS8/PLLGD9+PDo6Okz9XLhwAQ899BByc3Px4osvYuzYsbbjtbW1Yffu3cjPz8fAgQMjmqPVvn378O677+Lxxx/HqlWrMHfuXOzevRsFBQVoa2vT65WXl6OiogJjx47FmjVrsGTJEgwcOND0vv7kyZNRW1uL0tJSrFu3Dt/97ndx6dIlnDp1qss5HDx4EHfffTdU1fxyOnLkSLS1tYW8/UP0udTb0QhRJFpbWwWAPPLIIyFlzc3Ncu7cOT0Zr+xLSkoEgPz4xz8OaWd3B2DZsmWiKIrpSjU3N1fS0tKkpaVFz9u1a5cAiOjOwF133SWJiYnXrGe0fPlyiYuL02+dNzQ0CACpra011dPuDGzZskV27NghiqLob5MsXLhQBg8eLCIS0Z0BEZGlS5eGvE3w8ccfS3R0tIwfP9509btmzRoBIBs2bNDzxowZIwDkV7/61TXHOnz4sADo8s6BFSx3Buyew7179woA+d3vfqfn5eTkSFFRUdh+m5ubBYD88pe/jHgumvj4eJk1a1ZI/s6dOwWA/PnPf+52n0SfNd4ZoJvCxYsXAQAJCQkhZQUFBUhJSdHT2rVrQ+qUlZWF5MXFxen7n376Kc6fP4/77rsPIoKDBw8CAJqamnDo0CGUlJQgMTFRr//AAw8gKysr4rnfcsstEdXVVFdXo6ioSG+XkZGBe+65B9XV1WHbjB8/HsnJydi8eTNEBJs3b9bvLFyPN998E+3t7ViwYIHp6nf27Nno06cPdu7caaofExOD0tLSa/arPafdfWyMjM9hR0cHLly4gPT0dCQlJZmu+pOSknDkyBF88MEHYfuJjo5GfX09mpubuzWHK1euICYmJiRf+9DmlStXutUfUW9gMEA3BW3BuHz5ckjZyy+/jLq6urBfrXO73bj99ttD8k+dOoWZM2ciOTkZCQkJSElJwZgxYwAAra2tAIAPP/wQAGw/EX7nnXeajs+dO4ePPvpIT9pc+/Tpg0uXLkV6qvjXv/6FgwcPIi8vDydOnNBTQUEBduzYoS+iVlFRUZgyZQpqamrw9ttv4/Tp0xG/RdAV7TGwnm90dDQGDx6sl2tuu+02REdHX7PfPn36AEC3HhurK1eu4LnnntO/ktivXz+kpKSgpaVFfw4B4Kc//SlaWlowdOhQjBgxAgsXLjR9DTUmJgYvvPAC/vSnPyE1NRX5+fn4xS9+gY8++uiac4iLi8P//ve/kPyrV6/q5USfdwwG6KaQmJiItLQ0/POf/wwpGzVqFAoLC5GXl2fbNiYmJuT9XK/XiwceeAA7d+7EokWL8Ic//AF1dXX47W9/C8D/dbHuuvfee5GWlqan5cuXAwAyMzPR2tqK06dPR9SPFtR8//vfR0ZGhp5WrFiBq1evYtu2bWHbTp8+HYcOHUJ5eTlycnIivntxI0W6+KWnp8PtduP999/v8VhPP/00KisrMXXqVLz++uvYtWsX6urq0LdvX9NzmJ+fj3//+9/YsGEDhg8fjvXr1+Puu+/G+vXr9ToLFixAQ0MDli1bhtjYWPzkJz/BsGHD9LtE4aSlpaGpqSkkX8vr379/j8+P6LPC3xmgm0ZRURHWr1+Pv//97xg5cuR19fX++++joaEBVVVVpg8M1tXVmeppv1dgd3v5+PHjpuPq6mrTLeHBgwcDAB5++GFs2rQJr732GhYvXtzlvEQENTU1GDt2LObNmxdS/rOf/QzV1dVhb8OPHj0aAwcORH19PV544YUux4qU9hgcP35cPycAaG9vR2NjIwoLC3vUr8fjwde+9jW89dZbOH36NAYMGNDtPrZu3YqSkhKsWLFCz7t69SpaWlpC6iYnJ6O0tBSlpaW4fPky8vPzUV5eju985zt6nSFDhuCZZ57BM888gw8++AC5ublYsWJF2LtOAJCbm4t33nkHPp/PFHT+7W9/g8fjwdChQ7t9XkSfNd4ZoJvGj370I3g8HsyaNQtnz54NKReRiPtyuVwhbUQEK1euNNVLS0tDbm4uqqqqTLed6+rqcPToUVPdvLw8FBYW6klbOIuLizFixAhUVlZi7969IXO5dOkSlixZAgD4y1/+gpMnT6K0tBTFxcUh6bHHHsOePXvC/mKhoihYtWoVli5d2q1vL3SlsLAQ0dHRWLVqlenxeuWVV9Da2oqioqIe97106VKICGbMmGH7FtCBAwdQVVUVtr3L5Qp53levXh3yq4UXLlwwHSckJCA9PV2/vd/W1qbf1tcMGTIEt9xyi+1bAEbFxcU4e/Ystm/fruedP38eW7ZswcMPP2z7eQKizxveGaCbRkZGBmpqajBt2jTceeed+i8QiggaGxtRU1MDVVVtPx9glZmZiSFDhuCHP/wh/vvf/6JPnz7Ytm2b7YfHli1bhqKiIowePRqzZs3CJ598gtWrVyM7O9t2AbOKiorC9u3bUVhYiPz8fEydOhV5eXmIiorCkSNHUFNTg1tvvRWVlZWorq6Gy+UKu8BOnDgRS5YswebNm/GDH/zAts6kSZMwadKka84rUikpKVi8eDEqKirw4IMPYuLEiTh+/DjWrVuHe++9N+RXD7vjvvvuw9q1azFv3jxkZmaafoGwvr4eb7zxBp5//vmw7b/xjW9g48aNSExMRFZWFvbu3Ys333wTffv2NdXLyspCQUEB7rnnHiQnJ2P//v3YunUrnnrqKQBAQ0MDxo0bh6lTpyIrKwtutxu1tbU4e/YsHn/88S7Pobi4GF/96ldRWlqKo0eP6r9A6PV6UVFR0ePHhugz1VtfYyDqqRMnTkhZWZmkp6dLbGysxMXFSWZmpsydO1cOHTpkqqv96JCdo0ePSmFhoSQkJEi/fv1k9uzZ+tfdjL96JyKybds2GTZsmMTExEhWVla3fnRI09zcLM8995yMGDFCPB6PxMbGyvDhw2Xx4sXS1NQk7e3t0rdvX7n//vu77OcrX/mK3HXXXSJi/mphV67nq4WaNWvWSGZmpkRFRUlqaqqUlZWF/dGh7jpw4IBMnz5d+vfvL1FRUXLrrbfKuHHjpKqqyvR1Rli+Wtjc3CylpaXSr18/SUhIkAkTJsixY8dk0KBBUlJSotd7/vnnZeTIkZKUlKT/e6msrJT29nYRETl//rzMnz9fMjMzJT4+XhITE2XUqFHy+uuvRzT/Tz75RJ588knp27eveDweGTNmjOzbt6/bjwNRb1FEunFvlYiIiL5w+JkBIiIih2MwQERE5HAMBoiIiByOwQAREZHDMRggIiJyOAYDREREDsdggIiIyOEi/gXCessvaakAFENSDfmqodxaTwHgsuSrlj5dYdralZvmogCqCrhU/74xQduqNo1hmZx1wtbJI0x744krln7tyq0p3Ny6ejC76rs753UDxpbAsSiAILjvcykQxZBgTIAPLv0YAHyBjrU8H1RLG389rcxa39yXqrfX2khI37C0C+57AyeuzdWung8KxDRHmOpoczT3DXR9fvbnH/4cQh+bcH2HPvaABJ5wgQKRYPJ5A+19ir6FwF/uUwGf9oQHnnR9C8AXSBImeS3H2v9XyGcoh01+JH1f79g3om9jn9bzCtfWWh5CDFvrpIyTs3vQBEDnNU7ca+nP2r91jHAnYK1PvWnp0oJr1unRzxGHW0OMa5Ldgm5czPW1SwmsO0pw/XEpwcXdbquXW8u0ICAwuBJuopFGIjDXFUu5aOVQ/GWBExLFsIVh31QOiKKELwcgqrVcCc4DgfaKub2eh+A4If3DPA9AgRgiKtGeZW1RhzYgglvt3LRiQ/ChV9PWBdW8AOmLnQSORQ0uQoGG/mNtcTIshnpeYEEyLm5iXAj9fIZxtakH56CVB4+NL2nQF3pjubZQWxdl8+t3cF/gg5iOzX1I4KXS5vHRF2xzQGKuawyGQttagxRrPftAw/DYegPBki/4RyD6oq8AvsAfgjEICD74wTXB+uCGW3AB+wU50vY9Gbs7fVvLu9sWCB3bZ+knInadAt2LNMKVG/uzTs4YDHTVj3V+9HkXcTBgfD9BARAF88JuPHYFOlYNW23fBf+apgJwu/xX8qo7sHC7gp0oxkXbbd5qi72eb6inqDBHG8Zj4wTD1XEb6in2x6ICXpcSuOpVA1d+Lv0K0QcVXqjwKVqZ/wVXq6MdexUXBCq81vzAlbJPCR77AvX0qz1FMR3rYwbmIkow39+vGhjDP5Y2tj6mBObgVSGiwNvpgogKdLr8L/4++K8CtaBfS17L1povoXVEez3SLlK8SvDY598KjO2U0L69hn68gX+VpgsQf2eid+oLdq53IoZy+3oScqVlT1to7XIBr+El0VontPbnq45il2mgdFHWVbteLOuNMa9nrt1ueCMncqPHoM+rHt8Z0LbWi2/jBbjLZquvwQrgVgKBQeCq3tTQEhyELNR2i77bJt9l6dNax5pvHcMuOFABccN/y1tVAgtv8Jaxz7RAB4MEbVE3Lvo+qOjUF3X/1n/sX9z99dz+BTyw1Rb3zpBgQOvfFTKGdqvaC3dgDDe02+AigbYSmINPQafq9l8NKm7D37bh6s+66HciZLEOOTavw+GPO236FpuxjGPqFyHBBdifOiydWY8VmK9+OgN5AvNCeKNe4CLphy+mRPTZ4gcIiYiIHI7BABERkcMxGCAiInI4BgNEREQOx2CAiIjI4RgMEBERORyDASIiIodjMEBERORwDAaIiIgcjsEAERGRwzEYICIicjgGA0RERA7HYICIiMjhGAwQERE5HIMBIiIih2MwQERE5HAMBoiIiByOwQAREZHDMRggIiJyOAYDREREDsdggIiIyOEYDBARETkcgwEiIiKHYzBARETkcAwGiIiIHI7BABERkcMxGCAiInI4BgNEREQOx2CAiIjI4RgMEBERORyDASIiIodjMEBERORw7p42VMLsX6tMy1PsKtjtK2GSXR27+tZ6XfVp1xaAGPIkkABAoNyQBMO+DwoEqqncZ9ma91WbpNVR9brGfn2iBssD+yKKf9/n34cPgC+wFQSOb2Cy9mk89nY1ppjb+P+jPVOGrTeQfDb7xq113zgZsfRPRPTF1O1gQEVwrVQNx9q+K5AUQ5mW5wbgUgBVDSSXfwuXoQPV0IE7cOy2GcSY77IM7rLUs7axjmEdyziXQLnPDYgK+FwqfFDgVV3wQYUXLv8xXBCo8AWSP1+FN7BA2+V5A4uzF269D62efbk7sLCrlnrGdoo+hg8ufevT+wrU9brgExXeThdEFPi8Ln+U06lAfP6taZ3stNka11xjMq6xxtQJ85pr7cu4NZZ3ILgu6+t9YAAJN5h1Qp2GSVjzjUGAMaCAzT4R0RdPj+4MWC6cw16gw1JPO1CUQF3Fn2wrR3r1Hu5OgF1Su8i3llmiHlGDdwX0q3bFenVud+Wv6guxOc+6r9Uz1lf1q3qfoZ4x+SwpeEfA2F/wToEpiT+JT4H4AifoVfx3BLq6OtfWUi0Y6OqCO1yQEG5r7LPTkh+yJmuZ1gGNC741GOiwDNph6IOIyJkUEeFlDxERkYPxA4REREQOx2CAiIjI4RgMEBERORyDASIiIodjMEBERORwDAaIiIgcjsEAERGRwzEYICIicjgGA0RERA73f7A/Ew48ThjkAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Class 0: Min= 0.0    Max= 1.0\n",
      "Class 1: Min= 0.0    Max= 1.0\n"
     ]
    }
   ],
   "source": [
    "class GradCAM:\n",
    "    def __init__(self, model, target_layer):\n",
    "        \"\"\"\n",
    "        Initialize the Grad-CAM class.\n",
    "\n",
    "        Args:\n",
    "            model (torch.nn.Module): The model to use for Grad-CAM.\n",
    "            target_layer (torch.nn.Module): The layer to compute gradients for.\n",
    "        \"\"\"\n",
    "        self.model = model\n",
    "        self.target_layer = target_layer\n",
    "        self.gradients = None  # To store gradients\n",
    "\n",
    "        # Register hooks\n",
    "        self.hook_forward()\n",
    "        self.hook_backward()\n",
    "\n",
    "    def hook_forward(self):\n",
    "        \"\"\"\n",
    "        Hook to capture forward pass activations.\n",
    "        \"\"\"\n",
    "        def forward_hook(module, input, output):\n",
    "            self.activations = output\n",
    "        self.target_layer.register_forward_hook(forward_hook)\n",
    "\n",
    "    def hook_backward(self):\n",
    "        \"\"\"\n",
    "        Hook to capture backward pass gradients.\n",
    "        \"\"\"\n",
    "        def backward_hook(module, grad_in, grad_out):\n",
    "            self.gradients = grad_out[0]\n",
    "        self.target_layer.register_backward_hook(backward_hook)\n",
    "\n",
    "    def generate(self, input_image, target_class):\n",
    "        \"\"\"\n",
    "        Generate Grad-CAM heatmap.\n",
    "\n",
    "        Args:\n",
    "            input_image (torch.Tensor): Input image tensor.\n",
    "            target_class (int): Class index for which to generate Grad-CAM.\n",
    "\n",
    "        Returns:\n",
    "            numpy.ndarray: Grad-CAM heatmap.\n",
    "        \"\"\"\n",
    "        # Forward pass\n",
    "        output = self.model(input_image)\n",
    "\n",
    "        # Backward pass for the target class\n",
    "        self.model.zero_grad()  # Zero gradients\n",
    "        one_hot_output = torch.zeros_like(output)\n",
    "        one_hot_output[0, target_class] = 1  # Set target class to 1\n",
    "        output.backward(gradient=one_hot_output)\n",
    "\n",
    "        # Compute Grad-CAM\n",
    "        gradients = self.gradients  # [Batch, Channels, H, W]\n",
    "        activations = self.activations  # [Batch, Channels, H, W]\n",
    "        weights = torch.mean(gradients, dim=(2, 3), keepdim=True)  # Global average pooling on gradients\n",
    "        grad_cam = torch.sum(weights * activations, dim=1).squeeze()  # Weighted sum of activations\n",
    "\n",
    "        # Apply ReLU to the heatmap\n",
    "        grad_cam = torch.relu(grad_cam)\n",
    "        grad_cam = grad_cam.detach().cpu().numpy()\n",
    "\n",
    "        # Normalize the heatmap\n",
    "        grad_cam = (grad_cam - np.min(grad_cam)) / (np.max(grad_cam) - np.min(grad_cam))\n",
    "        return grad_cam\n",
    "\n",
    "\n",
    "# Instantiate Grad-CAM\n",
    "target_layer = model.conv1  # Specify the target layer\n",
    "grad_cam = GradCAM(model, target_layer)\n",
    "\n",
    "# Preprocess image\n",
    "input_image = preprocess_image(image_path)\n",
    "input_image.requires_grad = True  # Ensure gradient computation\n",
    "\n",
    "# Dictionary to store saliency maps as numpy arrays\n",
    "saliency_maps = {}\n",
    "\n",
    "# Generate Grad-CAM for each class\n",
    "output = model(input_image)\n",
    "num_classes = output.size(1)\n",
    "\n",
    "for class_idx in range(num_classes):\n",
    "    heatmap = grad_cam.generate(input_image, class_idx)\n",
    "    \n",
    "    # Store heatmap as numpy array\n",
    "    saliency_maps[class_idx] = heatmap\n",
    "    \n",
    "    # Visualize the heatmap\n",
    "    plt.imshow(heatmap, cmap='jet', alpha=0.5)\n",
    "    plt.title(f\"Grad-CAM for Class {class_idx}\")\n",
    "    plt.axis('off')\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "# At this point, saliency_maps is a dictionary containing numpy arrays\n",
    "# Example usage:\n",
    "class_0_map = saliency_maps[0]  # Numpy array for class 0\n",
    "class_1_map = saliency_maps[1]  # Numpy array for class 1\n",
    "\n",
    "# Print confirmation\n",
    "print(\"Class 0: Min=\", np.min(class_0_map), \"   Max=\", np.max(class_0_map))\n",
    "print(\"Class 1: Min=\", np.min(class_1_map), \"   Max=\", np.max(class_1_map))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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RxQeHq7rr7/etti2efNmXXnllSopKdHMmTP12GOP6eqrr9ZHH33ki3n22Wf1q1/9St26ddP8+fM1Y8YMXXTRRQGvz+qLL76QpAqfhdG3b1+53W5fOwAAABCKqPeo9/ydT/UeAKAOGACAo/397383YWFhJiwszAwcONDcc8895m9/+5spLS2tEHv8+PEKxzIyMkzHjh1tx4YMGWKGDBnie56fn28kmUWLFvmODR8+3PTs2dOcPHnSd8zr9ZqLL77YdOnSxXds0aJFRpJJT083Xq/Xd/zOO+80YWFh5vDhw8YYYw4fPmwaN25sUlNTzYkTJ2z5WPsNHDjQpKam2tpXrlxpJJmcnJwAd+iM6dOnG0lm//795q677jKdO3f2tfXv39/cdtttxhhjJJmsrCxfW1lZmSkpKbGN9cMPP5jExETzi1/8osJ9iomJMd9//73veF5enpFk7rzzzirzK+//6KOP2o6PHj3aREZGmp07d/qO7dmzxzRu3NgMHjzYd6z8XqelpZmysrIqz2WMMX369DFNmjQ5a1y58ePHm5SUFNsx/zlVWlpqevToYYYNG+Y7Nm/ePN99r8yoUaNM9+7dq51LuaysLBMWFhawrWXLluaGG26o8ZgAAACAU1DvUe+VOx/rPX+xsbFm/PjxP3ocADif8Zd+AOBwl112mXJzc3X11Vdr48aNmjt3rjIyMnTBBRdUeOuVmJgY3+MjR47owIEDGjJkiP75z3/qyJEj1T7noUOHtHbtWv385z9XcXGxDhw4oAMHDujgwYPKyMjQ9u3b9e9//9vWZ+LEiba35Bg0aJA8Ho/+9a9/STr9diDFxcW67777KuzetPa75ZZblJeXp507d/qOLVmyRG3bttWQIUOqfQ3jxo3Tjh07tH79et9/A73ViySFhYUpMjJSkuT1enXo0CGVlZWpX79++vzzzyvEjx49WhdccIHv+YABA5Samqq333672vmV83g8+vvf/67Ro0erY8eOvuOtW7fWuHHj9OGHH6qoqMjWZ8KECQoLCzvr2EVFRWrcuHGNc7KyzqkffvhBR44c0aBBg2z3pfytiF577TV5vd6A4yQkJOj777/X+vXra3T+EydO+L43/qKjo3XixIkajQcAAAA4CfUe9d75XO8BAGofi34AEAL69++vlStX6ocfftCnn36q7OxsFRcX67rrrtOWLVt8cR99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",
      "text/plain": [
       "<Figure size 1800x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Values for both classes have been saved as numpy arrays.\n"
     ]
    }
   ],
   "source": [
    "# Plotting the SHAP values for both classes next to each other\n",
    "plt.figure(figsize=(18, 6))\n",
    "\n",
    "# Plot for Class 0 SHAP values\n",
    "plt.subplot(1, 2, 1)  # (rows, columns, index)\n",
    "plt.imshow(class_0_map, cmap='coolwarm')\n",
    "plt.title(\"Saliency Map for Class 0\")\n",
    "#plt.colorbar()\n",
    "\n",
    "# Plot for Class 1 SHAP values\n",
    "plt.subplot(1, 2, 2)  # (rows, columns, index)\n",
    "plt.imshow(class_0_map, cmap='coolwarm')\n",
    "plt.title(\"Saliency Map for Class 1\")\n",
    "#plt.colorbar()\n",
    "\n",
    "# Display the plots\n",
    "plt.tight_layout()  # Adjust layout for better visualization\n",
    "plt.show()\n",
    "\n",
    "# Saving the SHAP values per class as numpy arrays\n",
    "np.save(\"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results_explainer_comparison/Gradcam_class_0.npy\", class_0_map)\n",
    "np.save(\"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results_explainer_comparison/Gradcam_class_1.npy\", class_1_map)\n",
    "\n",
    "print(\"Values for both classes have been saved as numpy arrays.\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 2. Grad-Cam++ "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Grad-CAM++ saliency maps are stored as variables in the `saliency_maps` dictionary.\n",
      "Class 0: Min= 0.0    Max= 1.0\n",
      "Class 1: Min= 0.0    Max= 1.0\n"
     ]
    }
   ],
   "source": [
    "class GradCAMPlusPlus:\n",
    "    def __init__(self, model, target_layer):\n",
    "        \"\"\"\n",
    "        Initialize the Grad-CAM++ class.\n",
    "\n",
    "        Args:\n",
    "            model (torch.nn.Module): The model to use for Grad-CAM++.\n",
    "            target_layer (torch.nn.Module): The layer to compute gradients for.\n",
    "        \"\"\"\n",
    "        self.model = model\n",
    "        self.target_layer = target_layer\n",
    "        self.gradients = None  # To store gradients\n",
    "\n",
    "        # Register hooks\n",
    "        self.hook_forward()\n",
    "        self.hook_backward()\n",
    "\n",
    "    def hook_forward(self):\n",
    "        \"\"\"\n",
    "        Hook to capture forward pass activations.\n",
    "        \"\"\"\n",
    "        def forward_hook(module, input, output):\n",
    "            self.activations = output\n",
    "        self.target_layer.register_forward_hook(forward_hook)\n",
    "\n",
    "    def hook_backward(self):\n",
    "        \"\"\"\n",
    "        Hook to capture backward pass gradients.\n",
    "        \"\"\"\n",
    "        def backward_hook(module, grad_in, grad_out):\n",
    "            self.gradients = grad_out[0]\n",
    "        self.target_layer.register_backward_hook(backward_hook)\n",
    "\n",
    "    def generate(self, input_image, target_class):\n",
    "        \"\"\"\n",
    "        Generate Grad-CAM++ heatmap.\n",
    "\n",
    "        Args:\n",
    "            input_image (torch.Tensor): Input image tensor.\n",
    "            target_class (int): Class index for which to generate Grad-CAM++.\n",
    "\n",
    "        Returns:\n",
    "            numpy.ndarray: Grad-CAM++ heatmap.\n",
    "        \"\"\"\n",
    "        # Forward pass\n",
    "        output = self.model(input_image)\n",
    "\n",
    "        # Backward pass for the target class\n",
    "        self.model.zero_grad()  # Zero gradients\n",
    "        one_hot_output = torch.zeros_like(output)\n",
    "        one_hot_output[0, target_class] = 1  # Set target class to 1\n",
    "        output.backward(gradient=one_hot_output)\n",
    "\n",
    "        # Compute Grad-CAM++\n",
    "        gradients = self.gradients  # [Batch, Channels, H, W]\n",
    "        activations = self.activations  # [Batch, Channels, H, W]\n",
    "\n",
    "        # Higher-order derivatives\n",
    "        gradients_power_2 = gradients ** 2\n",
    "        gradients_power_3 = gradients_power_2 * gradients\n",
    "\n",
    "        # Alpha coefficients\n",
    "        alpha_num = gradients_power_2\n",
    "        alpha_denom = 2 * gradients_power_2 + torch.sum(activations * gradients_power_3, dim=(2, 3), keepdim=True)\n",
    "        alpha_denom = torch.where(alpha_denom != 0, alpha_denom, torch.ones_like(alpha_denom))  # Avoid division by zero\n",
    "        alphas = alpha_num / alpha_denom\n",
    "\n",
    "        # Weights\n",
    "        weights = torch.sum(alphas * torch.relu(gradients), dim=(2, 3), keepdim=True)\n",
    "\n",
    "        # Compute Grad-CAM++ heatmap\n",
    "        grad_cam_pp = torch.sum(weights * activations, dim=1).squeeze()  # Weighted sum of activations\n",
    "\n",
    "        # Apply ReLU to the heatmap\n",
    "        grad_cam_pp = torch.relu(grad_cam_pp)\n",
    "        grad_cam_pp = grad_cam_pp.detach().cpu().numpy()\n",
    "\n",
    "        # Normalize the heatmap\n",
    "        grad_cam_pp = (grad_cam_pp - np.min(grad_cam_pp)) / (np.max(grad_cam_pp) - np.min(grad_cam_pp))\n",
    "        return grad_cam_pp\n",
    "\n",
    "\n",
    "# Instantiate Grad-CAM++\n",
    "target_layer = model.conv1  # Specify the target layer\n",
    "grad_cam_pp = GradCAMPlusPlus(model, target_layer)\n",
    "\n",
    "# Dictionary to store saliency maps as numpy arrays\n",
    "saliency_maps = {}\n",
    "\n",
    "# Generate Grad-CAM++ for each class\n",
    "output = model(input_image)\n",
    "num_classes = output.size(1)\n",
    "\n",
    "for class_idx in range(num_classes):\n",
    "    heatmap = grad_cam_pp.generate(input_image, class_idx)\n",
    "    \n",
    "    # Store heatmap as numpy array\n",
    "    saliency_maps[class_idx] = heatmap\n",
    "    \n",
    "    # Visualize the heatmap\n",
    "    plt.imshow(heatmap, cmap='jet', alpha=0.5)\n",
    "    plt.title(f\"Grad-CAM++ for Class {class_idx}\")\n",
    "    plt.axis('off')\n",
    "    plt.show()\n",
    "\n",
    "# At this point, saliency_maps is a dictionary containing numpy arrays\n",
    "# Example usage:\n",
    "class_0_map = saliency_maps[0]  # Numpy array for class 0\n",
    "class_1_map = saliency_maps[1]  # Numpy array for class 1\n",
    "\n",
    "# Print confirmation\n",
    "print(\"Grad-CAM++ saliency maps are stored as variables in the `saliency_maps` dictionary.\")\n",
    "\n",
    "# At this point, saliency_maps is a dictionary containing numpy arrays\n",
    "# Example usage:\n",
    "class_0_map = saliency_maps[0]  # Numpy array for class 0\n",
    "class_1_map = saliency_maps[1]  # Numpy array for class 1\n",
    "\n",
    "# Print confirmation\n",
    "print(\"Class 0: Min=\", np.min(class_0_map), \"   Max=\", np.max(class_0_map))\n",
    "print(\"Class 1: Min=\", np.min(class_1_map), \"   Max=\", np.max(class_1_map))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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ea9s5L3Az3sAgMrHK/0AwOLS09Pd3j1Z/HkCxW8TUnznZcm2ubm5bv+HvSxNmjTRoEGD9OqrrzpCZkknT54s95hDhw5V3bp1NWfOHF26dMlpX+nzGz58uBo1aqS5c+dq06ZNFbrrU7oaPBcsWKA5c+aob9++Htu5u3aZmZnKyMhw23716tVOn3GxZcsWZWZmavjw4eWuMTg4WEOHDtX777/v9HYxOTk5WrZsmRITExUVFVXucSXprrvuUteuXTV79my355Kfn68nn3zSa202m81x96Ykfffdd1q9erVTuzNnzrj0LV6MKygokCSdPn3aaX9ISIg6deokY4wuX77ssYbBgwerQYMGWrRokdP2RYsWqU6dOkpJSfHYFwAAALA68h5573rOewCAyscr/QDA4qZMmaILFy7ojjvuUMeOHVVYWKgvvvhCK1asUKtWrfTAAw9IuhqyQkJCdNttt+m///u/de7cOb322mtq0qSJ2yBXloULFyoxMVFdu3bVhAkT1KZNG+Xk5CgjI0Pff/+9du3aVa7xoqKiNH/+fP3yl79Unz59NG7cONWvX1+7du3ShQsX9MYbbzja1q5dW2PGjNHLL7+s4OBgjR07ttz1F/v1r39dZptbb71Vq1at0h133KGUlBRlZWXplVdeUadOnXTu3DmX9u3atVNiYqIeeughFRQUaMGCBWrYsKGmTZtWoRqfeeYZrVu3TomJiZo0aZJq1aqlV199VQUFBZo3b16FxpSuXsdVq1YpKSlJAwYM0M9//nPdeOONql27tvbu3atly5apfv36mj17ttv+KSkpeuGFFzRs2DCNGzdOJ06c0MKFC9WuXTvt3r3b0W7WrFnavHmzUlJSFB8frxMnTuhPf/qTmjdvrsTERElX52dsbKxuvPFGxcTE6JtvvtHLL7+slJQUrx8+Hx4ert/97ndKTU3V3XffreTkZH366ad6++23NXv2bDVo0KDC1wcAAADwN/Ieea+iakLek66+irV4vl2+fFm7d+/WM888I0m6/fbb1a1btwpfIwC4LhkAgKV98skn5he/+IXp2LGjiYyMNCEhIaZdu3ZmypQpJicnx6ntBx98YLp162bCwsJMq1atzNy5c83//d//GUkmKyvL0W7gwIFm4MCBjsdZWVlGklm8eLHTeIcPHzb33XefiY2NNbVr1zbNmjUzt956q1m5cqWj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",
      "text/plain": [
       "<Figure size 1800x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Values for both classes have been saved as numpy arrays.\n"
     ]
    }
   ],
   "source": [
    "# Plotting the SHAP values for both classes next to each other\n",
    "plt.figure(figsize=(18, 6))\n",
    "\n",
    "# Plot for Class 0 SHAP values\n",
    "plt.subplot(1, 2, 1)  # (rows, columns, index)\n",
    "plt.imshow(class_0_map, cmap='coolwarm')\n",
    "plt.title(\"Saliency Map for Class 0\")\n",
    "#plt.colorbar()\n",
    "\n",
    "# Plot for Class 1 SHAP values\n",
    "plt.subplot(1, 2, 2)  # (rows, columns, index)\n",
    "plt.imshow(class_0_map, cmap='coolwarm')\n",
    "plt.title(\"Saliency Map for Class 1\")\n",
    "#plt.colorbar()\n",
    "\n",
    "# Display the plots\n",
    "plt.tight_layout()  # Adjust layout for better visualization\n",
    "plt.show()\n",
    "\n",
    "# Saving the SHAP values per class as numpy arrays\n",
    "np.save(\"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results_explainer_comparison/Gradcamplusplus_class_0.npy\", class_0_map)\n",
    "np.save(\"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results_explainer_comparison/Gradcamplusplus_class_1.npy\", class_1_map)\n",
    "\n",
    "print(\"Values for both classes have been saved as numpy arrays.\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 3. Smoothed Grad-Cam++"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Generating Smoothed Grad-CAM++:   6%|▋         | 64/1000 [00:00<00:02, 316.86it/s]/tmp/ipykernel_6507/3159117808.py:79: RuntimeWarning: invalid value encountered in divide\n",
      "  grad_cam_pp = (grad_cam_pp - np.min(grad_cam_pp)) / (np.max(grad_cam_pp) - np.min(grad_cam_pp))\n",
      "Generating Smoothed Grad-CAM++: 100%|██████████| 1000/1000 [00:03<00:00, 314.31it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAgMAAABRCAYAAABPAqWCAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAFkFJREFUeJzt3X1UVGUeB/Dv8DYwICgCgiwvKuiImLDmW5hIgq4viAWihMmLa2d9TTM142ziaYkUWNd03XJbX1ZBEwHXWEk04FSiB00jRY00XhQBJVERbFn12T88c5dhBhjMQrvfzzlzTj73uff+ZuZy7/e+PJNCCCFAREREsmXU1QUQERFR12IYICIikjmGASIiIpljGCAiIpI5hgEiIiKZYxggIiKSOYYBIiIimWMYICIikjmGASIiIpljGKAnkkKhwMKFC3/29RQUFEChUKCgoOBnX9fjFB0dDXd3964u46l27949rFixAi4uLjAyMsK0adO6uiTJ07pd0tOLYeBX4MyZMwgLC4ObmxvMzc3h7OyMoKAgbNy4satLa1dhYSHi4+Nx8+bNri7FYGVlZVi4cCH69+8PlUoFlUoFLy8vLFiwAN98801Xl9em27dvY82aNRgyZAisrKxgYWEBb29vrFy5ElevXtU7T3h4OBQKBVauXKl3uuaApVAosGvXLr19/Pz8oFAo4O3t/djey+OydetWJCUlISwsDDt27MDSpUt/kfVmZWVh4sSJsLOzg5mZGXr37o3w8HDk5eX9Iuv/qc6fP4/f/e53sLKygq2tLV555RVcv369q8uin8ikqwugn6awsBABAQFwdXXF3Llz4ejoiMuXL+P48ePYsGEDFi1a1NUltqmwsBBr1qxBdHQ0unfv3tXldCg7OxszZsyAiYkJIiMjMWTIEBgZGeHChQvIzMzE3/72N5SVlcHNza2rS9Xy/fffIzAwEJWVlZg+fTpeffVVmJmZ4ZtvvsE//vEPZGVlobS0VGue27dv45NPPoG7uzt2796N9957DwqFQu/yzc3NkZaWhlmzZmm1l5eXo7CwEObm5j/be/sp8vLy4OzsjPXr1/8i6xNCIDY2Ftu3b4evry9ef/11ODo6orq6GllZWRg3bhyOHj2K55577hep51FcuXIFY8aMgY2NDd59913cuXMHycnJOHPmDIqKimBmZtbVJdIjYhh4yiUkJMDGxgYnTpzQOaBeu3ata4r6Fbp06RJmzpwJNzc3fPbZZ3ByctKavnbtWmzevBlGRu1fbGtsbISlpeXPWaqWe/fu4aWXXkJtbS0KCgowevRorekJCQlYu3atznwZGRm4f/8+tm7dihdeeAGff/45/P399a5j0qRJOHDgAOrq6mBnZye1p6WloVevXvD09ER9fX27dRYUFCAgIABlZWWduv2xfft2xMTE4FH+f2vXrl17rCH0wYMHaG5ubjP8pKSkYPv27ViyZAn+/Oc/a4WruLg47Ny5EyYmT/Yu+d1330VjYyO++uoruLq6AgCGDx+OoKAgbN++Ha+++moXV0iPircJnnKXLl3CoEGD9O7UHBwctP6tuQ+fnp4OLy8vWFhYYNSoUThz5gwA4MMPP4SHhwfMzc0xduxYlJeX6ywzPT0dQ4cOhYWFBezs7DBr1ixUVVXp9MvLy8Pzzz8PS0tLdO/eHSEhITh//rw0PT4+HsuXLwcA9OnTR7rc3Hqd+/fvh7e3N5RKJQYNGoRPP/1UZ11VVVWIjY1Fr169pH5bt27V6XflyhVMmzYNlpaWcHBwwNKlS/Gf//xHp58+69atQ2NjI7Zt26YTBADAxMQEixcvhouLi9QWHR0NKysrXLp0CZMmTUK3bt0QGRkJAPjiiy8wffp0uLq6QqlUwsXFBUuXLsXdu3d1lq35DMzNzeHt7Y2srCyDagYeHtSLi4sRFxenEwQAwNraGgkJCTrtqampCAoKQkBAAAYOHIjU1NQ21xESEgKlUon09HSt9rS0NISHh8PY2Njgen8J5eXlUCgUyM/PR0lJibTtae7PNzY2YtmyZXBxcYFSqcSAAQOQnJysEzg0f0+pqakYNGgQlEql3u0TAO7evYvExESo1WokJyfrvcryyiuvYPjw4W3Wbeg2U1NTg5iYGPzmN7+BUqmEk5MTQkJCtP62Tp48iQkTJsDOzg4WFhbo06cPYmNjO/zsMjIyMGXKFCkIAEBgYCD69++PvXv3djg/Pbme7BhKHXJzc8OxY8dw9uxZg+7LfvHFFzhw4AAWLFgAAEhMTMSUKVOwYsUKbN68GfPnz0d9fT3WrVuH2NhYrfuYmrOwYcOGITExEbW1tdiwYQOOHj2K06dPS4HkyJEjmDhxIvr27Yv4+HjcvXsXGzduhJ+fH06dOgV3d3e89NJLKC0txe7du7F+/XrpjNLe3l5a35dffonMzEzMnz8f3bp1w/vvv4/Q0FBUVlaiZ8+eAIDa2lqMHDlS2jHb29sjJycHc+bMwe3bt7FkyRIAD3fG48aNQ2VlJRYvXozevXtj586dBt+nzc7OhoeHB0aMGGFQf4179+5hwoQJGD16NJKTk6FSqQA8DFVNTU2YN28eevbsiaKiImzcuBFXrlzROqjm5uYiNDQUXl5eSExMxA8//CDt6A1x4MABAA8PNIa6evUq8vPzsWPHDgBAREQE1q9fj02bNum9DKxSqRASEoLdu3dj3rx5AIDi4mKUlJTgo48+euKepbC3t8fOnTuRkJCAO3fuIDExEQAwcOBACCEwdepU5OfnY86cOfDx8cGhQ4ewfPlyVFVV6dxSyMvLw969e7Fw4ULY2dm1eVXjyy+/xI0bN7BkyZJHDkeGbjOhoaEoKSnBokWL4O7ujmvXruHw4cOorKyU/j1+/HjY29vjzTffRPfu3VFeXo7MzMx2119VVYVr167h2Wef1Zk2fPhwHDx48JHeFz0hBD3VcnNzhbGxsTA2NhajRo0SK1asEIcOHRLNzc06fQEIpVIpysrKpLYPP/xQABCOjo7i9u3bUvuqVasEAKlvc3OzcHBwEN7e3uLu3btSv+zsbAFAvP3221Kbj4+PcHBwED/88IPUVlxcLIyMjMTs2bOltqSkJK11tK7VzMxMXLx4UWsZAMTGjRultjlz5ggnJydRV1enNf/MmTOFjY2NaGpqEkII8Ze//EUAEHv37pX6NDY2Cg8PDwFA5Ofn69SgcevWLQFATJs2TWdafX29uH79uvTSrE8IIaKiogQA8eabb+rM17KfRmJiolAoFKKiokJq8/HxEU5OTuLmzZtSW25urgAg3Nzc2qxZw9fXV9jY2HTYr6Xk5GRhYWEhbQ+lpaUCgMjKytLql5+fLwCI9PR0kZ2dLRQKhaisrBRCCLF8+XLRt29fIYQQ/v7+YtCgQe2uU7MsfdtCe7Zt2yYedTemr679+/cLAOJPf/qTVntYWJhQKBRa2yMAYWRkJEpKSjpc14YNG/R+hm3RfB4tt0tDtpn6+noBQCQlJbW57KysLAFAnDhxwqBaNE6cOCEAiH/+858605YvXy4AiB9//LFTy6QnB28TPOWCgoJw7NgxTJ06FcXFxVi3bh0mTJgAZ2dn6aywpXHjxmmdvWjOdENDQ9GtWzed9u+//x7Aw8uK165dw/z587XuiU6ePBlqtRr//ve/AQDV1dX4+uuvER0dDVtbW6nfM888g6CgoE6dPQQGBqJfv35ay7C2tpZqEkIgIyMDwcHBEEKgrq5Oek2YMAG3bt3CqVOnAAAHDx6Ek5MTwsLCpOWpVCqD7nHevn0bAGBlZaUzbezYsbC3t5def/3rX3X6aM6WW7KwsJD+u7GxEXV1dXjuuecghMDp06cB/P+zjIqKgo2NjdQ/KCgIXl5eHdatqb3l92qI1NRUTJ48WZrP09MTQ4cObfdWwfjx42Fra4s9e/ZACIE9e/YgIiKizf63bt3S+r5u3boFAKivr9dqv3PnjtZ8bU1v2VZXV4empqZOvWeNgwcPwtjYGIsXL9ZqX7ZsGYQQyMnJ0Wr39/c36LvQbEOd/S5aMmSbsbCwgJmZGQoKCtp8TkNzBS87Oxv//e9/DV6/5naEUqnUmabZJ+i7zUVPB4aBX4Fhw4YhMzMT9fX1KCoqwqpVq9DQ0ICwsDCcO3dOq2/Le30ApINMy3vdLds1O5SKigoAwIABA3TWr1arpent9Rs4cCDq6urQ2Nho0PtqXSsA9OjRQ6rp+vXruHnzJrZs2aJ1QLa3t0dMTAyA/z9EWVFRAQ8PD517tfrqbE2zA299YAIePmdx+PDhNofWmZiY6L2kX1lZKQUmKysr2NvbSw/oaQ6Mms/S09NTZ/7WdV+/fh01NTXSS1OrtbU1GhoaOnyPGufPn8fp06fh5+eHixcvSq+xY8ciOztbOqi1ZmpqiunTpyMtLQ2ff/45Ll++jJdffrnN9YSEhGh9X5ox/r/97W+12lv/1oSvr6/WdM1omdbf/7p16wx+zy1VVFSgd+/eOgftgQMHStNb6tOnj0HLtba2BoBOfRetGbLNKJVKrF27Fjk5OejVqxfGjBmDdevWoaamRlqOv78/QkNDsWbNGtjZ2SEkJATbtm3r8PkZTRjR1+/HH3/U6kNPHz4z8CtiZmaGYcOGYdiwYejfvz9iYmKQnp6O1atXS33aul/ZVrt4hKe0H5eOanrw4AEAYNasWYiKitLb95lnnvnJddjY2MDJyQlnz57Vmaa5gqLvYUvg4c659QiD+/fvIygoCDdu3MDKlSuhVqthaWmJqqoqREdHS++rM4YNG6Z1oFq9ejXi4+OhVqtx+vRpXL58WSfw6aMJNUuXLtU77j4jI0MKWq29/PLL+OCDDxAfH48hQ4a0e8ackpKideZaXFyMN954A7t27UKvXr2k9t69e2vNl5qaqnX2mZubi6SkJBw+fFirX9++fdt5l4+PoQc/tVoN4OFvgjzKjxt1ZptZsmQJgoODsX//fhw6dAh//OMfkZiYiLy8PPj6+kKhUGDfvn04fvw4PvnkExw6dAixsbFISUnB8ePH9V4BAyA9OFtdXa0zrbq6Gra2tnqvGtDTgWHgV0rzkI++P9xHoRk7/+233+KFF17Qmvbtt99K01v2a+3ChQuws7OThta1NW7dUPb29ujWrRvu37+PwMDADus/e/YshBBa69VXpz6TJ0/GRx99hKKionaf+DbEmTNnUFpaih07dmD27NlSe+sDmuaz/O6773SW0bru1gdJzcEwODgYu3fvxq5du7Bq1ap26xJCIC0tDQEBAZg/f77O9HfeeQepqalthoHRo0fD1dUVBQUFeocrtjR06FCtf2uG1Pn5+bU7tNDPz0/r31euXAGADr9/Q7m5ueHIkSNoaGjQujpw4cIFafqjGD16NHr06IHdu3fjrbfe6vRDhIZuMxr9+vXDsmXLsGzZMnz33Xfw8fFBSkqK1hWskSNHYuTIkUhISEBaWhoiIyOxZ88e/P73v9e7TGdnZ9jb2+PkyZM604qKiuDj49Op90RPFt4meMrl5+frPXvX3Js35DK4IZ599lk4ODjggw8+0LpMmJOTg/Pnz2Py5MkAHp49+Pj4YMeOHVq/LHj27Fnk5uZi0qRJUpsmFDzqLxAaGxsjNDQUGRkZes/aW/4q2qRJk3D16lXs27dPamtqasKWLVsMWteKFSugUqkQGxuL2tpanemduYKiORC0nEcIgQ0bNmj1a/lZai4DAw8PAK1v//j5+SEwMFB6acJAWFgYBg8ejISEBBw7dkynloaGBsTFxQEAjh49ivLycsTExCAsLEznNWPGDOTn57f5i4UKhQLvv/8+Vq9e3anRC0+SSZMm4f79+9i0aZNW+/r166FQKDBx4sRHWq5KpcLKlStx/vx5rFy5Uu/2smvXLhQVFemd39BtpqmpSbpkr9GvXz9069ZN+rutr6/XWb/mQN7RrYLQ0FBkZ2fj8uXLUttnn32G0tJSTJ8+vd156cnGKwNPuUWLFqGpqQkvvvgi1Go1mpubUVhYiI8//hju7u5tnsV1lqmpKdauXYuYmBj4+/sjIiJCGlro7u6udUk5KSkJEydOxKhRozBnzhxpaKGNjQ3i4+Olfpqzw7i4OMycOROmpqYIDg7u1I/yvPfee8jPz8eIESMwd+5ceHl54caNGzh16hSOHDmCGzduAADmzp2LTZs2Yfbs2fjqq6/g5OSEnTt3SkP9OuLp6Ym0tDRERERgwIAB0i8QCiFQVlaGtLQ0GBkZGTTkT61Wo1+/fnjjjTdQVVUFa2trZGRk6H3gKzExEZMnT8bo0aMRGxuLGzduYOPGjRg0aJDeZxhaMzU1RWZmJgIDAzFmzBiEh4fDz88PpqamKCkpQVpaGnr06IGEhASkpqbC2NhYCnatTZ06FXFxcdizZw9ef/11vX1CQkIQEhLSYV1PquDgYAQEBCAuLg7l5eUYMmQIcnNz8a9//QtLlizReqC1s5YvX46SkhKkpKQgPz8fYWFhcHR0RE1NDfbv34+ioiIUFhbqndfQbaa0tBTjxo1DeHg4vLy8YGJigqysLNTW1mLmzJkAgB07dmDz5s148cUX0a9fPzQ0NODvf/87rK2ttcK6Pm+99RbS09MREBCA1157DXfu3EFSUhIGDx782PY11EV+4dEL9Jjl5OSI2NhYoVarhZWVlTAzMxMeHh5i0aJFora2VqsvALFgwQKttrKyMr1DkVoOG2vp448/Fr6+vkKpVApbW1sRGRkprly5olPXkSNHhJ+fn7CwsBDW1tYiODhYnDt3TqffO++8I5ydnYWRkZHW0DJ9tQohhJubm4iKitJqq62tFQsWLBAuLi7C1NRUODo6inHjxoktW7Zo9auoqBBTp04VKpVK2NnZiddee018+umnHQ4tbOnixYti3rx5wsPDQ5ibmwsLCwuhVqvFH/7wB/H1119r9Y2KihKWlpZ6l3Pu3DkRGBgorKyshJ2dnZg7d640dHLbtm1afTMyMsTAgQOFUqkUXl5eIjMzU0RFRRk0tFCjvr5evP3222Lw4MFCpVIJc3Nz4e3tLVatWiWqq6tFc3Oz6Nmzp3j++efbXU6fPn2Er6+vEKLtbaS1p2looRBCNDQ0iKVLl4revXsLU1NT4enpKZKSksSDBw+0+rW1jXZk3759Yvz48cLW1laYmJgIJycnMWPGDFFQUCD10Te00JBtpq6uTixYsECo1WphaWkpbGxsxIgRI7SG1J46dUpEREQIV1dXoVQqhYODg5gyZYo4efKkQfWfPXtWjB8/XqhUKtG9e3cRGRkpampqOv050JNFIUQXPiFGREREXY7PDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHM/Q/7Lc4qqB8nAwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Generating Smoothed Grad-CAM++: 100%|██████████| 1000/1000 [00:03<00:00, 312.94it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "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W6OoqEjnDV/p6ekIDQ1FQEAA4uLi0NbWhuzsbIwdO1bnPQx9GRsbo7i4GMHBwXjxxRcxb948+Pv7w9jYGHV1dcjPz8fQoUORlpaGvLw8GBoaSsGur9mzZyMlJQV79uzBG2+8obNPeHg4wsPDB6zrcRUWFoZp06YhJSUFDQ0NGD9+PMrLy/H3v/8diYmJGje0PqjVq1ejrq4OWVlZqKysRGRkJBwdHXH16lXs27cPNTU1qK6u1jmvvtvMuXPnEBQUhHnz5sHb2xtGRkYoKSlBS0sLFixYAADIycnB1q1b8corr8Dd3R2dnZ3461//Cmtra42wrktjYyNyc3MB3LupGADeffddAPfOmjypIZDAcSFPurKyMhEXFydUKpWwtLQUJiYmwsPDQ6xYsUK0tLRo9AUgEhISNNrq6+t1DkXqPWyst48++kj4+fkJpVIphg0bJhYuXCguXbqkVdfhw4eFv7+/MDMzE9bW1iIsLEycPn1aq98777wjRo4cKQwMDDSGlumqVQghXF1dRXR0tEZbS0uLSEhIEM7OzsLY2Fg4OjqKoKAgsW3bNo1+jY2NYvbs2cLc3FzY2tqK3/3ud+LTTz8dcGhhb+fPnxfLli0THh4ewtTUVJiZmQmVSiV++9vfiq+//lqjb3R0tLCwsNC5nNOnT4vg4GBhaWkpbG1txdKlS6Whkzt37tToW1RUJLy8vIRSqRTe3t6iuLhYREdH6zW0UK29vV28/fbb4tlnnxXm5ubC1NRU+Pj4iLVr14orV66Inp4eMXz4cPHCCy/cdzmjRo0Sfn5+Qoj+t5G+nqShhUII0dnZKZKSksSIESOEsbGx8PT0FJmZmeLu3bsa/frbRgeyd+9eMX36dDFs2DBhZGQknJycxPz580VVVZXUR9fQQn22mdbWVpGQkCBUKpWwsLAQNjY2YtKkSRpDao8fPy6ioqKEi4uLUCqVwt7eXrz88sviq6++GrB2dV26HoGBgQ/8XtDjQyHEIN4hRkRERIOO9wwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzDEMEBERyRzDABERkcwxDBAREckcwwAREZHMMQwQERHJHMMAERGRzP0fK82E2pCxACwAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Smoothed Grad-CAM++ saliency maps are stored as variables in the `smoothed_saliency_maps` dictionary.\n",
      "Class 0: Min= nan    Max= nan\n",
      "Class 1: Min= nan    Max= nan\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "from tqdm import tqdm  # Optional, for progress visualization\n",
    "\n",
    "class SmoothedGradCAMPlusPlus(GradCAMPlusPlus):\n",
    "    def __init__(self, model, target_layer, num_samples=50, noise_level=0.1):\n",
    "        \"\"\"\n",
    "        Initialize the Smoothed-Grad-CAM++ class.\n",
    "\n",
    "        Args:\n",
    "            model (torch.nn.Module): The model to use for Smoothed-Grad-CAM++.\n",
    "            target_layer (torch.nn.Module): The layer to compute gradients for.\n",
    "            num_samples (int): Number of noisy samples to average.\n",
    "            noise_level (float): Standard deviation of noise to add.\n",
    "        \"\"\"\n",
    "        super().__init__(model, target_layer)\n",
    "        self.num_samples = num_samples\n",
    "        self.noise_level = noise_level\n",
    "\n",
    "    def generate(self, input_image, target_class):\n",
    "        \"\"\"\n",
    "        Generate Smoothed-Grad-CAM++ heatmap.\n",
    "\n",
    "        Args:\n",
    "            input_image (torch.Tensor): Input image tensor.\n",
    "            target_class (int): Class index for which to generate Smoothed-Grad-CAM++.\n",
    "\n",
    "        Returns:\n",
    "            numpy.ndarray: Smoothed-Grad-CAM++ heatmap.\n",
    "        \"\"\"\n",
    "        smoothed_heatmap = None\n",
    "\n",
    "        for _ in tqdm(range(self.num_samples), desc=\"Generating Smoothed Grad-CAM++\"):\n",
    "            # Add noise to the input image\n",
    "            noise = torch.normal(0, self.noise_level, size=input_image.size()).to(input_image.device)\n",
    "            noisy_input = input_image + noise\n",
    "            noisy_input = torch.clamp(noisy_input, 0, 1)  # Ensure valid image range\n",
    "\n",
    "            # Compute Grad-CAM++ for the noisy input\n",
    "            heatmap = super().generate(noisy_input, target_class)\n",
    "\n",
    "            # Accumulate the heatmaps\n",
    "            if smoothed_heatmap is None:\n",
    "                smoothed_heatmap = heatmap\n",
    "            else:\n",
    "                smoothed_heatmap += heatmap\n",
    "\n",
    "        # Average the heatmaps\n",
    "        smoothed_heatmap /= self.num_samples\n",
    "\n",
    "        return smoothed_heatmap\n",
    "\n",
    "\n",
    "# Instantiate Smoothed-Grad-CAM++\n",
    "num_samples = 50  # Number of noisy samples\n",
    "noise_level = 0.1  # Standard deviation of noise\n",
    "smoothed_grad_cam_pp = SmoothedGradCAMPlusPlus(model, target_layer, num_samples, noise_level)\n",
    "\n",
    "# Dictionary to store smoothed saliency maps as numpy arrays\n",
    "smoothed_saliency_maps = {}\n",
    "\n",
    "# Generate Smoothed-Grad-CAM++ for each class\n",
    "output = model(input_image)\n",
    "num_classes = output.size(1)\n",
    "\n",
    "for class_idx in range(num_classes):\n",
    "    smoothed_heatmap = smoothed_grad_cam_pp.generate(input_image, class_idx)\n",
    "    \n",
    "    # Store smoothed heatmap as numpy array\n",
    "    smoothed_saliency_maps[class_idx] = smoothed_heatmap\n",
    "    \n",
    "    # Visualize the smoothed heatmap\n",
    "    plt.imshow(smoothed_heatmap, cmap='jet', alpha=0.5)\n",
    "    plt.title(f\"Smoothed Grad-CAM++ for Class {class_idx}\")\n",
    "    plt.axis('off')\n",
    "    plt.show()\n",
    "\n",
    "# At this point, smoothed_saliency_maps is a dictionary containing numpy arrays\n",
    "# Example usage:\n",
    "smoothed_class_0_map = smoothed_saliency_maps[0]  # Numpy array for class 0\n",
    "smoothed_class_1_map = smoothed_saliency_maps[1]  # Numpy array for class 1\n",
    "\n",
    "# Print confirmation\n",
    "print(\"Smoothed Grad-CAM++ saliency maps are stored as variables in the `smoothed_saliency_maps` dictionary.\")\n",
    "print(\"Class 0: Min=\", np.min(smoothed_class_0_map), \"   Max=\", np.max(smoothed_class_0_map))\n",
    "print(\"Class 1: Min=\", np.min(smoothed_class_1_map), \"   Max=\", np.max(smoothed_class_1_map))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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ea9s5L3Az3sAgMrHK/0AwOLS09Pd3j1Z/HkCxW8TUnznZcm2ubm5bv+HvSxNmjTRoEGD9OqrrzpCZkknT54s95hDhw5V3bp1NWfOHF26dMlpX+nzGz58uBo1aqS5c+dq06ZNFbrrU7oaPBcsWKA5c+aob9++Htu5u3aZmZnKyMhw23716tVOn3GxZcsWZWZmavjw4eWuMTg4WEOHDtX777/v9HYxOTk5WrZsmRITExUVFVXucSXprrvuUteuXTV79my355Kfn68nn3zSa202m81x96Ykfffdd1q9erVTuzNnzrj0LV6MKygokCSdPn3aaX9ISIg6deokY4wuX77ssYbBgwerQYMGWrRokdP2RYsWqU6dOkpJSfHYFwAAALA68h5573rOewCAyscr/QDA4qZMmaILFy7ojjvuUMeOHVVYWKgvvvhCK1asUKtWrfTAAw9IuhqyQkJCdNttt+m///u/de7cOb322mtq0qSJ2yBXloULFyoxMVFdu3bVhAkT1KZNG+Xk5CgjI0Pff/+9du3aVa7xoqKiNH/+fP3yl79Unz59NG7cONWvX1+7du3ShQsX9MYbbzja1q5dW2PGjNHLL7+s4OBgjR07ttz1F/v1r39dZptbb71Vq1at0h133KGUlBRlZWXplVdeUadOnXTu3DmX9u3atVNiYqIeeughFRQUaMGCBWrYsKGmTZtWoRqfeeYZrVu3TomJiZo0aZJq1aqlV199VQUFBZo3b16FxpSuXsdVq1YpKSlJAwYM0M9//nPdeOONql27tvbu3atly5apfv36mj17ttv+KSkpeuGFFzRs2DCNGzdOJ06c0MKFC9WuXTvt3r3b0W7WrFnavHmzUlJSFB8frxMnTuhPf/qTmjdvrsTERElX52dsbKxuvPFGxcTE6JtvvtHLL7+slJQUrx8+Hx4ert/97ndKTU3V3XffreTkZH366ad6++23NXv2bDVo0KDC1wcAAADwN/Ieea+iakLek66+irV4vl2+fFm7d+/WM888I0m6/fbb1a1btwpfIwC4LhkAgKV98skn5he/+IXp2LGjiYyMNCEhIaZdu3ZmypQpJicnx6ntBx98YLp162bCwsJMq1atzNy5c83//d//GUkmKyvL0W7gwIFm4MCBjsdZWVlGklm8eLHTeIcPHzb33XefiY2NNbVr1zbNmjUzt956q1m5cqWj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",
      "text/plain": [
       "<Figure size 1800x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Values for both classes have been saved as numpy arrays.\n"
     ]
    }
   ],
   "source": [
    "# Plotting the SHAP values for both classes next to each other\n",
    "plt.figure(figsize=(18, 6))\n",
    "\n",
    "# Plot for Class 0 SHAP values\n",
    "plt.subplot(1, 2, 1)  # (rows, columns, index)\n",
    "plt.imshow(class_0_map, cmap='coolwarm')\n",
    "plt.title(\"Saliency Map for Class 0\")\n",
    "#plt.colorbar()\n",
    "\n",
    "# Plot for Class 1 SHAP values\n",
    "plt.subplot(1, 2, 2)  # (rows, columns, index)\n",
    "plt.imshow(class_0_map, cmap='coolwarm')\n",
    "plt.title(\"Saliency Map for Class 1\")\n",
    "#plt.colorbar()\n",
    "\n",
    "# Display the plots\n",
    "plt.tight_layout()  # Adjust layout for better visualization\n",
    "plt.show()\n",
    "\n",
    "# Saving the SHAP values per class as numpy arrays\n",
    "np.save(\"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results_explainer_comparison/smoothed_gradcam_class_0.npy\", smoothed_class_0_map)\n",
    "np.save(\"/home/tim/Documents/XX_Perturbation_SHAP/perturbation_shap/app/results_explainer_comparison/smoothed_gradcam_class_1.npy\", smoothed_class_1_map)\n",
    "\n",
    "print(\"Values for both classes have been saved as numpy arrays.\")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Eigen-CAM\n",
    "from https://github.com/shyhyawJou/EigenCAM-Pytorch/blob/main/cam.py\n",
    "\n",
    " relies on the principal components of the feature maps extracted from a specific layer during the forward pass, which are not class-specific by design. It highlights the dominant features across all classes for the input image."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "perturbation_shap",
   "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
