{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "1d1f4b5b-4e05-4957-b92f-4ecee448c3cd",
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.metrics.pairwise import cosine_similarity, manhattan_distances\n",
    "from matplotlib.backends.backend_pdf import PdfPages\n",
    "from sklearn.metrics import roc_curve\n",
    "from joblib import Parallel, delayed\n",
    "from tqdm.notebook import tqdm\n",
    "from random import shuffle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "263621fb-e477-42ee-a743-4b4c18ef9c79",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import networkx as nx\n",
    "import seaborn as sns\n",
    "import numpy as np\n",
    "\n",
    "sns.set_theme()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d800b686-5c68-4643-9b1e-f081a91a44d3",
   "metadata": {
    "tags": []
   },
   "source": [
    "#### FUNCTIONS to load data "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "cc29b147-3e95-455d-ade5-8bec76b5866a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_embeddings(model):\n",
    "    return np.load(f'../RFW/embd_{model}.npy')\n",
    "def get_names():\n",
    "    return np.load('../RFW/x_names.npy')\n",
    "def get_ids():\n",
    "    return np.load('../RFW/x_ids.npy')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e0592605-e959-48d5-be25-66680eaedd6d",
   "metadata": {
    "tags": []
   },
   "source": [
    "#### FUNCTIONS of score comparisons"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "7883570f-a209-446e-b0f4-6504847241fc",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_impostors(id_label):\n",
    "    # Generate the index matrix of same-id comparisons\n",
    "    comparison_matrix = (id_label[:, None] == id_label[None, :]).astype(int)\n",
    "    genuines_matrix = comparison_matrix.copy()\n",
    "    genuines_matrix[np.tril_indices(genuines_matrix.shape[0])] = 0\n",
    "    # Invert index matrix to get diff-id comparisons matrix\n",
    "    impostors_matrix = (1 - comparison_matrix)\n",
    "    impostors_matrix[np.tril_indices(impostors_matrix.shape[0])] = 0\n",
    "    return impostors_matrix, genuines_matrix"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "19fb83be-d8ca-497b-b1c8-1c6c3a0a4e48",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_group_scores(scores, index_matrix, img_ids):\n",
    "    A = scores[img_ids][:,img_ids]\n",
    "    B = index_matrix[img_ids][:,img_ids]\n",
    "    return A[B == 1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "a57cc324-b7a2-4dde-8355-e744855e729f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_eer_and_fnmr(genuine_scores, impostor_scores, fmr_targets=[0.01, 0.001]):\n",
    "    y_true = np.concatenate([np.ones_like(genuine_scores), np.zeros_like(impostor_scores)])\n",
    "    y_scores = np.concatenate([genuine_scores, impostor_scores])\n",
    "\n",
    "    fpr, tpr, thresholds = roc_curve(y_true, y_scores)\n",
    "\n",
    "    # Compute EER (where FPR ~= 1 - TPR)\n",
    "    fnr = 1 - tpr\n",
    "    eer_idx = np.nanargmin(np.abs(fpr - fnr))\n",
    "    eer = (fpr[eer_idx] + fnr[eer_idx]) / 2\n",
    "    eer_threshold = thresholds[eer_idx]\n",
    "\n",
    "    print(f\"\\n=== EER: {eer:.4f} at threshold {eer_threshold:.4f} ===\")\n",
    "\n",
    "    # Compute FNMR @ target FMRs\n",
    "    for target in fmr_targets:\n",
    "        idx = np.searchsorted(fpr, target)\n",
    "        if idx >= len(fnr):\n",
    "            print(f\"FMR {target:.4f} is out of range.\")\n",
    "            continue\n",
    "        print(f\"FNMR @ FMR={target:.4f}: {fnr[idx]:.4f} (threshold={thresholds[idx]:.4f})\")\n",
    "\n",
    "    return eer, eer_threshold"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "08132cbb-1d4b-47fc-87cb-c2362532fb38",
   "metadata": {
    "tags": []
   },
   "source": [
    "### ALGORITHM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "3aaffe49-47ea-4e52-9b6c-a58a7203f094",
   "metadata": {},
   "outputs": [],
   "source": [
    "def inital_clusters(vectors, threshold=0.5):\n",
    "    sims = cosine_similarity(vectors)\n",
    "    np.fill_diagonal(sims, 0)\n",
    "    \n",
    "    locations = np.where(sims > threshold)\n",
    "    \n",
    "    leafs = set(range(len(vectors))) - set(np.unique(locations))\n",
    "    leafs = [[num] for num in list(leafs)]\n",
    "    \n",
    "    G = nx.Graph()\n",
    "    for a, b in zip(locations[0], locations[1]):\n",
    "        G.add_edge(a, b)\n",
    "    \n",
    "    clusters = [list(component) for component in nx.connected_components(G)]\n",
    "    \n",
    "    return clusters + leafs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8f6c591d-3b57-41d4-8e73-a80875144341",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_direction(id_labels, embeddings):\n",
    "    # Map labels to integers and get counts in one pass.\n",
    "    unique_labels, integer_labels, counts = np.unique(id_labels, return_inverse=True, return_counts=True)\n",
    "    # Compute per-sample weights: each sample gets weight = 1 / (number of times its label occurs).\n",
    "    weights = 1 / counts[integer_labels]\n",
    "    # Compute the weighted sum using @\n",
    "    v = weights.astype(np.float32) @ embeddings\n",
    "    # The sum of weights is equal to the number of unique labels, so we normalize by that\n",
    "    v /= unique_labels.size\n",
    "    # Normalize the vector v\n",
    "    return v / np.linalg.norm(v)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "630fa970-ffd3-4699-9899-7a5a0f790e25",
   "metadata": {},
   "outputs": [],
   "source": [
    "def grow(seed, threshold = 0.3):\n",
    "    elements = seed\n",
    "    for _ in range(1000): # 1000 means it did not converge \n",
    "        v = get_direction(ids[elements], embeddings[elements])\n",
    "        \n",
    "        proj = embeddings @ v\n",
    "\n",
    "        for i in np.argsort(proj)[::-1]:\n",
    "            if (i not in elements):\n",
    "                if proj[i] < threshold:\n",
    "                    return elements\n",
    "                elements = elements + [i]\n",
    "                break\n",
    "            \n",
    "    return elements"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e35f47a7-3548-4725-8b8d-5ea857d0151a",
   "metadata": {},
   "source": [
    "### CLUSTER FILTERING"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "aa31a44c-c3e3-4729-9e63-c4af5d6f263d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def biased_clusters(clusters, scores, impostors_matrix, label_id, threshold=0.1):\n",
    "    filtered = []\n",
    "    indices = []\n",
    "    for i, val in enumerate(clusters):\n",
    "        B = impostors_matrix[val][:,val]\n",
    "        if np.any(B != 0) & (np.unique(label_id[val]).shape[0] >= 20):\n",
    "            A = scores[val][:,val]\n",
    "            B = impostors_matrix[val][:,val]\n",
    "            a = np.mean(A[B == 1])\n",
    "            if (a > threshold):\n",
    "                filtered.append(val)\n",
    "                indices.append(i)\n",
    "                # print(f\"{i} CLUSTER AVG {a:.2f} |  {len(A[B == 1])} comparisons\")\n",
    "    return filtered, indices"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6e5223f1-9141-4d4e-b702-2d9ae6694148",
   "metadata": {},
   "source": [
    "### Load Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "dae6189b-4b9e-481e-9e7a-90631bcab077",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_name = \"partialFC\"\n",
    "# model_name = \"elasticface\"\n",
    "# model_name = \"cosface\"\n",
    "# model_name = \"arcface\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b4bfacf6-1794-4a87-81f1-5af338cc8db6",
   "metadata": {},
   "outputs": [],
   "source": [
    "embeddings = get_embeddings(model_name)\n",
    "names = get_names()\n",
    "ids = get_ids()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4c9978a3-8eab-4b82-9408-fcb0aef22c0e",
   "metadata": {},
   "source": [
    "### Compute scores"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "40aedec4-d7aa-4a99-a84a-7f7515eeab02",
   "metadata": {},
   "outputs": [],
   "source": [
    "scores = cosine_similarity(embeddings)\n",
    "impostors_matrix, genuines_matrix = get_impostors(ids)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "b63d7816-527e-4b35-b27c-b4b461471f65",
   "metadata": {},
   "source": [
    "### Compute initial clusters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "8dd874a6-67c6-4cd8-9f20-30884d8dbe3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "clusters = inital_clusters(embeddings)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "91e42d43-8ee2-428b-9685-f3c1d20a22b7",
   "metadata": {},
   "source": [
    "# LABELS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "533fe159-63d9-4d16-9af6-e9a056e312c0",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "df = pd.read_csv('labels_RFW.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "86d1166e-bf70-4df6-8db1-93ca8283d656",
   "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>File Name</th>\n",
       "      <th>Gender</th>\n",
       "      <th>Age</th>\n",
       "      <th>Skin Color</th>\n",
       "      <th>Ancestry</th>\n",
       "      <th>Hair Color</th>\n",
       "      <th>Bangs</th>\n",
       "      <th>Bald</th>\n",
       "      <th>Beard</th>\n",
       "      <th>Glasses</th>\n",
       "      <th>Headwear</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>African/m.03c31xs/m.03c31xs_0001.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>young</td>\n",
       "      <td>dark</td>\n",
       "      <td>black</td>\n",
       "      <td>blonde</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>stubble</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>African/m.03c31xs/m.03c31xs_0004.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>young</td>\n",
       "      <td>dark</td>\n",
       "      <td>black</td>\n",
       "      <td>black</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>stubble</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>African/m.03c31xs/m.03c31xs_0003.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>young</td>\n",
       "      <td>dark</td>\n",
       "      <td>black</td>\n",
       "      <td>black</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>African/m.03c31xs/m.03c31xs_0002.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>young</td>\n",
       "      <td>dark</td>\n",
       "      <td>black</td>\n",
       "      <td>black</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>full</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>African/m.01ng51t/m.01ng51t_0002.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>senior</td>\n",
       "      <td>dark</td>\n",
       "      <td>black</td>\n",
       "      <td>black</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>mustache</td>\n",
       "      <td>no</td>\n",
       "      <td>hat</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>40602</th>\n",
       "      <td>Indian/m.0fgsc6/m.0fgsc6_0002.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>middle-aged</td>\n",
       "      <td>medium</td>\n",
       "      <td>south_asian</td>\n",
       "      <td>black</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>40603</th>\n",
       "      <td>Indian/m.0h06p_/m.0h06p__0001.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>senior</td>\n",
       "      <td>medium</td>\n",
       "      <td>south_asian</td>\n",
       "      <td>gray</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>40604</th>\n",
       "      <td>Indian/m.0h06p_/m.0h06p__0003.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>senior</td>\n",
       "      <td>medium</td>\n",
       "      <td>south_asian</td>\n",
       "      <td>gray</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>40605</th>\n",
       "      <td>Indian/m.0h06p_/m.0h06p__0002.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>senior</td>\n",
       "      <td>medium</td>\n",
       "      <td>south_asian</td>\n",
       "      <td>gray</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>40606</th>\n",
       "      <td>Indian/m.0h06p_/m.0h06p__0004.jpg</td>\n",
       "      <td>male</td>\n",
       "      <td>middle-aged</td>\n",
       "      <td>medium</td>\n",
       "      <td>south_asian</td>\n",
       "      <td>gray</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "      <td>no</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>40607 rows × 11 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "                                  File Name Gender          Age Skin Color  \\\n",
       "0      African/m.03c31xs/m.03c31xs_0001.jpg   male        young       dark   \n",
       "1      African/m.03c31xs/m.03c31xs_0004.jpg   male        young       dark   \n",
       "2      African/m.03c31xs/m.03c31xs_0003.jpg   male        young       dark   \n",
       "3      African/m.03c31xs/m.03c31xs_0002.jpg   male        young       dark   \n",
       "4      African/m.01ng51t/m.01ng51t_0002.jpg   male       senior       dark   \n",
       "...                                     ...    ...          ...        ...   \n",
       "40602     Indian/m.0fgsc6/m.0fgsc6_0002.jpg   male  middle-aged     medium   \n",
       "40603     Indian/m.0h06p_/m.0h06p__0001.jpg   male       senior     medium   \n",
       "40604     Indian/m.0h06p_/m.0h06p__0003.jpg   male       senior     medium   \n",
       "40605     Indian/m.0h06p_/m.0h06p__0002.jpg   male       senior     medium   \n",
       "40606     Indian/m.0h06p_/m.0h06p__0004.jpg   male  middle-aged     medium   \n",
       "\n",
       "          Ancestry Hair Color Bangs Bald     Beard Glasses Headwear  \n",
       "0            black     blonde    no   no   stubble      no       no  \n",
       "1            black      black    no   no   stubble      no       no  \n",
       "2            black      black    no   no        no      no       no  \n",
       "3            black      black    no   no      full      no       no  \n",
       "4            black      black    no   no  mustache      no      hat  \n",
       "...            ...        ...   ...  ...       ...     ...      ...  \n",
       "40602  south_asian      black    no   no        no      no       no  \n",
       "40603  south_asian       gray    no   no        no      no       no  \n",
       "40604  south_asian       gray    no   no        no      no       no  \n",
       "40605  south_asian       gray    no   no        no      no       no  \n",
       "40606  south_asian       gray    no   no        no      no       no  \n",
       "\n",
       "[40607 rows x 11 columns]"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "a62b8e14-399e-4b9e-9ef1-c444d6edf1fc",
   "metadata": {},
   "outputs": [],
   "source": [
    "names = np.array([path.replace('RFW/data/', '') for path in names])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "a9c6b6e8-4141-4ae3-b26a-802c3b71a00b",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.spatial.distance import hamming"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "939ca1eb-ce71-44c1-8bbe-3769fcb999cb",
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.preprocessing import OneHotEncoder"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "3eaa9148-0f3c-4fb0-87fa-28aece4ac9dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "attributes = [\n",
    "    \"Gender\",\"Age\",\"Skin Color\",\"Ancestry\",\"Hair Color\",\n",
    "    \"Bangs\",\"Bald\",\"Beard\",\"Glasses\",\"Headwear\"\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "afe49a03-3bb0-43ae-b9a0-ecb28d707c94",
   "metadata": {},
   "outputs": [],
   "source": [
    "encoder = OneHotEncoder(sparse=False, handle_unknown=\"ignore\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "5675f055-bf4a-4d36-b85f-b9b7a23f7be7",
   "metadata": {},
   "outputs": [],
   "source": [
    "file_set = set(names)\n",
    "\n",
    "# Filter the DataFrame\n",
    "df = df[df['File Name'].isin(file_set)]\n",
    "df = df.reset_index(drop=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "72c20c27-06d6-429d-ae79-59ef86c3a5c5",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/mpcdf/soft/SLE_15/packages/x86_64/anaconda/3/2023.03/lib/python3.10/site-packages/sklearn/preprocessing/_encoders.py:828: FutureWarning: `sparse` was renamed to `sparse_output` in version 1.2 and will be removed in 1.4. `sparse_output` is ignored unless you leave `sparse` to its default value.\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "X_encoded = encoder.fit_transform(df[attributes])  # shape: (n_samples, n_features_encoded)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "0004efee-9123-47b6-89b4-09b447eac502",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(40602, 50)"
      ]
     },
     "execution_count": 38,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "X_encoded.shape"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3399e656-23eb-4b7b-8571-16dfe2605fbc",
   "metadata": {},
   "source": [
    "# EVALUATION"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "86f98a84-2a75-4ffd-9dff-028b6f38df96",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<style>#sk-container-id-2 {color: black;background-color: white;}#sk-container-id-2 pre{padding: 0;}#sk-container-id-2 div.sk-toggleable {background-color: white;}#sk-container-id-2 label.sk-toggleable__label {cursor: pointer;display: block;width: 100%;margin-bottom: 0;padding: 0.3em;box-sizing: border-box;text-align: center;}#sk-container-id-2 label.sk-toggleable__label-arrow:before {content: \"▸\";float: left;margin-right: 0.25em;color: #696969;}#sk-container-id-2 label.sk-toggleable__label-arrow:hover:before {color: black;}#sk-container-id-2 div.sk-estimator:hover label.sk-toggleable__label-arrow:before {color: black;}#sk-container-id-2 div.sk-toggleable__content {max-height: 0;max-width: 0;overflow: hidden;text-align: left;background-color: #f0f8ff;}#sk-container-id-2 div.sk-toggleable__content pre {margin: 0.2em;color: black;border-radius: 0.25em;background-color: #f0f8ff;}#sk-container-id-2 input.sk-toggleable__control:checked~div.sk-toggleable__content {max-height: 200px;max-width: 100%;overflow: auto;}#sk-container-id-2 input.sk-toggleable__control:checked~label.sk-toggleable__label-arrow:before {content: \"▾\";}#sk-container-id-2 div.sk-estimator input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-2 div.sk-label input.sk-toggleable__control:checked~label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-2 input.sk-hidden--visually {border: 0;clip: rect(1px 1px 1px 1px);clip: rect(1px, 1px, 1px, 1px);height: 1px;margin: -1px;overflow: hidden;padding: 0;position: absolute;width: 1px;}#sk-container-id-2 div.sk-estimator {font-family: monospace;background-color: #f0f8ff;border: 1px dotted black;border-radius: 0.25em;box-sizing: border-box;margin-bottom: 0.5em;}#sk-container-id-2 div.sk-estimator:hover {background-color: #d4ebff;}#sk-container-id-2 div.sk-parallel-item::after {content: \"\";width: 100%;border-bottom: 1px solid gray;flex-grow: 1;}#sk-container-id-2 div.sk-label:hover label.sk-toggleable__label {background-color: #d4ebff;}#sk-container-id-2 div.sk-serial::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: 0;}#sk-container-id-2 div.sk-serial {display: flex;flex-direction: column;align-items: center;background-color: white;padding-right: 0.2em;padding-left: 0.2em;position: relative;}#sk-container-id-2 div.sk-item {position: relative;z-index: 1;}#sk-container-id-2 div.sk-parallel {display: flex;align-items: stretch;justify-content: center;background-color: white;position: relative;}#sk-container-id-2 div.sk-item::before, #sk-container-id-2 div.sk-parallel-item::before {content: \"\";position: absolute;border-left: 1px solid gray;box-sizing: border-box;top: 0;bottom: 0;left: 50%;z-index: -1;}#sk-container-id-2 div.sk-parallel-item {display: flex;flex-direction: column;z-index: 1;position: relative;background-color: white;}#sk-container-id-2 div.sk-parallel-item:first-child::after {align-self: flex-end;width: 50%;}#sk-container-id-2 div.sk-parallel-item:last-child::after {align-self: flex-start;width: 50%;}#sk-container-id-2 div.sk-parallel-item:only-child::after {width: 0;}#sk-container-id-2 div.sk-dashed-wrapped {border: 1px dashed gray;margin: 0 0.4em 0.5em 0.4em;box-sizing: border-box;padding-bottom: 0.4em;background-color: white;}#sk-container-id-2 div.sk-label label {font-family: monospace;font-weight: bold;display: inline-block;line-height: 1.2em;}#sk-container-id-2 div.sk-label-container {text-align: center;}#sk-container-id-2 div.sk-container {/* jupyter's `normalize.less` sets `[hidden] { display: none; }` but bootstrap.min.css set `[hidden] { display: none !important; }` so we also need the `!important` here to be able to override the default hidden behavior on the sphinx rendered scikit-learn.org. See: https://github.com/scikit-learn/scikit-learn/issues/21755 */display: inline-block !important;position: relative;}#sk-container-id-2 div.sk-text-repr-fallback {display: none;}</style><div id=\"sk-container-id-2\" class=\"sk-top-container\"><div class=\"sk-text-repr-fallback\"><pre>NearestNeighbors(n_neighbors=100)</pre><b>In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook. <br />On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.</b></div><div class=\"sk-container\" hidden><div class=\"sk-item\"><div class=\"sk-estimator sk-toggleable\"><input class=\"sk-toggleable__control sk-hidden--visually\" id=\"sk-estimator-id-2\" type=\"checkbox\" checked><label for=\"sk-estimator-id-2\" class=\"sk-toggleable__label sk-toggleable__label-arrow\">NearestNeighbors</label><div class=\"sk-toggleable__content\"><pre>NearestNeighbors(n_neighbors=100)</pre></div></div></div></div></div>"
      ],
      "text/plain": [
       "NearestNeighbors(n_neighbors=100)"
      ]
     },
     "execution_count": 39,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from sklearn.neighbors import NearestNeighbors\n",
    "# Fit the nearest neighbors model to your embeddings\n",
    "neigh = NearestNeighbors(n_neighbors=100)\n",
    "neigh.fit(embeddings)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "92a4b48b-8320-4f4c-b0a4-7fdd70727ac4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.35 1.3296099290780141 0.19988679457971142 0.9219858156028369 0.0 48.0\n",
      "0.3 2.3551756013660796 0.6357207753469776 1.6026932463455281 0.5319644716046119 95.38068844807468\n"
     ]
    }
   ],
   "source": [
    "groups = clusters.copy()\n",
    "for threshold in [0.4, 0.35, 0.3, 0.25, 0.2]:\n",
    "    groups = Parallel(n_jobs=-1, backend=\"loky\")(\n",
    "            delayed(grow)(g, threshold)\n",
    "            for g in groups\n",
    "        )\n",
    "    biased_groups, indices = biased_clusters(groups, scores, impostors_matrix, ids, threshold=0.1)\n",
    "    \n",
    "    aggr = []\n",
    "    ksearch = []\n",
    "    avg_size = []\n",
    "    for i, cl in enumerate(biased_groups):\n",
    "        aggr.append(np.mean(manhattan_distances(X_encoded[cl])[np.triu_indices(len(X_encoded[cl]), k = 1)])/2)\n",
    "        distances, indices = neigh.kneighbors([embeddings[cl[0]]], n_neighbors=len(cl))\n",
    "        ksearch.append(np.mean(manhattan_distances(X_encoded[indices[0]])[np.triu_indices(len(cl), k = 1)])/2)\n",
    "        avg_size.append(len(X_encoded[cl]))\n",
    "    print(threshold, np.mean(ksearch), np.std(ksearch), np.mean(aggr), np.std(aggr), np.mean(avg_size))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "fc83729f-f28a-4073-bc3c-5b125f4c85a4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABJ8AAAGACAYAAAADNcOYAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy88F64QAAAACXBIWXMAAA9hAAAPYQGoP6dpAAB7h0lEQVR4nO3dd3iT1f//8VeSUkonlDIKWEDZG2SDMj+yEWXJUECqgCC4EeTjV0EBZYhYQBEV/LjYgiIoMsTBcKCAbGRToEKhpaUruX9/8CMSCyVpk6Ypz8d1cV3JOec+9zvJIRzeOfe5TYZhGAIAAAAAAAA8wOztAAAAAAAAAJB/kXwCAAAAAACAx5B8AgAAAAAAgMeQfAIAAAAAAIDHkHwCAAAAAACAx5B8AgAAAAAAgMeQfAIAAAAAAIDHkHwCAAAAAACAx5B8AgAAAAAAgMeQfAKQpdatW6ty5cqqUqWKGjZsqP79+2vx4sWy2Ww56nfr1q166623chzfsmXLVLlyZZ04cSLHffmCq59H5cqVVb16dbVu3VrPPPOMjhw54tDuxIkT9nb//lOzZk1JUqNGjTRq1Kgbnuv5559X3bp1PflyAADIEeYpedPatWvVv39/1atXT3Xr1tX999+vxYsXu9zPk08+6TCH2bp1a47icvXzOHr0qP3cBw4cyNG5gVudn7cDAJD31alTR6NHj9b58+e1adMmvfjii9q8ebOmT5+e7T63bdummJgYPf74426M9NZQp04djRkzRhkZGfrzzz8VExOjjRs3asWKFSpdurRD2759++ree+91KDObr/zuUL58eZ06deqG5zl16pTKly/v/hcAAIAbMU/JW959911NnTpVnTt31iOPPKICBQroxx9/1Nq1a9WzZ0+X+nriiSc0YMAA/fnnnxo/fryHIr6xdevWOTyuWLFirscA5BcknwDcVHBwsOrVqydJatu2rcqWLavXX39d7dq1U7t27bwc3a0nODhYderUkSTVr19fwcHBGjt2rJYvX64RI0Y4tI2MjLS3/bdy5crp+++/v+F5Tp06pdq1a7srbAAAPIJ5St5x6NAhzZgxQ507d9a0adPs5U2bNlVKSorL/ZUtW1Zly5ZVamqqO8N02rfffquoqCj5+/tr3bp1Gjp0qFfiAPIDLrsD4LLevXurQIEC+vzzz+1l58+f10svvaQOHTqoTp06qlu3rvr166cff/zR4diry+NjYmIkyWEp9bXL253t76pdu3ape/fuqlmzptq2bavPPvvMod6V/nbt2qXBgwerUaNGql27ttq3b6/XX389U7uTJ0/q2WefVZMmTVSjRg1169ZN69evv+H7dujQIR06dOiG9dl1NUEUFxfn0nHly5fXuXPnlJqaKsMwdOedd6p27drKyMiQzWbT6dOnVa5cObfHCwCAJzFPucIb85Rly5YpIyNDgwcPzlQXEBCQo/huxpX+bvZ5SFc+k+3bt6tFixa6++67tXPnTp05cybb8QG3OlY+AXBZcHCwypQpo927d9vLzp07p4SEBEVHR6tUqVLKyMjQl19+qUceeUSffPKJffVNTEyM0tLStHjxYi1ZskQLFy6091GyZEmX+7vqxRdf1JAhQ/TEE09o5cqV+r//+z+Fh4frnnvucam/5ORkDR48WJGRkZowYYJCQkJ05MiRTHsMxMbGqlevXipYsKCee+45RURE6Msvv9Tw4cP17rvvqnnz5pnet44dO0qS9u3bl+33/nquToQqVKiQqc5msykjI8OhzGKxyGQyqXz58jIMQ7GxsbJarUpOTpbFYtHBgwdVpEgRpaenc9kdAMDnME/x3jzljz/+kL+/v6pUqZJlu+zE587+bvZ5SNKGDRtks9l09913y9/fX++//77Wr1+vPn36uBQbgCtIPgHIlsKFC+vkyZP25xUrVnTYW8Fqtapx48Zav369li5dap80VatWTZLsl3vd6JIwZ/u7ql+/fvZf2e666y7t2bNHc+bMsU8inO3v0KFDunDhgp555hn7sU2aNMk00Zg5c6YuXryoVatWqWzZsvbzHjhwQDExMS5PmlyVkZEhq9WqQ4cOaerUqapfv/5191F444039MYbbziUPffccxo8eLB9VdOpU6d09uxZVaxYUf7+/tq5c6fuuOMOSSL5BADwScxTvDNPiYuLU+HChe37S96Iu+Nztb+bfR7SlUvuChUqpEaNGslsNis4OFjr1q0j+QRkE8knAG6zZMkSffLJJzp69KguXbpkL3f1crDs9Hf33Xc7PG/evLk+/PBDZWRkyM/Pz+n+oqKiFBQUpPfee08Wi0V169ZVuXLlZDKZHPr//vvvVadOHZUuXdphZVG9evX0ySefKC0tTf7+/g7HuGvF0w8//KDq1as7vNaYmJhMy9klqX//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fGmsMM/5hy+MLcm3xld2+MJYkXxjvDBW8gZfGCtS/h4vjBX3ygtjxevJp+DgYCUlJalatWr65ZdfJEnffPONJCk0NNSboQEAAAAAACCHvJ7+KleunI4ePepQdvToUQUEBKhkyZJeigoAAAAAAADu4PXkU+PGjbV582alpKTYyzZs2KBGjRrliaVhAAAAAAAAyD6vZ3f69u2rjIwMPfPMM9qyZYtef/11bd++XY888oi3QwMAAAAAAEAOef1ud5L022+/6dVXX9X+/ftVqlQpPf744+rcubO3wwIAAAAAAEAO5YnkEwAAAAAAAPInr192BwAAAAAAgPyL5BMAAAAAAAA8huQTAAAAAAAAPIbkEwAAAAAAADyG5BMAAAAAAAA8huQTAAAAAAAAPIbkUy6aN2+eKleurKeeespj53jrrbdUuXLl6/6Bb0hNTdWMGTPUunVr1axZU23bttXUqVM9ci7GS+7L6j1PSEjQ+PHjvfI5LFu2TN9+++1N2z3//PN68MEHs30expz7/fs9bd68uYYNG6YDBw549Jx79uy5abvWrVvrrbfeytY51qxZ4/C6tm7desNYnBlTufndeitijgNnMMfJ/5jnMO7ciTlO/prj+Hk7gFvJ999/L0n68ccfZbVaZbFYPHKegIAALViwwCN9w/NGjhypn3/+WSNHjlTlypV18OBBLV68WM8884xHzsd4yX03es+DgoL08MMPq2vXrtq4caPmzJmTazEtX75cpUuXVtu2bbNs99hjjyktLS1H52LMud/V99QwDB09elQzZ85U//79tWrVKkVERLj9fDExMSpdurSqVq1603bh4eHZOkfjxo21cOFCxcXFacSIEVm2dWZM5fZ3662GOQ6cwRzn1sA8h3HnTsxx8s8ch+RTLklKStKvv/6qu+66S99//7127NihunXreuRcZrNZderU8Ujf8Kw1a9Zo48aNmjFjhjp06CBJatKkiXr16uWxczJecl9W73mZMmVUpkwZ/fXXX7kblJOioqJy3Adjzv2ufU/r1q2rMmXKqF+/flq5cqUefvhhr8VVrVq1bB9buHBh1alTRydOnLhp25uNKW98t95KmOPAGcxxbh3Mcxh37sQcJ//McbjsLpds3rxZ6enpevzxx1WwYEH7L4TXqly5st577z3NnTtXzZs3V+3atTVgwACHNnv27NEjjzyiO++8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o40bN2rFihUqXbq0t8NzWkhIiFavXq2KFStKktauXatChQopPT3dy5FJTzzxhAYMGKA///xT48eP93Y4AADcFHOcvIM5DgCST8AtLjIyUnXq1PF2GDkSHBxsfw3169dXcHCwxo4dq+XLl2vEiBHeDc4FrVu31urVqzVy5EhJ0urVq9WmTRstX77cy5FJZcuWVdmyZZWamurtUAAAcApznLyDOQ4A30r9A8h158+f10svvaQOHTqoTp06qlu3rvr166cff/zxuu3Xr1+v/v37q169eqpXr54eeOABrV271qHNyZMn9eyzz6pJkyaqUaOGunXrpvXr17st5tq1a0uS4uLisn1eZ16HuzVt2lTnzp3T3r17FR8fr23btqldu3aZ2u3atUsPP/yw6tatq9q1a6tfv376+eefM7X79ttv1blzZ9WoUUMdO3bUpk2brnteT38eAADkRcxxmOMAyD0kn4BbnM1mU0ZGhsMfwzDs9efOnVNCQoKio6M1Z84czZw5U2XKlNEjjzyi33//3aGvTz75RMOGDVNgYKBeffVVvfHGG2revLk2bNhgbxMbG6tevXrp119/1XPPPac5c+aocuXKGj58uH744Qe3vKYzZ85IkipUqJCt8zrzOjyhQIECatOmjVavXq21a9eqTp06ioiIcGhz9OhRPfjgg4qLi9OkSZM0bdo0Wa1WDRo0SLt27bK327Fjhx5//HGVLFlSs2bN0kMPPaSxY8dmOmdufB4AAHgDcxzmOMxxgLyDy+6AW9wbb7yhN954w6Hsueee0+DBgyVJFStW1PTp0+11VqtVjRs31vr167V06VL7UvBLly5p6tSpatKkiebOnWtv36JFC2VkZNifz5w5UxcvXtSqVatUtmxZSdJdd92lAwcOKCYmRs2bN8/W68jIyJDVatWhQ4c0depU1a9fXz179nT5vM6+Dk9p3769Jk6cqMjISLVv3z5T/YIFC5Samqp33nlHpUqVkiQ1aNBALVq00Lvvvqs333xTkjR37lyFhoZq1qxZKliwoKQr+1z897//dejPU58HAADexhyHOQ5zHCDvIPkE3OL69++vbt26OZRFRkY6PF+yZIk++eQTHT16VJcuXbKXX7vke/v27UpKSlL37t0zncPP75+vmu+//1516tRR6dKlHSY69erV0yeffKK0tDT5+/u79Bp++OEHVa9e3f68efPmiomJUUBAgMvndfZ1eErTpk0VHx+vEydOaOrUqYqNjXWo//3331WhQgX7pEySwsLCVLduXf3xxx/2sj///FN33nmnfVImXZlc/psnPg8AAPIC5jjMcZjjAHkHySfgFleiRIksbzv8wQcfaPLkyerZs6eeeOIJFSlSRJL06KOPOtyhJD4+XpJUvHjxLM8XHx+vuLg4h4nUtS5duqTw8HCXXkPdunX1wgsvKDU1VWvXrtX8+fP1yiuv6NVXX3X5vM6+Dk8pUKCAHn74Yf3999+KiIjINDFLTEzMNHGWpCJFijhcInDu3DmFhYVlavNvnvg8AADIC5jjOJ6XOY4j5jhA7iL5BCBLK1euVIMGDfTKK6/YyzIyMpSQkODQ7uo/+lf3IriRIkWKqHLlynriiSeuWx8aGupyjEFBQfbJZf369XX+/HktXbpUvXv3Vq1atVw6r7Ovw5OGDh16w7qQkBD75PFa8fHxDu9dRESELl68mKnNv7n6eZhMpqxCBwDAZzDHyX3McYBbF8knAFkymUwqVKiQQ9mXX36ZaW+AunXrKjAwUEuXLlXXrl0d6jIyMuzLue+66y799NNPuv322xUUFOSRmJ955hl9/fXXmjlzpubNm+fSeZ19Hdc6dOiQJOmOO+5w46u4vtq1a2vhwoU6deqUfVn6xYsXtX37dt199932djVq1NDWrVuVmppqX5Z+vTvBuPp5FC5cWNKVOwQBAODLmONc/3VcizkOAHch+QQgSy1bttScOXM0d+5c1axZU7t27dKCBQsy/VoUHBysp59+WhMmTNCjjz6q+++/X0FBQdq1a5eOHTumSZMmSZJGjhypTZs2qX///nrooYdUqlQpnTt3Tr///ruSk5Mdfn3MrhIlSqh379768MMP9ccff6h27dpOn9fZ13Gtjh07SpL27duX49hv5qGHHtLy5cs1ZMgQDR8+XH5+fpo3b57S09MVHR1tbzdkyBCtXbtWw4cP14MPPqjTp08rJiYmU3+ufh7lypVTqVKl9O677yosLEyBgYEqXbq0ihUr5vHXDgCAOzHHYY5zLeY4gGeRfAKQpaFDh+ry5cv68MMPlZCQoGrVqikmJkbPPvtsprb9+/dXyZIl9cEHH2jMmDGSpEqVKtnvKiNd2ehzyZIlmjlzpqZNm6YLFy4oPDxc1apVU48ePdwW95AhQ7R48WK99dZbmjdvnkvndeZ1eEv58uX14Ycfavr06RozZoxsNpuqVaum9957z778XpKqV6+ut956SzNmzNDw4cN122236dVXX9Ujjzzi0J+rn4e/v7/efPNNjR8/3r4nxpgxYzRw4EBPv3QAANyKOQ5znGsxxwE8y2QYhuHtIAAAAAAAAJA/mb0dAAAAAAAAAPIvkk8AAAAAAADwGJJPAAAAAAAA8BiSTwAAAAAAAPAYkk8AAAAAAADwGJJPAAAAAAAA8BiSTwAAAAAAAPAYkk8AAAAAAADwGJJPAAAAAAAA8BiSTwAAAAAAAPAYkk8AAAAAAADwGJJPAAAAAAAA8Jj/B7/7OnrGuOAGAAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 1200x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "\n",
    "# Set the Seaborn style\n",
    "sns.set(style=\"darkgrid\")\n",
    "sns.set_style(\"darkgrid\", {\"axes.facecolor\": \".93\"})\n",
    "\n",
    "\n",
    "# Example data\n",
    "data = {\n",
    "    'Dataset': ['Database: RFW'] * 8 + ['Database: CelebA'] * 8,\n",
    "    'Algorithm': ['ArcFace', 'ArcFace', 'CosFace', 'CosFace', 'ElasticFace', 'ElasticFace', 'PartialFC', 'PartialFC'] * 2,\n",
    "    'Method': ['Baseline', 'Ours'] * 4 * 2,\n",
    "    'Mean': [\n",
    "        2.99, 2.11, 2.82, 1.84, 2.97, 2.10, 2.41, 1.76,\n",
    "        8.14, 7.19, 7.97, 6.88, 7.63, 6.89, 7.87, 6.96\n",
    "    ],\n",
    "    'Std': [\n",
    "        0.61, 0.66, 0.62, 0.63, 0.63, 0.66, 0.68, 0.60,\n",
    "        1.27, 1.31, 1.37, 1.34, 1.47, 1.31, 1.39, 1.33\n",
    "    ]\n",
    "}\n",
    "\n",
    "df = pd.DataFrame(data)\n",
    "\n",
    "# Pivot the data for easier plotting\n",
    "pivot_mean = df.pivot(index=['Dataset', 'Algorithm'], columns='Method', values='Mean').reset_index()\n",
    "pivot_std = df.pivot(index=['Dataset', 'Algorithm'], columns='Method', values='Std').reset_index()\n",
    "\n",
    "# Plotting parameters\n",
    "methods = ['Baseline', 'Ours']\n",
    "algorithms = ['ArcFace', 'CosFace', 'ElasticFace', 'PartialFC']\n",
    "datasets = ['Database: RFW', 'Database: CelebA']\n",
    "bar_width = 0.3\n",
    "x = np.arange(len(algorithms))\n",
    "with plt.rc_context(rc={\"font.family\": \"DejaVu Sans Mono\"}):\n",
    "    # your plotting code here\n",
    "\n",
    "    # Set up the figure\n",
    "    fig, axes = plt.subplots(1, 2, figsize=(12, 4), sharey=True)\n",
    "\n",
    "    for i, dataset in enumerate(datasets):\n",
    "        ax = axes[i]\n",
    "        \n",
    "        mean_subset = pivot_mean[pivot_mean['Dataset'] == dataset]\n",
    "        std_subset = pivot_std[pivot_std['Dataset'] == dataset]\n",
    "\n",
    "        baseline_scores = mean_subset['Baseline'].values\n",
    "        ours_scores = mean_subset['Ours'].values\n",
    "\n",
    "        baseline_err = std_subset['Baseline'].values\n",
    "        ours_err = std_subset['Ours'].values\n",
    "\n",
    "        # Plot bars\n",
    "        ax.bar(x - bar_width/2, baseline_scores, yerr=baseline_err, width=bar_width, label='Baseline', color='skyblue', capsize=3,)\n",
    "        ax.bar(x + bar_width/2, ours_scores, yerr=ours_err, width=bar_width, label='LFA (Ours)', color='mediumseagreen', capsize=3)\n",
    "\n",
    "        # Annotate above the top of the error bars\n",
    "        offset = .1  # vertical space above the error bar\n",
    "        x_offset = .11  # vertical space above the error bar\n",
    "        \n",
    "        \n",
    "        # Annotate\n",
    "        for j in range(len(x)):\n",
    "            # ax.text(x[j] - bar_width/2, baseline_scores[j] + baseline_err[j] + offset, f'{baseline_scores[j]:.1f}', ha='center', fontsize=9)\n",
    "            # ax.text(x[j] + bar_width/2, ours_scores[j] + ours_err[j] + offset, f'{ours_scores[j]:.1f}', ha='center', fontsize=9)\n",
    "            \n",
    "            ax.text(x[j] - bar_width/2 - x_offset, baseline_scores[j] + offset, f'{baseline_scores[j]:.1f}', ha='center', fontsize=9)\n",
    "            ax.text(x[j] + bar_width/2 + x_offset, ours_scores[j] + offset, f'{ours_scores[j]:.1f}', ha='center', fontsize=9)\n",
    "\n",
    "        ax.set_title(dataset)\n",
    "        ax.set_xticks(x)\n",
    "        ax.set_xticklabels(algorithms)\n",
    "        ax.set_xlabel(\"Face Rec. Model\")\n",
    "        if i == 0:\n",
    "            ax.set_ylabel(\"Avg Intra-Group Distance\")\n",
    "        # ax.set_ylim(0, 1)\n",
    "\n",
    "    for ax in axes:\n",
    "        ax.xaxis.grid(False)  # Just remove horizontal lines\n",
    "\n",
    "\n",
    "    # Legend only on one plot\n",
    "    axes[1].legend(loc='lower right')\n",
    "\n",
    "    plt.tight_layout()\n",
    "    plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "032a5063-ed18-40a8-8001-c1f9ca528e00",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
