{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5e395bc9",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import csv\n",
    "import numpy as np\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "9da0cb90",
   "metadata": {},
   "outputs": [],
   "source": [
    "def load_csv(path, n_runs, optimise_iter):\n",
    "\n",
    "    solution = np.zeros((n_runs, optimise_iter))\n",
    "    header = []\n",
    "\n",
    "    with open(path, 'r', newline='') as csvfile:\n",
    "        wandb_reader = csv.reader(csvfile, delimiter=',', quotechar='|')\n",
    "        for i, row in enumerate(wandb_reader):\n",
    "            if i==0:\n",
    "                continue\n",
    "            header.append(row[1])\n",
    "            solution[i-1,:] = [float(k) for k in row[2:]]\n",
    "            \n",
    "    return header, solution\n",
    "\n",
    "def load_csv_nan(path, n_runs, optimise_iter):\n",
    "\n",
    "    solution = np.zeros((n_runs, optimise_iter))\n",
    "    header = []\n",
    "\n",
    "    with open(path, 'r', newline='') as csvfile:\n",
    "        wandb_reader = csv.reader(csvfile, delimiter=',', quotechar='|')\n",
    "        for i, row in enumerate(wandb_reader):\n",
    "            if i==0:\n",
    "                continue\n",
    "            header.append(row[1])\n",
    "            solution[i-1,:] = [np.nan if k == \"\" else float(k) for k in row[2:]]\n",
    "        \n",
    "    return header, solution"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cb535e30",
   "metadata": {},
   "source": [
    "# IMC"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "a2333147",
   "metadata": {},
   "outputs": [],
   "source": [
    "optimise_iter = 35\n",
    "n_runs = 98\n",
    "path='meta_obj_csv/'\n",
    "\n",
    "_, imc_solution_msa = load_csv(path+'final_imc_manatee_solution_98_runs_35_iter.csv', n_runs, optimise_iter)\n",
    "_, imc_solution_mas = load_csv(path+'final_imc_manatee_ucb_scal_exhaust_latest_solution_98_runs_35_iter.csv', n_runs, optimise_iter)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "434b19b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy import stats\n",
    "\n",
    "def compute_modes(solutions, digits):\n",
    "    n_runs = solutions.shape[0]\n",
    "    modes, quantities = np.zeros(n_runs), np.zeros(n_runs)\n",
    "    for run in range(n_runs):\n",
    "        res = stats.mode(np.round(solutions[run,:], digits))\n",
    "        modes[run], quantities[run] = res.mode, res.count\n",
    "    return modes, quantities\n",
    "\n",
    "def make_df(array, method, y_label):\n",
    "    df = pd.DataFrame(array.T)\n",
    "    df['Method'] = method\n",
    "    df.columns = [y_label, 'Method']\n",
    "    return df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "a759229f",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2714263/3908738643.py:7: FutureWarning: Unlike other reduction functions (e.g. `skew`, `kurtosis`), the default behavior of `mode` typically preserves the axis it acts along. In SciPy 1.11.0, this behavior will change: the default value of `keepdims` will become False, the `axis` over which the statistic is taken will be eliminated, and the value None will no longer be accepted. Set `keepdims` to True or False to avoid this warning.\n",
      "  res = stats.mode(np.round(solutions[run,:], digits))\n"
     ]
    }
   ],
   "source": [
    "digits = 2\n",
    "imc_modes_msa, imc_quantities_msa = compute_modes(imc_solution_msa, digits)\n",
    "imc_modes_mas, imc_quantities_mas = compute_modes(imc_solution_mas, digits)\n",
    "\n",
    "imc_modes_msa_df = make_df(imc_modes_msa, \"M-SA\", \"Cofactor\")\n",
    "imc_modes_mas_df = make_df(imc_modes_mas, \"M-AS\", \"Cofactor\")\n",
    "\n",
    "to_concat = [imc_modes_mas_df]\n",
    "c = 0\n",
    "total_len = len(imc_modes_msa_df)\n",
    "\n",
    "for frame in to_concat:\n",
    "    frame.index = range(total_len*(c+1), total_len*(c+2))\n",
    "    c=c+1\n",
    "\n",
    "imc_df_all = pd.concat([imc_modes_msa_df, imc_modes_mas_df])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "ead566d8",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<AxesSubplot: xlabel='Cofactor', ylabel='Method'>"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAlgAAAG9CAYAAADJBwhwAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy89olMNAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAxeklEQVR4nO3deVyU9d7/8feAIqIoLqAiZVYObqCiUhmdjuKamQtoVmpqPs5x7VF2uq27xW6rY3ZXJ49L2uJy1DTANTWXsDL1KBqalku3ZhiiCOICBrJdvz/8MScCDZivDmOv5+PBw/F7fa/r+7m+w8y8meuaa2yWZVkCAACAMR6uLgAAAOBmQ8ACAAAwjIAFAABgGAELAADAMAIWAACAYQQsAAAAwwhYAAAAhhGwAAAADKvi6gL+qDp06KDc3Fz5+/u7uhQAAFBGaWlp8vLy0p49e67Zj4DlIpcvX1ZBQYGrywAAAOWQn5+vsnwJDgHLRQICAiRJ8fHxLq4EAACUVWRkZJn6cQ4WAACAYQQsAAAAwwhYAAAAhhGwAAAADCNgAQAAGEbAAgAAMIyABQAAYBgBCwAAwDACFgAAgGEELAAAAMMIWAAAAIYRsAAAAAwjYAEAABhGwAIAADCMgHWTsSxLlmW5ugwAAP7QCFg3EcuyNGnSJE2aNImQBQCAC1VxdQEw58KFCzp06JDjtp+fn2sLAgDgD4p3sAAAAAwjYAEAABhGwAIAADCMgAUAAGAYAQsAAMAwAhYAAIBhBCwAAADDCFgAAACGEbAAAAAMI2ABAAAYRsACAAAwjIAFAABgGAELAADAMAIWAACAYQQsAAAAwwhYAAAAhhGwAAAADCNgAQAAGEbAAgAAMIyABQAAYBgBCwAAwDACFgAAgGEELAAAAMMIWAAAAIYRsAAAAAwjYAEAABhGwAIAADCMgAUAAGAYAQsAAMAwAhYAAIBhBCwAAADDCFgAAACGEbAAAAAMI2ABAAAYRsACAAAwjIB1EyksLCz1NgAAuLEIWDeRzMzMUm8DAIAbi4AFAABgGAELAADAMAIWAACAYQQsAAAAwwhYAAAAhhGwAAAADCNgAQAAGEbAAgAAMIyABQAAYBgBCwAAwDACFgAAgGEELAAAAMMIWAAAAIYRsAAAAAwjYAEAABhGwAIAADCMgAUAAGAYAQsAAMAwAhYAAIBhBCwAAADDCFgAAACGEbAAAAAMI2ABAAAYRsACAAAwjIAFAABgGAELAADAMAIWAACAYQQsAAAAwwhYAAAAhhGwAAAADKt0AWvXrl0KDg5WcHCwJk2aVGofy7LUpUsXBQcHq2XLlmXabkFBgT799FM99thjioiIUOvWrRUREaHBgwfrzTffVEZGxlXXnTp1qoKDg9WlSxdZllWh/QIAAH8clS5gFalWrZo2btyorKysEsu2b9+ukydPqlq1amXe3rPPPqu//e1vyszM1JAhQzR58mQ98sgjatCggZYtW6aTJ0+Wul5ubq5WrVqlJk2a6OTJk9q+fXuF9wkAriUhIUEjR45UQkKCq0sB4KQqri7garp166a1a9dq7dq1Gjx4cLFlsbGxCgwMVGBgoPbu3fu72/ruu++0bt06hYSEaOnSpapatWqx5ZcuXbrqups3b9b58+c1Y8YMTZw4UTExMYqIiKjYTgHAVeTk5Gj27Nk6e/asZs+erdDQUHl7e7u6LAAVVGnfwbrjjjvUrl07xcXFFWvPyMhQfHy8BgwYIA+PspWflJQkSerYsWOJcCVJNWrUUI0aNUpdNyYmRrfddpvCw8P10EMPacuWLdc8nAgAFREXF+d4bsnIyCjx3AfAvVTagCVJAwcO1IEDB3TkyBFH2+rVq1VQUKCoqKgyb+eWW26RJH3xxRdKTU0t83onTpzQrl27NGDAAElSVFSU8vLytHLlyjJvAwB+T0pKiuLi4hzneFqWpbi4OKWkpLi4MgAVVakDVq9evVSjRo1if8nFxcWpU6dOCgwMLPN2QkND1blzZx0/flyRkZF67LHH9Oabb2rjxo26ePHiVdeLjY2VzWZTv379JF15V61t27aKjY2t8D4BwK9ZlqU5c+aU+ADN1doBuIdKHbB8fHzUu3dvrVmzRrm5uUpMTNTRo0cVHR1d7m3NmDFDL7/8slq3bq1vv/1WH330kZ588knde++9+t///V8VFBQU65+fn6+VK1fq3nvvVYMGDRztUVFROn78uPbs2eP0/gFAcnKy9u7dq8LCwmLthYWF2rt3r5KTk11UGQBnVOqAJUnR0dE6f/68Pv/8c8XGxqpOnTqKjIws0S8nJ0dpaWnFfn598nrVqlX12GOPadmyZUpMTFRMTIyefPJJ+fj46MMPP9SHH35YbHtffPGF0tLSdM899ygpKcnx07p1a1WtWlUxMTHXfd8B3PyCgoLUrl27EueUenh4KCwsTEFBQS6qDIAzKn3AatOmjex2uxYtWqQNGzaob9++8vLyKtFv/fr1ioiIKPYzb968Urfp5eWlNm3aaNy4cfr4449ls9lKnFBaFKDefPNNde/e3fHTv39/5eXl/e7hRQAoC5vNptGjR8tms5WpHYB7qLSXafi1qKgoTZ06VZKuengwIiJC8+fPL9ZWdHL7tdxxxx2qXbt2sZPfU1JStG3bNnXp0kV9+/Ytsc7PP/+st956y3HhUgBwRmBgoKKjoxUTEyPLsmSz2RQdHa1GjRq5ujQAFeQWAatfv37KzMyUr6+vmjVrVmqfgIAABQQElLosKSlJhYWFatq0aYllCQkJOn/+vFq1auVoW758uQoLCzVs2DDdc889JdYpKCjQ/PnzFRsbS8ACYER0dLQ+//xznT17VnXr1q3QuaYAKg+3CFh+fn6aMGFChdf/v//7P40fP14dOnRQeHi4AgMDdfnyZR0+fFiffvqpqlatqmeeeUbSlRNLly9frrp16yo8PLzU7Xl6eqpbt25atmyZDhw4oJCQkArXBgCS5O3trbFjx2rOnDkaPXo0FxkF3JxbBCxnhYWF6bnnntOOHTu0atUqnT17VoWFhQoICFCPHj00YsQINW/eXJL09ddf69SpUxo0aJA8PT2vus2ePXtq2bJlio2NJWABMCI8PPyqf9gBcC82i4usuETRJyHj4+ONbTMpKUnjx4+XJM2cOVNNmjQxtm0AAFD21+9K/ylCAAAAd0PAAgAAMIyABQAAYBgBCwAAwDACFgAAgGEELAAAAMMIWAAAAIYRsAAAAAwjYAEAABhGwAIAADCMgAUAAGAYAQsAAMAwAhYAAIBhBCwAAADDCFgAAACGEbAAAAAMI2ABAAAYRsACAAAwjIAFAABgGAELAADAMAIWAACAYQQsAAAAwwhYAAAAhhGwAAAADCNgAQAAGEbAAgAAMIyABQAAYBgBCwAAwDACFgAAgGEErJuIr69vqbcBAMCNRcC6iXh4eJR6GwAA3Fi8CgMAABhGwAIAADCMgAUAAGAYAQsAAMAwAhYAAIBhBCwAAADDCFgAAACGEbAAAAAMI2ABAAAYRsACAAAwjIAFAABgGAELAADAMAIWAACAYQQsAAAAwwhYAAAAhhGwAAAADCNgAQAAGEbAAgAAMIyABQAAYBgBCwAAwLAq5ek8c+bMCg1is9k0bty4Cq0LAADgbpwOWDabzXHbsqwS7ZZlEbAAAMAfSrkC1r/+9a8SbQsWLNDWrVvVp08fhYeHq379+kpPT9euXbu0du1a3X///Xr88ceNFQwAAFDZlStghYeHF/v/qlWrtGPHDn3yySdq1apVsWX9+/fXY489piFDhqhbt27OVwoAAOAmnDrJfcGCBerVq1eJcFUkJCREvXr10sKFC50ZBgAAwK04FbCOHz+ugICAa/YJCAjQ8ePHnRkGAADArTgVsGrWrKnExMRr9vnmm2/k4+PjzDAAAABuxamAdf/992vPnj2aNm2asrKyii3LysrSG2+8ocTERHXu3NmpIgEAANxJuU5y/61nnnlGCQkJWrBggWJjY9WiRQvVq1dPZ8+e1aFDh5SVlaVbbrlFEydONFUvAABApefUO1j16tVTbGysoqOjVVBQoN27d2vDhg3avXu3CgoKNGjQIMXExKhevXqm6gUAAKj0nHoHS5Lq1KmjV199VZMnT9aPP/6ozMxM+fr66vbbb1eVKk5vHgAAwO0YS0BVqlSR3W43tTkAAAC3ZSxg7dmzR4cOHdLFixfl6+urli1bqkOHDqY2DwAA4DacDljfffed/uu//stxraui7x6UpKZNm2ratGkKCQlxdhgAAAC34VTASkpK0vDhw5WVlaX27dvr7rvvlr+/v9LS0rRz50598803GjlypGJjY3XbbbcZKhkAAKBycypgzZ49W5cuXdI//vEP9erVq9iyCRMmaMOGDZo4caLee+89TZs2zalC8ftq166tFi1aOG4DAADXcCpg7dixQ926dSsRror07NlT69at044dO5wZBmVks9kcQbboMC0AALjxnLoO1rlz59S0adNr9rn99tt17tw5Z4ZBOdhsNsIVAAAu5lTAqlu3ro4dO3bNPj/++KPq1KnjzDAAAABuxamAdffdd2vLli1at25dqcs3btyo+Ph4derUyZlhAAAA3IpT52CNGzdO8fHx+tvf/qYlS5borrvukr+/v9LT05WQkKBvvvlGNWrU0JgxY0zVCwAAUOk5FbCaNGmi+fPna9KkSUpMTFRiYqJsNpssy5L0n+tgcYkGAADwR+L0hUZDQ0P12WefKTExUQcPHnR8F2GLFi3Uvn17EzUCAAC4FWNflRMWFqawsDBTmwMAAHBbTp3kDgAAgJKc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alVHX+a6bN5//30dPHhQBw8e1IkTJ+Th4aGDBw9etX95n5/PnTund999V/Hx8Tp//rwaN26s6OhojRgxQlWqOBeRbJZlWU5tATdcbGysXnzxRdntdj388MO6fPmyFi9erAsXLmjp0qUKDg52dYlu5a233tKSJUvUuXNntWnTRt7e3tq3b59Wr16tmjVraunSpbrjjjsc/Z999lmtWbNGnTt3VpcuXZScnKyFCxfq1ltv1SeffCIfHx8X7o37mT9/vv75z3/ql19+KRGwCgoKNHz4cCUkJKhv377q2LGjDh8+rKVLl6pjx46aP3++PDx4I/5aUlNTNWzYMJ07d079+/fXnXfeqezsbP3000/y9/fXmDFjJDHXzsrPz9fDDz+sgwcPql+/fmrTpo1yc3O1adMm7d69W3369NFbb70libkuj+DgYNWqVUstWrTQjz/+qIyMjGsGrPI8P2dlZenhhx/W8ePH9eijjyo4OFi7d+/W6tWrNWDAAE2dOtW54i24lfPnz1thYWHWn/70JyszM9PRfvLkSatt27bW0KFDXVide9q/f7914cKFEu3Lli2z7Ha79eSTTzraduzYYdntdmv06NHF+m7YsMGy2+3WjBkzrnu9N5MTJ05Ybdq0sRYsWGDZ7Xbr8ccfL7Y8NjbWstvt1quvvlqs/aOPPrLsdru1cuXKG1esmxo2bJh17733WikpKdfsx1w7Z/v27ZbdbrfeeOONYu35+flW3759rebNmzues5nrsktKSnLcHjJkiNWiRYur9i3v8/O7775r2e12a968ecXap0yZYtntdishIcGp2onIbiY+Pl5ZWVkaOHCgatas6WgPDAxUjx49tGvXLp06dcqFFbqfkJAQ1apVq0R77969JUlHjhxxtK1evVqSNGLEiGJ9e/ToocaNGzuWo2xefPFF3XnnnRo6dGipy682348++qi8vb21atWq612iW/vmm2+0c+dOjRo1So0aNVJBQYEuXbpUal/m2jmZmZmSVOKUAk9PT9WvX1+enp7y8vKSxFyXx6233lrmvuV9fl69erWqV6+uRx55pFh70frO3g8ELDfz7bffSpLatWtXYllR24EDB25oTTer1NRUSVL9+vUdbd9++608PDzUtm3bEv3btWunEydO6Pz58zeoQvcWExOjPXv26LXXXiv1cIhlWTpw4IACAgLUuHHjYsu8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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.set_context(\"paper\")\n",
    "sns.set(font='Arial')\n",
    "sns.set(font_scale=1.2)\n",
    "sns.set_style(\"white\")\n",
    "\n",
    "sns.boxplot(data=imc_df_all, x=\"Cofactor\", y=\"Method\")\n",
    "#plt.savefig(\"cofactor-mode-boxplot.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "ebc23e42",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Percentage of runs with mode = 1 for M-SA: 0.99\n",
      "Percentage of runs with mode = 1 for M-AS: 0.90\n"
     ]
    }
   ],
   "source": [
    "print(f\"Percentage of runs with mode = 1 for M-SA: {((imc_modes_msa == 1.).sum() / imc_modes_msa.shape[0]):.2f}\")\n",
    "print(f\"Percentage of runs with mode = 1 for M-AS: {((imc_modes_mas == 1.).sum() / imc_modes_mas.shape[0]):.2f}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a438bfae",
   "metadata": {},
   "source": [
    "# CITE-seq"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "23811fa2",
   "metadata": {},
   "outputs": [],
   "source": [
    "optimise_iter = 36\n",
    "n_runs = 100\n",
    "path='meta_obj_csv/'\n",
    "\n",
    "_, citeseq_solution_msa = load_csv(path+'new_citeseq_msa_solution_100_runs_36_iter.csv', n_runs, optimise_iter)\n",
    "_, citeseq_solution_mas = load_csv(path+'new_citeseq_mas_solution_100_runs_36_iter.csv', n_runs, optimise_iter)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "8735c914",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2714263/3908738643.py:7: FutureWarning: Unlike other reduction functions (e.g. `skew`, `kurtosis`), the default behavior of `mode` typically preserves the axis it acts along. In SciPy 1.11.0, this behavior will change: the default value of `keepdims` will become False, the `axis` over which the statistic is taken will be eliminated, and the value None will no longer be accepted. Set `keepdims` to True or False to avoid this warning.\n",
      "  res = stats.mode(np.round(solutions[run,:], digits))\n"
     ]
    }
   ],
   "source": [
    "citeseq_modes_msa, citeseq_quantities_msa = compute_modes(citeseq_solution_msa, digits)\n",
    "citeseq_modes_mas, citeseq_quantities_mas = compute_modes(citeseq_solution_mas, digits)\n",
    "\n",
    "citeseq_modes_msa_df = make_df(citeseq_modes_msa, \"M-SA\", \"% HVGs\")\n",
    "citeseq_modes_mas_df = make_df(citeseq_modes_mas, \"M-AS\", \"% HVGs\")\n",
    "\n",
    "to_concat = [citeseq_modes_mas_df]\n",
    "c = 0\n",
    "total_len = len(citeseq_modes_msa_df)\n",
    "\n",
    "for frame in to_concat:\n",
    "    frame.index = range(total_len*(c+1), total_len*(c+2))\n",
    "    c=c+1\n",
    "\n",
    "citeseq_df_all = pd.concat([citeseq_modes_msa_df, citeseq_modes_mas_df])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "f2e918f1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.0, 0.51)"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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7fpCa926tbp3765t27YpLCxMbdq0yfvXqVMnVa5cudDjrFarLBaLwsPDJd2efWvbtq2sViuhC27ll38gMnLkyFKspOyx2WyylHYRDmBsCyruH0RlZ2dr0aJFhe5btGiRunXrJk9PT7vqeOeddwrUk7v95ZdflqTfbGOxFP0rNCcnR/Pnzy903/z58xUTE1Pku0ju4NatW1qzZk2h+9asWaMRI0YUeoesuMr0/6yvr6/69eunDRs26ObNm9q/f7+OHz+uyMhIu/tatGiRZs2apZYtW+rQoUNatmyZJk2apIcfflhvvPGGsrOz87XPysrS2rVr9fDDD+ebGRs0aJBOnjyp5ORkh68PAOA6Nm7cWOC1Ild2drY2btxoV3+nT5/WgQMHlJOTk297Tk6ODhw4oNOnTxepjT2Sk5OVlpZW6L60tLR77rVt6dKlDu23V5me6ZKkyMhIxcbG6rPPPtOOHTtUtWrVQqdwMzIyCnwh+fr65q3VKleunB599FE9+uijunnzplJSUrRz507985//1Pvvv6/KlSvrqaeeyjt227ZtunjxosaMGZO3JkySWrZsqXLlyik2NlYhISEldNWAWb/8TXnFihXy8fEpxWpKX0ZGRt6skD2zCGURY3ubM8a0T58+Wrp0aaHBy9PTU3369LGrv3r16ik4OFiHDh3KF6o8PDzUtm1b1atXT5KK1KaoQkJC5OfnV2jwqly58j33uvY///M/+vTTT++635nKfOhq06aNAgMDtWLFCh09elRRUVEqX758gXaffPKJpk+fnm/bM888o4kTJxZoW758+bzbi3369FG/fv0UFxeXL3Tl/pXi/PnzC52K3bRpk1588cU73poEXJWPj889+8Ls7hhbx3h6emrixIl66623CuybNGmSXbcWpdvhb9y4cRo/fnyh23PDYVHaFJWHh4emTJmil156qcC+KVOm3FO3FqXbEzKPPPJIobcYBw0a5NRbi5ILhC7p9oXPnTtXku54a7FLly5avnx5vm25C+jvpnHjxqpSpYrOnz+ft+3s2bPauXOnQkNDNXDgwALH/PDDD3rzzTfzHrYKALg3hIWFKSYmJt9i+ho1aig0NLRY/dWpUyfvjo7NZpPFYlFkZKRq165tVxt7tG3bVs2bN8+3mL558+Zq06ZNsfpzdWPGjNGGDRvyLab38vLS6NGjnX4ulwhd4eHhSktLk5+fn5o0aVJom4CAAAUEBBS679SpU8rJyVHDhg0L7EtKSlJqaqpatGiRty0+Pl45OTkaNWqUOnXqVOCY7OxsLV++XFarldAFAPeY6OjofH9M5egDUiMjI/XZZ5/p8uXLqlatWqGTC0VpY48ZM2Zo5MiRstls8vDw0IwZMxzqz9W9+eabevbZZ/N9XhJcYh7R399fEydOLHbq/L//+z/17dtXI0aM0P/+7/8qLi5OH374oV566SU9+eSTKleunCZPnizp9uLE+Ph4VatWTR07diy0P09PT/Xs2VMpKSn6+uuvi3tZAAAXFBAQoC5duki6fZflTr/wF5WPj4/Gjx+vmjVravz48YXeAi5KG3tUqVJFUVFR8vDw0ODBg1WlShWH+nN1jRs3zlsfV69ePTVu3LhEzuMSM12OateunaZNm6bdu3dr3bp1unz5snJychQQEKDevXtrzJgxatq0qSRpx44dOnfunKKiou56f75Pnz5avXq1rFZrkZ+MDwBwD1OnTnX4oai/1LFjxzv+om9PG3uMGDFCI0aMcFp/ru5vf/tbiZ+jzIWuBx98UMeOHStS2xUrVhSpXbVq1TR69OgizZR17dq1SOfv1KlTkesEAABwiduLAAAAro7QBQAAYAChCwAAwABCFwAAgAGELgAAAAMIXQAAAAYQugAAAAwgdAEAABhA6AIAADCA0AUAAGAAoQsAAMAAQhcAAIABhC4AAAADCF0AAAAGELoAAAAMIHQBAAAYQOgCAAAwgNAFAABgAKELAADAAEIXAACAAYQuAAAAAwhdAAAABhC6AAAADCB0AQAAGEDoAgAAMIDQBQAAYAChCwAAwABCFwAAgAGELgAAAAO8SrsAAKXP29tbzZo1y/sYcDd8jaMsIHQBkMViUXR0dN7HgLvhaxxlAaELgCReiOD++BpHaWNNFwAAgAGELgAAAAMIXQAAAAYQugAAAAwgdAEAABhA6AIAADCA0AUAAGAAoQsAAMAAQhcAAIABhC4AAAADCF0AAAAGELoAAAAMIHQBAAAYQOgCAAAwgNAFAABgAKELAADAAEIXAACAAYQuAAAAAwhdAAAABnjZ0/jtt98u1kksFosmTJhQrGMBAADcgcOhy2Kx5H1ss9kKbLfZbIQuAABwz7MrdP3zn/8ssO2DDz7QF198of79+6tjx46qUaOGLl26pD179ujjjz9W165d9dhjjzmtYAAAAFdkV+jq2LFjvs/XrVun3bt361//+pdatGiRb19ERIQeffRRjRgxQj179nS8UgAAABfm0EL6Dz74QH379i0QuHK1atVKffv21T/+8Q9HTgMAAODyHApdJ0+eVEBAwF3bBAQE6OTJk46cBgAAwOU5FLoqVaqk/fv337XNvn375Ovr68hpAAAAXJ5Doatr165KTk5WdHS00tPT8+1LT0/XvHnztH//fnXv3t2hIgEAAFydXQvpf23y5MlKSkrSBx98IKvVqmbNmql69eq6fPmyUlJSlJ6ervvvv1/PPfecs+oFAABwSQ7NdFWvXl1Wq1WRkZHKzs7W3r17tXHjRu3du1fZ2dmKiopSbGysqlev7qx6AQAAXJJDM12SVLVqVb366quaPXu2vvvuO6WlpcnPz0+NGjWSl5fD3QMAALgFp6UiLy8vBQYGOqs7AAAAt+K00JWcnKyUlBRdv35dfn5+at68uUJCQpzVPQAAgEtzOHQdPnxYU6ZMyXsWV+57LUpSw4YNFR0drVatWjl6GgAoFbacLKf24Yz+intuAKXLodB16tQpjR49Wunp6Wrfvr0eeugh1axZUxcvXtSXX36pffv2aezYsbJarWrQoIGTSgYAc27837oy3R8A1+FQ6FqyZIlu3Lihv/71r+rbt2++fRMnTtTGjRv13HPP6W9/+5uio6MdKhQAAMCVORS6du/erZ49exYIXLn69OmjhIQE7d6925HTAIBR3t7eslqtTu3TZrNJUt7yi9Lg7e1daucG4GDounr1qho2bHjXNo0aNdK2bdscOQ0AGGWxWOTj41PaZQBwMw49HLVatWo6ceLEXdt89913qlq1qiOnAQAAcHkOha6HHnpIW7duVUJCQqH7N23apMTERHXu3NmR0wAAALg8h24vTpgwQYmJiXr++ef14Ycf6sEHH1TNmjV16dIlJSUlad++fapYsaKefvppZ9ULAADgkhwKXfXr19fy5cs1depU7d+/X/v375fFYslbMJr7nC4eFwEAAO51Dj8ctXXr1vr000+1f/9+HTlyJO+9F5s1a6b27ds7o0YAAACX57S3AWrXrp3atWvnrO4AAADcikML6QEAAFA0Ds903bp1S4mJifrqq690/fp1ZWdnF2hjsVj0+uuvO3oqAAAAl+VQ6Dp//rzGjh2r7777Lm/xfGEI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8dHEiRO1adMmbd++XW+88Ybat2+vDRs2aNKkScW9BAAOInQBcCsLFixQZmamZs6cmW/7r5/0npmZabKsvHVbGzZs0MWLF7V582ZVqVJFffv2LdC2Y8eOql+/vs6dO6f4+HiHzlu7dm0NGDBAy5YtU/369bVv3z4W0wOlhNAFwG0kJydr5cqVeumll/ItGG/cuLGOHz+uH374QdLtv/Dbt2+fHnjgAWO1Va1aVT179tS1a9f07LPPKiMjQwMGDJC3t3eBtp6ennr55Zfl5eWlOXPm5D2x/tdu3bqljIyMfNuuXLmS7/Zjrp9++kk//fSTvLy8VK5cOeddGIAiY00XALeQkZGhmTNnqlevXurTp0++fY8//rgSEhL02GOPqWfPntq9e7euX7+uJ5980miNUVFRSkhIUHJyct7nd9KpUyctXLhQ06ZN04wZM7R48WJ17NhRAQEByszM1IULF7R7926lpqYqKChIlStXliSdP39e4eHhCgwMVFBQkGrXrq309HRt375dFy9e1MiRI1WpUiUj1wsgP0IXALewcOFCpaamatasWQX2NWvWTIsWLdJbb72lDz/8UPfdd5/mzJmjrl27Gq3xoYceUoMGDfT9998rODhYgYGBd23fo0cPbdmyRbGxsfriiy+0fft2paWlqXz58rrvvvvUrVs39enTR127ds17PETdunU1ceJEJSUlac+ePbp69ar8/f3VsGFDTZ48Wf369TNxqQAKYbHx9vQAAAAljjVdAAAABhC6AAAADCB0AQAAGEDoAgAAMIDQBQAAYAChCwAAwABCFwAAgAGELgAAAAMIXQAAAAYQugAAAAwgdAEAABhA6AIAADCA0AUAAGDA/wOUjA4VA9Z0ywAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.set_context(\"paper\")\n",
    "sns.set(font='Arial')\n",
    "sns.set(font_scale=1.2)\n",
    "sns.set_style(\"white\")\n",
    "\n",
    "ax = sns.boxplot(data=citeseq_df_all, x=\"% HVGs\", y=\"Method\")\n",
    "ax.set_xlim([0., 0.51])\n",
    "#plt.savefig(\"hvgs-mode-boxplot.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "0507c1c0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Interquartile range for M-SA: 0.05\n"
     ]
    }
   ],
   "source": [
    "print(f'Interquartile range for M-SA: {(citeseq_modes_msa_df.describe().loc[\"75%\"] - citeseq_modes_msa_df.describe().loc[\"25%\"])[0]:.2f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "8a6fd251",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Interquartile range for M-AS: 0.08\n"
     ]
    }
   ],
   "source": [
    "print(f'Interquartile range for M-AS: {(citeseq_modes_mas_df.describe().loc[\"75%\"] - citeseq_modes_mas_df.describe().loc[\"25%\"])[0]:.2f}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7bd51d05",
   "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.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
