{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4ce6f328",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "import sys\n",
    "sys.path.append(\"../../src\")\n",
    "import feature.scrna_dataset as scrna_dataset\n",
    "import model.sdes as sdes\n",
    "import model.generate as generate\n",
    "import model.scrna_ae as scrna_ae\n",
    "import model.util as model_util\n",
    "import analysis.fid as fid\n",
    "import torch\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.font_manager as font_manager\n",
    "import os\n",
    "import h5py"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "5f452f18",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Plotting defaults\n",
    "font_list = font_manager.findSystemFonts(fontpaths=[\"/home/anon/modules/fonts\"])\n",
    "for font in font_list:\n",
    "    font_manager.fontManager.addfont(font)\n",
    "plot_params = {\n",
    "    \"figure.titlesize\": 22,\n",
    "    \"axes.titlesize\": 22,\n",
    "    \"axes.labelsize\": 20,\n",
    "    \"legend.fontsize\": 18,\n",
    "    \"font.size\": 13,\n",
    "    \"xtick.labelsize\": 16,\n",
    "    \"ytick.labelsize\": 16,\n",
    "    \"font.family\": \"Roboto\",\n",
    "    \"font.weight\": \"bold\",\n",
    "    \"svg.fonttype\": \"none\"\n",
    "}\n",
    "plt.rcParams.update(plot_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "df22d86f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define device\n",
    "if torch.cuda.is_available():\n",
    "    DEVICE = \"cuda\"\n",
    "else:\n",
    "    DEVICE = \"cpu\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6dfb808b",
   "metadata": {},
   "source": [
    "### Define constants and paths"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "c770c048",
   "metadata": {},
   "outputs": [],
   "source": [
    "latent_space = False\n",
    "latent_dim = 200"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d5c1a69f",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_file = \"/data/anon/branched_diffusion/data/scrna/covid_flu/processed/covid_flu_processed_reduced_genes.h5\"\n",
    "autoencoder_path = \"/data/anon/branched_diffusion/models/trained_models/scrna_vaes/covid_flu/covid_flu_processed_reduced_genes_ldvae_d%d/\" % latent_dim\n",
    "\n",
    "models_base_path = \"/data/anon/branched_diffusion/models/trained_models/scrna_covid_flu_continuous_class_extension\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2cb7c2d0",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_before_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_branched_2classes/1/last_ckpt.pth\")\n",
    "branched_after_model_path = os.path.join(models_base_path, \"extension/1/last_ckpt.pth\")\n",
    "label_guided_before_model_path = os.path.join(models_base_path, \"scrna_covid_flu_continuous_labelguided_2classes/1/last_ckpt.pth\")\n",
    "label_guided_afterone_model_path = os.path.join(models_base_path, \"extension/2/last_ckpt.pth\")\n",
    "label_guided_afterall_model_path = os.path.join(models_base_path, \"extension/4/last_ckpt.pth\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "7ee483d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define the branches\n",
    "classes_01 = [0, 1]\n",
    "branch_defs_01 = [((0, 1), 0.5795795795795796, 1), ((0,), 0, 0.5795795795795796), ((1,), 0, 0.5795795795795796)]\n",
    "\n",
    "classes_012 = [0, 1, 5]\n",
    "branch_defs_012 = [((0, 1, 5), 6.786786786786787e-01, 1), ((0, 1), 0.5795795795795796, 0.6786786786786787), ((5,), 0, 0.6786786786786787), ((0,), 0, 0.5795795795795796), ((1,), 0, 0.5795795795795796)]\n",
    "\n",
    "classes_2 = [5]\n",
    "branch_defs_2 = [((5,), 0, 0.6786786786786787)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "0364c0a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "out_path = \"/home/anon/branched_diffusion/figures/scrna_covid_flu_class_extension\"\n",
    "\n",
    "os.makedirs(out_path, exist_ok=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c77b6452",
   "metadata": {},
   "source": [
    "### Create data loaders"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "98285ad2",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset_01 = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "dataset_012 = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "dataset_2 = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=(autoencoder_path if latent_space else None))\n",
    "\n",
    "# Limit classes\n",
    "inds_01 = np.isin(dataset_01.cell_cluster, classes_01)\n",
    "dataset_01.data = dataset_01.data[inds_01]\n",
    "dataset_01.cell_cluster = dataset_01.cell_cluster[inds_01]\n",
    "inds_012 = np.isin(dataset_012.cell_cluster, classes_012)\n",
    "dataset_012.data = dataset_012.data[inds_012]\n",
    "dataset_012.cell_cluster = dataset_012.cell_cluster[inds_012]\n",
    "inds_2 = np.isin(dataset_2.cell_cluster, classes_2)\n",
    "dataset_2.data = dataset_2.data[inds_2]\n",
    "dataset_2.cell_cluster = dataset_2.cell_cluster[inds_2]\n",
    "\n",
    "data_loader_01 = torch.utils.data.DataLoader(dataset_01, batch_size=128, shuffle=True, num_workers=0)\n",
    "data_loader_012 = torch.utils.data.DataLoader(dataset_012, batch_size=128, shuffle=True, num_workers=0)\n",
    "data_loader_2 = torch.utils.data.DataLoader(dataset_2, batch_size=128, shuffle=True, num_workers=0)\n",
    "input_shape = next(iter(data_loader_01))[0].shape[1:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "5be5f5ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "sde = sdes.VariancePreservingSDE(0.1, 5, input_shape)\n",
    "t_limit = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "51fbc829",
   "metadata": {},
   "outputs": [],
   "source": [
    "# TODO: this is currently rather inefficient; a decision-tree-style structure\n",
    "# would be better\n",
    "\n",
    "def class_time_to_branch(c, t, branch_defs):\n",
    "    \"\"\"\n",
    "    Given a class and a time (both scalars), return the\n",
    "    corresponding branch index.\n",
    "    \"\"\"\n",
    "    for i, branch_def in enumerate(branch_defs):\n",
    "        if c in branch_def[0] and t >= branch_def[1] and t <= branch_def[2]:\n",
    "            return i\n",
    "    raise ValueError(\"Undefined class and time\")\n",
    "        \n",
    "def class_time_to_branch_tensor(c, t, branch_defs):\n",
    "    \"\"\"\n",
    "    Given tensors of classes and a times, return the\n",
    "    corresponding branch indices as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.tensor([\n",
    "        class_time_to_branch(c_i, t_i, branch_defs) for c_i, t_i in zip(c, t)\n",
    "    ], device=DEVICE)\n",
    "\n",
    "def class_to_class_index_tensor(c, classes):\n",
    "    \"\"\"\n",
    "    Given a tensor of classes, return the corresponding class indices\n",
    "    as a tensor.\n",
    "    \"\"\"\n",
    "    return torch.argmax(\n",
    "        (c[:, None] == torch.tensor(classes, device=c.device)).int(), dim=1\n",
    "    ).to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c35fc1e8",
   "metadata": {},
   "source": [
    "### Import models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "40e98368",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_before_model = model_util.load_model(\n",
    "    scrna_ae.MultitaskResNet, branched_before_model_path\n",
    ").to(DEVICE)\n",
    "branched_after_model = model_util.load_model(\n",
    "    scrna_ae.MultitaskResNet, branched_after_model_path\n",
    ").to(DEVICE)\n",
    "\n",
    "label_guided_before_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_before_model_path\n",
    ").to(DEVICE)\n",
    "label_guided_afterone_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_afterone_model_path\n",
    ").to(DEVICE)\n",
    "label_guided_afterall_model = model_util.load_model(\n",
    "    scrna_ae.LabelGuidedResNet, label_guided_afterall_model_path\n",
    ").to(DEVICE)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0b38e89f",
   "metadata": {},
   "source": [
    "### Introducing a new digit class"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3738e63d",
   "metadata": {},
   "source": [
    "**Fine-tune new branch on branched model**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "4701b0f9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.54it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.57it/s]\n"
     ]
    }
   ],
   "source": [
    "# Samples before fine-tuning\n",
    "branched_before_samples = {}\n",
    "for class_to_sample in classes_01:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    samples = generate.generate_continuous_branched_samples(\n",
    "        branched_before_model, sde, class_to_sample,\n",
    "        lambda c, t: class_time_to_branch_tensor(c, t, branch_defs_01),\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "    ).cpu().numpy()\n",
    "    branched_before_samples[class_to_sample] = samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "f5283fb3",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.55it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.56it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:58<00:00,  8.57it/s]\n"
     ]
    }
   ],
   "source": [
    "# Samples after fine-tuning\n",
    "branched_after_samples = {}\n",
    "for class_to_sample in classes_012:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    samples = generate.generate_continuous_branched_samples(\n",
    "        branched_after_model, sde, class_to_sample,\n",
    "        lambda c, t: class_time_to_branch_tensor(c, t, branch_defs_012),\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "    ).cpu().numpy()\n",
    "    branched_after_samples[class_to_sample] = samples"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fe9b5ca7",
   "metadata": {},
   "source": [
    "#### Train additional digit on label-guided (linear) model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "a7395b41",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.71it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.71it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.71it/s]\n"
     ]
    }
   ],
   "source": [
    "label_guided_before_samples = {}\n",
    "for class_to_sample in classes_012:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    samples = generate.generate_continuous_label_guided_samples(\n",
    "        label_guided_before_model, sde, class_to_sample,\n",
    "        lambda c: class_to_class_index_tensor(c, classes_012),\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "    ).cpu().numpy()\n",
    "    label_guided_before_samples[class_to_sample] = samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "f5d9ec71",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.73it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.73it/s]\n"
     ]
    }
   ],
   "source": [
    "label_guided_afterone_samples = {}\n",
    "for class_to_sample in classes_012:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    samples = generate.generate_continuous_label_guided_samples(\n",
    "        label_guided_afterone_model, sde, class_to_sample,\n",
    "        lambda c: class_to_class_index_tensor(c, classes_012),\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "    ).cpu().numpy()\n",
    "    label_guided_afterone_samples[class_to_sample] = samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "8db51bf0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.73it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.72it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████████████████| 500/500 [00:57<00:00,  8.72it/s]\n"
     ]
    }
   ],
   "source": [
    "label_guided_afterall_samples = {}\n",
    "for class_to_sample in classes_012:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    samples = generate.generate_continuous_label_guided_samples(\n",
    "        label_guided_afterall_model, sde, class_to_sample,\n",
    "        lambda c: class_to_class_index_tensor(c, classes_012),\n",
    "        sampler=\"pc\", t_limit=t_limit, num_samples=1000, verbose=True\n",
    "    ).cpu().numpy()\n",
    "    label_guided_afterall_samples[class_to_sample] = samples"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e5f431ba",
   "metadata": {},
   "source": [
    "**Compare FIDs**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "c2b292e5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sampling class: 0\n",
      "Sampling class: 1\n",
      "Sampling class: 5\n"
     ]
    }
   ],
   "source": [
    "# Sample objects from the original dataset\n",
    "true_samples = {}\n",
    "for class_to_sample in classes_012:\n",
    "    print(\"Sampling class: %s\" % class_to_sample)\n",
    "    inds = np.where(dataset_012.cell_cluster == class_to_sample)[0]\n",
    "    sample_inds = np.random.choice(inds, size=1000, replace=False)\n",
    "    true_samples[class_to_sample] = dataset_012.data[sample_inds]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "988af0c2",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "[rank: 0] Global seed set to 0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[34mINFO    \u001b[0m File                                                                                                      \n",
      "         \u001b[35m/data/anon/branched_diffusion/models/trained_models/scrna_vaes/covid_flu/covid_flu_proc\u001b[0m\n",
      "         \u001b[35messed_reduced_genes_ldvae_d200/\u001b[0m\u001b[95mmodel.pt\u001b[0m already downloaded                                                \n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "No GPU/TPU found, falling back to CPU. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n",
      "/home/anon/miniconda3/envs/scanpy/lib/python3.9/site-packages/scvi/data/fields/_layer_field.py:91: UserWarning: adata.X does not contain unnormalized count data. Are you sure this is what you want?\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "if not latent_space:\n",
    "    dataset_with_ae = scrna_dataset.SingleCellDataset(data_file, autoencoder_path=autoencoder_path)\n",
    "\n",
    "def compute_fid(gen_samples, true_samples, latent=True):\n",
    "    if latent_space:\n",
    "        if latent:\n",
    "            return fid.compute_fid(\n",
    "                gen_samples,\n",
    "                dataset_01.encode_batch(torch.tensor(true_samples, device=DEVICE)).cpu().numpy()\n",
    "            )\n",
    "        else:\n",
    "            return fid.compute_fid(\n",
    "                dataset_01.decode_batch(torch.tensor(gen_samples, device=DEVICE)).cpu().numpy(),\n",
    "                true_samples\n",
    "            )\n",
    "    else:\n",
    "        gen_samples[gen_samples < 0] = 0  # Generated values should never be above 0\n",
    "        if latent:\n",
    "            return fid.compute_fid(\n",
    "                dataset_with_ae.encode_batch(torch.tensor(gen_samples, device=DEVICE)).cpu().numpy(),\n",
    "                dataset_with_ae.encode_batch(torch.tensor(true_samples, device=DEVICE)).cpu().numpy()\n",
    "            )\n",
    "        else:\n",
    "            return fid.compute_fid(\n",
    "                dataset_with_ae.decode_batch(dataset_with_ae.encode_batch(torch.tensor(gen_samples, device=DEVICE))).cpu().numpy(),\n",
    "                dataset_with_ae.decode_batch(dataset_with_ae.encode_batch(torch.tensor(true_samples, device=DEVICE))).cpu().numpy()\n",
    "            )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "3cf09210",
   "metadata": {},
   "outputs": [],
   "source": [
    "branched_before_fids = {}\n",
    "branched_after_fids = {}\n",
    "label_guided_before_fids = {}\n",
    "label_guided_afterone_fids = {}\n",
    "label_guided_afterall_fids = {}\n",
    "\n",
    "latent = True\n",
    "\n",
    "for c in branched_before_samples.keys():\n",
    "    branched_before_fids[c] = compute_fid(branched_before_samples[c], true_samples[c], latent)\n",
    "for c in branched_after_samples.keys():\n",
    "    branched_after_fids[c] = compute_fid(branched_after_samples[c], true_samples[c], latent)\n",
    "for c in label_guided_before_samples.keys():\n",
    "    label_guided_before_fids[c] = compute_fid(label_guided_before_samples[c], true_samples[c], latent)\n",
    "for c in label_guided_afterone_samples.keys():\n",
    "    label_guided_afterone_fids[c] = compute_fid(label_guided_afterone_samples[c], true_samples[c], latent)\n",
    "for c in label_guided_afterall_samples.keys():\n",
    "    label_guided_afterall_fids[c] = compute_fid(label_guided_afterall_samples[c], true_samples[c], latent)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "c64ee930",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{0: 17.2739535824701, 1: 20.49201217272837}\n",
      "{0: 17.949118294077465, 1: 20.02476075887448, 5: 23.667188069263155}\n",
      "{0: 20.895702081753146, 1: 21.95368448451966, 5: 27.22452015744281}\n",
      "{0: 24.710472874965525, 1: 27.29852642512873, 5: 31.712231368373526}\n",
      "{0: 18.359188765377848, 1: 20.05407637633136, 5: 27.03074935550749}\n"
     ]
    }
   ],
   "source": [
    "print(branched_before_fids)\n",
    "print(branched_after_fids)\n",
    "print(label_guided_before_fids)\n",
    "print(label_guided_afterone_fids)\n",
    "print(label_guided_afterall_fids)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "3663d5e5",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(ncols=2, sharey=True, figsize=(10, 5))\n",
    "\n",
    "bar_width = 0.3\n",
    "x = np.arange(len(branched_after_fids))\n",
    "\n",
    "branched_before_vals = [branched_before_fids[c] for c in classes_01] + [0]\n",
    "branched_after_vals = [branched_after_fids[c] for c in classes_012]\n",
    "ax[0].bar(\n",
    "    x, branched_before_vals, bar_width, label=\"Before training on new class\", color=\"royalblue\"\n",
    ")\n",
    "ax[0].bar(\n",
    "    x + bar_width, branched_after_vals, bar_width, label=\"After training on new class\", color=\"slateblue\"\n",
    ")\n",
    "ax[0].set_title(\"Branched model\")\n",
    "\n",
    "label_guided_before_vals = [label_guided_before_fids[c] for c in classes_01] + [0]\n",
    "label_guided_afterone_vals = [label_guided_afterone_fids[c] for c in classes_012]\n",
    "label_guided_afterall_vals = [label_guided_afterall_fids[c] for c in classes_012]\n",
    "ax[1].bar(\n",
    "    x, label_guided_before_vals, bar_width, label=\"Before training on new class\", color=\"darkorange\"\n",
    ")\n",
    "ax[1].bar(\n",
    "    x + bar_width, label_guided_afterone_vals, bar_width, label=\"After training on new class\", color=\"goldenrod\"\n",
    ")\n",
    "ax[1].bar(\n",
    "    x + (2 * bar_width), label_guided_afterall_vals, bar_width, label=\"After training on all classes\", color=\"firebrick\"\n",
    ")\n",
    "ax[1].set_title(\"Label-guided (linear) model\")\n",
    "for i in range(2):\n",
    "    ax[i].set_xticks(x + bar_width, labels=[c for c in classes_012])\n",
    "    ax[i].set_xlabel(\"Class\")\n",
    "    ax[i].legend()\n",
    "ax[0].set_ylabel(\"FID\")\n",
    "fig.suptitle(\"Fréchet inception distance between true and generated objects\")\n",
    "fig.savefig(\n",
    "    os.path.join(out_path, \"scrna_class_extension_fid.svg\"),\n",
    "    format=\"svg\"\n",
    ")\n",
    "plt.show()"
   ]
  }
 ],
 "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
