{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "89b4d75b-e5ef-4b47-8154-bb5274d4a3ab",
   "metadata": {},
   "source": [
    "# LOAD DATA FILE GENERATED BY STEP"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4ba64dc5-bbb9-4569-a576-0514d859bf48",
   "metadata": {},
   "source": [
    "## Import Libraries"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "24eeca47-0044-418b-b7a7-053b47a220fc",
   "metadata": {},
   "outputs": [],
   "source": [
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\", category=DeprecationWarning)\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "warnings.filterwarnings( \"ignore\", module = \"pandas\\..*\" )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "7e3914e7-d97b-4d9a-b4ee-e1497db113ab",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "WARNING:tensorflow:From C:\\Users\\XN2\\miniconda3\\envs\\ad\\lib\\site-packages\\keras\\src\\losses.py:2976: The name tf.losses.sparse_softmax_cross_entropy is deprecated. Please use tf.compat.v1.losses.sparse_softmax_cross_entropy instead.\n",
      "\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import datetime\n",
    "import importlib\n",
    "import utils as ut\n",
    "import numpy as np\n",
    "import pandas as pd"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "363755c1-825b-4969-8ed4-86db717f9f51",
   "metadata": {},
   "source": [
    "## Path Setting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "7ac70a15-32c3-40b5-a0f4-ff575a116d05",
   "metadata": {},
   "outputs": [],
   "source": [
    "submission_version = False\n",
    "\n",
    "if submission_version:\n",
    "    DS_path_main = '\\notebooks\\dataset\\mvtec_anomaly_detection'\n",
    "else:\n",
    "    DS_path_main = 'D:\\DS\\AD\\mvtec_anomaly_detection'\n",
    "\n",
    "DS_name = 'hazelnut'\n",
    "DS_path = os.path.join(DS_path_main,DS_name)\n",
    "codes_path = os.getcwd()\n",
    "results_path=os.path.join(codes_path,'results')\n",
    "results_subs        = os.path.join(results_path,    'test_results')\n",
    "path_csv    = os.path.join(results_path,     'csv')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c74e530a-5ea9-4148-9db3-abcac924f5c7",
   "metadata": {},
   "source": [
    "# Default Parameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "879d0d25-3a9c-42cb-bb69-b49c01af0f64",
   "metadata": {},
   "outputs": [],
   "source": [
    "epochs = 30000\n",
    "num_explained_classes = 1\n",
    "batch_size = 16\n",
    "image_size=[128, 128]\n",
    "latent_dim=32\n",
    "mask_type          = 'blend_2'\n",
    "postfix_csv = f'{DS_name}_{epochs}_9'"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2c076d88-e472-405a-bc4b-cf2367d4a444",
   "metadata": {},
   "source": [
    "# Read Pre-Computed CSV "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "62216961-9677-439e-854a-63b69456fc52",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "E:\\Cloud\\Shared\\XAI\\ICLR_Supplementary\\notebooks\\E6_XAD\\results\\csv/testresults_hazelnut_30000_9_BPT.csv  ***  True\n"
     ]
    }
   ],
   "source": [
    "csv_filename = f'{path_csv}/testresults_{postfix_csv}_BPT.csv'\n",
    "print(csv_filename,' *** ',os.path.exists(csv_filename))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "15244e68-f808-49da-ae46-2f84c8034445",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Unnamed: 0</th>\n",
       "      <th>a_type_id</th>\n",
       "      <th>a_type</th>\n",
       "      <th>img_no</th>\n",
       "      <th>mask_type</th>\n",
       "      <th>target_segs</th>\n",
       "      <th>num_samples</th>\n",
       "      <th>score_function</th>\n",
       "      <th>methods</th>\n",
       "      <th>aucI_pred</th>\n",
       "      <th>...</th>\n",
       "      <th>max_IoU</th>\n",
       "      <th>IoU_mean</th>\n",
       "      <th>IoU_std</th>\n",
       "      <th>au_IoU</th>\n",
       "      <th>anomaly_score</th>\n",
       "      <th>anomaly_score_val</th>\n",
       "      <th>AM_roc_score</th>\n",
       "      <th>auroc</th>\n",
       "      <th>auroc_gs1</th>\n",
       "      <th>auroc_gs2</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>crack</td>\n",
       "      <td>0</td>\n",
       "      <td>blend_2</td>\n",
       "      <td>100</td>\n",
       "      <td>500</td>\n",
       "      <td>lm_p_loss</td>\n",
       "      <td>BPT-100</td>\n",
       "      <td>155.072594</td>\n",
       "      <td>...</td>\n",
       "      <td>0.891720</td>\n",
       "      <td>0.201748</td>\n",
       "      <td>0.179980</td>\n",
       "      <td>0.201746</td>\n",
       "      <td>0.552117</td>\n",
       "      <td>173.42245</td>\n",
       "      <td>0.949351</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>1</td>\n",
       "      <td>0</td>\n",
       "      <td>crack</td>\n",
       "      <td>0</td>\n",
       "      <td>blend_2</td>\n",
       "      <td>100</td>\n",
       "      <td>500</td>\n",
       "      <td>lm_p_loss</td>\n",
       "      <td>BPT-500</td>\n",
       "      <td>158.137002</td>\n",
       "      <td>...</td>\n",
       "      <td>0.784298</td>\n",
       "      <td>0.195690</td>\n",
       "      <td>0.164375</td>\n",
       "      <td>0.195688</td>\n",
       "      <td>0.552117</td>\n",
       "      <td>173.42245</td>\n",
       "      <td>0.949351</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>2</td>\n",
       "      <td>0</td>\n",
       "      <td>crack</td>\n",
       "      <td>0</td>\n",
       "      <td>blend_2</td>\n",
       "      <td>100</td>\n",
       "      <td>500</td>\n",
       "      <td>lm_p_loss</td>\n",
       "      <td>BPT-1000</td>\n",
       "      <td>158.733311</td>\n",
       "      <td>...</td>\n",
       "      <td>0.731544</td>\n",
       "      <td>0.193035</td>\n",
       "      <td>0.156335</td>\n",
       "      <td>0.193033</td>\n",
       "      <td>0.552117</td>\n",
       "      <td>173.42245</td>\n",
       "      <td>0.949351</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "      <td>0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>3 rows × 29 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "   Unnamed: 0  a_type_id a_type  img_no mask_type  target_segs  num_samples  \\\n",
       "0           0          0  crack       0   blend_2          100          500   \n",
       "1           1          0  crack       0   blend_2          100          500   \n",
       "2           2          0  crack       0   blend_2          100          500   \n",
       "\n",
       "  score_function   methods   aucI_pred  ...   max_IoU  IoU_mean   IoU_std  \\\n",
       "0      lm_p_loss   BPT-100  155.072594  ...  0.891720  0.201748  0.179980   \n",
       "1      lm_p_loss   BPT-500  158.137002  ...  0.784298  0.195690  0.164375   \n",
       "2      lm_p_loss  BPT-1000  158.733311  ...  0.731544  0.193035  0.156335   \n",
       "\n",
       "     au_IoU  anomaly_score  anomaly_score_val  AM_roc_score  auroc  auroc_gs1  \\\n",
       "0  0.201746       0.552117          173.42245      0.949351      0          0   \n",
       "1  0.195688       0.552117          173.42245      0.949351      0          0   \n",
       "2  0.193033       0.552117          173.42245      0.949351      0          0   \n",
       "\n",
       "   auroc_gs2  \n",
       "0          0  \n",
       "1          0  \n",
       "2          0  \n",
       "\n",
       "[3 rows x 29 columns]"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df = pd.read_csv(csv_filename)\n",
    "df.head(3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "c469e83a-d494-414a-90b1-7446bf90fe59",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total Time: \t\t 0:48:34.843569\n"
     ]
    }
   ],
   "source": [
    "print('Total Time: \\t\\t',str(datetime.timedelta(seconds=np.sum(df.time_total))))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "390cdb3d-b3eb-4a82-aa0c-5ce9bad22177",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "************************************************************************************************************************\n",
      "Classes in Test:\t['crack', 'cut', 'good', 'hole', 'print']\n",
      "Classes in GT:\t\t['crack', 'cut', 'hole', 'print']\n",
      "************************************************************************************************************************\n",
      "Classes in Test:\t['good', 'crack', 'cut', 'hole', 'print']\n",
      "Found 391 images belonging to 1 classes.\n",
      "Found 110 images belonging to 5 classes.\n",
      "Found 70 images belonging to 4 classes.\n"
     ]
    }
   ],
   "source": [
    "importlib.reload(ut)   \n",
    "train_dir,gtruth_dir,test_dir = ut.Data.get_subdata(DS_path=DS_path)\n",
    "anomaly_types    = sorted(os.listdir(test_dir))\n",
    "anomaly_types_gt = sorted(os.listdir(gtruth_dir))\n",
    "anomaly_types,anomaly_types_gt = ut.Data.get_anomaly_types(test_dir=test_dir,dataset=DS_name,gt_dir=gtruth_dir)\n",
    "image_counter_cls = []\n",
    "for i in range(len(anomaly_types)):\n",
    "    image_counter_cls.extend([[anomaly_types[i],len(os.listdir(os.path.join(test_dir,anomaly_types[i])))]])\n",
    "train_dir,gtruth_dir,test_dir = ut.Data.get_subdata(DS_path=DS_path)\n",
    "\n",
    "augmentation_target = 'custom' # 'medium', 'full' minimal , custom\n",
    "train_generator,test_generator, gtruth_generator= ut.Data2.load_data(train_dir = train_dir,\n",
    "                                                                     test_dir = test_dir,\n",
    "                                                                     gtruth_dir = gtruth_dir,\n",
    "                                                                     dataset=DS_name,\n",
    "                                                                     classes = anomaly_types,\n",
    "                                                                     augmentation_target=augmentation_target,\n",
    "                                                                    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "6c08308a-45f2-4dd4-814e-14b09567726e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'good': 40, 'crack': 18, 'cut': 17, 'hole': 18, 'print': 17}\n",
      "cut 17\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "(['good', 'crack', 'cut', 'hole', 'print'], [40, 18, 17, 18, 17])"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "image_counter = {class_name: len(os.listdir(os.path.join(test_dir, class_name))) for class_name in test_generator.class_indices}\n",
    "print(image_counter)\n",
    "\n",
    "min_image_count = min(image_counter.items(), key=lambda x: x[1])[1]\n",
    "min_image_class = min(image_counter.items(), key=lambda x: x[1])[0]\n",
    "print(min_image_class,min_image_count)\n",
    "from collections import Counter\n",
    "Counter(test_generator.classes)\n",
    "lbls_cls = list(image_counter.keys())\n",
    "count_cls = list(image_counter.values())\n",
    "lbls_cls,count_cls"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c5f0edd4-5519-4fba-8f9c-4a98a396161c",
   "metadata": {},
   "source": [
    "# Read Pre-Computed CSV File for Anomaly Scores"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "5b50a0c6-4607-4bd8-b5e9-54d6da8f5f46",
   "metadata": {},
   "outputs": [],
   "source": [
    "postfix_csv = f'{DS_name}_{epochs}_AS'\n",
    "ascsv_filename = f'{path_csv}/testresults_{postfix_csv}.csv'\n",
    "df_as= pd.read_csv(ascsv_filename)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "7d3a6206-c3a7-45c0-8908-483b6aa472e0",
   "metadata": {},
   "outputs": [],
   "source": [
    "anomaly_scores = list(df_as[['anomaly_score_new', 'good_or_bad', 'a_type_id', 'img_no', 'a_type']].itertuples(index=False, name=None))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f1f7a1ab-0f56-4495-b7c5-18d78ab2b83a",
   "metadata": {},
   "source": [
    "## FIND Delta Opt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "bad129e3-01a9-4b27-8697-796fcaf011dc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "36 (0.36697295, False, 0, 36, 'good')\n",
      "delta_opt\t:\t0.36698295000000003\n"
     ]
    }
   ],
   "source": [
    "delta_opt = ut.Evaluate.find_optimal_separation_threshold(anomaly_scores)\n",
    "print(f'delta_opt\\t:\\t{delta_opt}')"
   ]
  },
  {
   "cell_type": "raw",
   "id": "1b7d5da0-3e71-419a-ab86-ef64385a8040",
   "metadata": {},
   "source": [
    "# Create HTML File for HeatMaps"
   ]
  },
  {
   "cell_type": "raw",
   "id": "1de29533-37e9-40f2-a93b-4faa475394b1",
   "metadata": {},
   "source": [
    "img_types = 'heatmaps'\n",
    "\n",
    "html_f_name = f'{results_path}/imgs_{DS_name}_{img_types}.html'\n",
    "\n",
    "header = '<html><body>'\n",
    "style = \"\"\"\n",
    "<style>\n",
    "    .dotg {height: 30px; width: 30px; background-color: #008000; border-radius: 50%; display: inline-block;}\n",
    "    .dotr {height: 30px; width: 30px; background-color: #ff0000; border-radius: 50%; display: inline-block;}\n",
    "    table {border: 1px solid black; width: 100%; border-collapse: collapse;}\n",
    "    th, td {font-size: 4em; border: 2px solid black; text-align: center; width: 0.13%;} /* Adjust width here */\n",
    "    th {background-color: #f2f2f2; position: -webkit-sticky; position: sticky; top: 0; z-index: 1;} /* Sticky header */\n",
    "    img figcaption {font-size: 4em; border: 1px dotted blue; text-align: left;}\n",
    "</style>\n",
    "\"\"\"\n",
    "\n",
    "# Method names (example)\n",
    "method_names = ['Input', 'Recons', 'Anom Map'] + list(np.unique(df.methods)) + ['Gt']\n",
    "\n",
    "# Write the initial HTML content with style and table header\n",
    "with open(html_f_name, 'w') as f:\n",
    "    f.write(header)\n",
    "    f.write(style)\n",
    "    f.write(\"<table>\")\n",
    "    \n",
    "    # Write the table headers with method names\n",
    "    f.write(\"<tr>\")\n",
    "    for method_name in method_names:\n",
    "        f.write(f\"<th>{method_name}</th>\")\n",
    "    f.write(\"</tr>\")\n",
    "\n",
    "def get_relative_img_path(a_type, img_id):\n",
    "    return f'test_results/{a_type}/{img_id}_{img_types}_{a_type}_{img_id}_{epochs}_{mask_type}.png'\n",
    "\n",
    "# Generate the HTML content for images\n",
    "for a_type in anomaly_types:\n",
    "    for img_id in range(image_counter.get(a_type, 0)):\n",
    "        results_subpath_cls = os.path.join(results_subs, str(a_type))\n",
    "        local_img_path = os.path.abspath(f'{results_subpath_cls}/{img_id}_{img_types}_{a_type}_{img_id}_{epochs}_{mask_type}.png')\n",
    "        \n",
    "        asn = df_as[(df_as.a_type == a_type) & (df_as.img_no == img_id)].anomaly_score_new\n",
    "        asn_bool = asn > delta_opt\n",
    "        pred_type = 'Anomalous' if asn_bool.iloc[0] else 'Good'\n",
    "        \n",
    "        if a_type != 'good' and os.path.exists(local_img_path):\n",
    "            relative_img_path = get_relative_img_path(a_type, img_id)\n",
    "            dot_class = 'dotg' if asn_bool.iloc[0] else 'dotr'\n",
    "            \n",
    "            # Add a new row for each image\n",
    "            with open(html_f_name, 'a') as f:\n",
    "                f.write(\"<tr>\")\n",
    "                f.write(f\"<td colspan={len(method_names)}><img src='{relative_img_path}' width='100%'>\")\n",
    "                f.write(f\"<figcaption><span class='{dot_class}'></span>&nbsp;{a_type}-{img_id}:&nbsp;&nbsp;{asn.iloc[0]:0.4f}&nbsp;&nbsp;{asn_bool.iloc[0]}-{pred_type}</figcaption></td>\")\n",
    "                f.write(\"</tr>\")\n",
    "\n",
    "# Close the table and the HTML document\n",
    "with open(html_f_name, 'a') as f:\n",
    "    f.write(\"</table>\")\n",
    "    f.write('</body></html>')\n",
    "\n",
    "print('Data Written in File:', html_f_name)\n"
   ]
  },
  {
   "cell_type": "raw",
   "id": "55ea9850-5ec0-48b5-b8fd-da5698329e95",
   "metadata": {},
   "source": [
    "# Create HTML for IoU Plots"
   ]
  },
  {
   "cell_type": "raw",
   "id": "2a121153-20f7-4980-93c4-9a86567894e5",
   "metadata": {},
   "source": [
    "img_types = 'IoU'\n",
    "\n",
    "html_f_name = f'{results_path}/imgs_{DS_name}_{img_types}.html'\n",
    "\n",
    "header = '<html><body>'\n",
    "style = \"\"\"\n",
    "<style>\n",
    "    .dotg {height: 30px; width: 30px; background-color: #008000; border-radius: 50%; display: inline-block;}\n",
    "    .dotr {height: 30px; width: 30px; background-color: #ff0000; border-radius: 50%; display: inline-block;}\n",
    "    table {border: 1px solid black; width: 100%; border-collapse: collapse;}\n",
    "    th, td {font-size: 4em; border: 2px solid black; text-align: center; width: 0.13%;} /* Adjust width here */\n",
    "    th {background-color: #f2f2f2; position: -webkit-sticky; position: sticky; top: 0; z-index: 1;} /* Sticky header */\n",
    "    img figcaption {font-size: 4em; border: 1px dotted blue; text-align: left;}\n",
    "</style>\n",
    "\"\"\"\n",
    "\n",
    "method_names = ['Input', 'Recons', 'Anom Map'] + list(np.unique(df.methods)) + ['Gt']\n",
    "\n",
    "with open(html_f_name, 'w') as f:\n",
    "    f.write(header)\n",
    "    f.write(style)\n",
    "    f.write(\"<table>\")\n",
    "    \n",
    "    # Write the table headers with method names\n",
    "    f.write(\"<tr>\")\n",
    "    for method_name in method_names:\n",
    "        f.write(f\"<th>{method_name}</th>\")\n",
    "    f.write(\"</tr>\")\n",
    "\n",
    "def get_relative_img_path(a_type, img_id):\n",
    "    return f'test_results/{a_type}/{img_id}_{img_types}_{a_type}_{img_id}_{epochs}_{mask_type}.png'\n",
    "\n",
    "for a_type in anomaly_types:\n",
    "    for img_id in range(image_counter.get(a_type, 0)):\n",
    "        results_subpath_cls = os.path.join(results_subs, str(a_type))\n",
    "        local_img_path = os.path.abspath(f'{results_subpath_cls}/{img_id}_{img_types}_{a_type}_{img_id}_{epochs}_{mask_type}.png')\n",
    "        \n",
    "        asn = df_as[(df_as.a_type == a_type) & (df_as.img_no == img_id)].anomaly_score_new\n",
    "        asn_bool = asn > delta_opt\n",
    "        pred_type = 'Anomalous' if asn_bool.iloc[0] else 'Good'\n",
    "        \n",
    "        if a_type != 'good' and os.path.exists(local_img_path):\n",
    "            relative_img_path = get_relative_img_path(a_type, img_id)\n",
    "            dot_class = 'dotg' if asn_bool.iloc[0] else 'dotr'\n",
    "\n",
    "            with open(html_f_name, 'a') as f:\n",
    "                f.write(\"<tr>\")\n",
    "                f.write(f\"<td colspan={len(method_names)}><img src='{relative_img_path}' width='80%'>\")\n",
    "                f.write(f\"<figcaption><span class='{dot_class}'></span>&nbsp;{a_type}-{img_id}:&nbsp;&nbsp;{asn.iloc[0]:0.4f}&nbsp;&nbsp;{asn_bool.iloc[0]}-{pred_type}</figcaption></td>\")\n",
    "                f.write(\"</tr>\")\n",
    "\n",
    "with open(html_f_name, 'a') as f:\n",
    "    f.write(\"</table>\")\n",
    "    f.write('</body></html>')\n",
    "\n",
    "print('Data Written in File:', html_f_name)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "574f71af-de63-491a-a20e-fb0195f2480a",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "ccd18566-49b9-47d2-996c-f8b9f7739cd8",
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_html(img_types, results_path, DS_name, anomaly_types, df_as, df, delta_opt, epochs, mask_type, image_counter, results_subs):\n",
    "    html_f_name = f'{results_path}/imgs_{DS_name}_{img_types}.html'\n",
    "    \n",
    "    header = '<html><body>'\n",
    "    style = \"\"\"\n",
    "    <style>\n",
    "        .dotg {height: 30px; width: 30px; background-color: #008000; border-radius: 50%; display: inline-block;}\n",
    "        .dotr {height: 30px; width: 30px; background-color: #ff0000; border-radius: 50%; display: inline-block;}\n",
    "        table {border: 1px solid black; width: 100%; border-collapse: collapse;}\n",
    "        th, td {font-size: 2em; border: 2px solid black; text-align: center; width: 0.13%;} /* Adjust width here */\n",
    "        th {background-color: #f2f2f2; position: -webkit-sticky; position: sticky; top: 0; z-index: 1;} /* Sticky header */\n",
    "        figcaption {font-size: 1em; border: 1px dotted blue; text-align: left;}\n",
    "    </style>\n",
    "    \"\"\"\n",
    "    \n",
    "    method_names = ['Input', 'Recons', 'Anom Map'] + list(np.unique(df.methods)) + ['Gt']\n",
    "    \n",
    "    def get_relative_img_path(a_type, img_id):\n",
    "        return f'test_results/{a_type}/{img_id}_{img_types}_{a_type}_{img_id}_{epochs}_{mask_type}.png'\n",
    "\n",
    "    with open(html_f_name, 'w') as f:\n",
    "        f.write(header)\n",
    "        f.write(style)\n",
    "        f.write(\"<table>\")\n",
    "        \n",
    "\n",
    "        f.write(\"<tr>\")\n",
    "        for method_name in method_names:\n",
    "            f.write(f\"<th>{method_name}</th>\")\n",
    "        f.write(\"</tr>\")\n",
    "        \n",
    "        for a_type in anomaly_types:\n",
    "            for img_id in range(image_counter.get(a_type, 0)):\n",
    "                results_subpath_cls = os.path.join(results_subs, str(a_type))\n",
    "                local_img_path = os.path.abspath(f'{results_subpath_cls}/{img_id}_{img_types}_{a_type}_{img_id}_{epochs}_{mask_type}.png')\n",
    "                \n",
    "                asn = df_as[(df_as.a_type == a_type) & (df_as.img_no == img_id)].anomaly_score_new\n",
    "                asn_bool = asn > delta_opt\n",
    "                pred_type = 'Anomalous' if asn_bool.iloc[0] else 'Good'\n",
    "                \n",
    "                if a_type != 'good' and os.path.exists(local_img_path):\n",
    "                    relative_img_path = get_relative_img_path(a_type, img_id)\n",
    "                    dot_class = 'dotg' if asn_bool.iloc[0] else 'dotr'\n",
    "                    \n",
    "                    # Add a new row for each image\n",
    "                    f.write(\"<tr>\")\n",
    "                    f.write(f\"<td colspan={len(method_names)}><img src='{relative_img_path}' width='100%'>\")\n",
    "                    f.write(f\"<figcaption><span class='{dot_class}'></span>&nbsp;{a_type}-{img_id}:&nbsp;&nbsp;{asn.iloc[0]:0.4f}&nbsp;&nbsp;{asn_bool.iloc[0]}-{pred_type}</figcaption></td>\")\n",
    "                    f.write(\"</tr>\")\n",
    "        \n",
    "        f.write(\"</table>\")\n",
    "        f.write('</body></html>')\n",
    "    \n",
    "    print('Data Written in File:', html_f_name)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "5a845dd6-cb87-4908-a826-942ff64fb8c9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Data Written in File: E:\\Cloud\\Shared\\XAI\\ICLR_Supplementary\\notebooks\\E6_XAD\\results/imgs_hazelnut_heatmaps.html\n"
     ]
    }
   ],
   "source": [
    "img_types = 'heatmaps'  # or 'IoU'\n",
    "generate_html(img_types, results_path, DS_name, anomaly_types, df_as, df, delta_opt, epochs, mask_type, image_counter, results_subs)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "a5289ceb-1af5-4f92-afed-ef52ba94dbfc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Data Written in File: E:\\Cloud\\Shared\\XAI\\ICLR_Supplementary\\notebooks\\E6_XAD\\results/imgs_hazelnut_IoU.html\n"
     ]
    }
   ],
   "source": [
    "img_types = 'IoU'  # or 'IoU'\n",
    "generate_html(img_types, results_path, DS_name, anomaly_types, df_as, df, delta_opt, epochs, mask_type, image_counter, results_subs)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2148067d-1ba9-41ae-9d3e-40f51907d3b4",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.18"
  },
  "widgets": {
   "application/vnd.jupyter.widget-state+json": {
    "state": {},
    "version_major": 2,
    "version_minor": 0
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
