{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Illustration of Data Balancing Procedure\n",
    "\n",
    "In this Jupyter notebook, we provide a lightweight example that shows the effect of data balancing iterations on a simulated joint distribution."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import matplotlib as mpl\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import scipy\n",
    "\n",
    "import warnings\n",
    "warnings.filterwarnings('ignore') # deprecation warnings from seaborn internals"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib as mpl\n",
    "\n",
    "mpl.rcParams['lines.linewidth'] = 4\n",
    "mpl.rcParams['xtick.labelsize'] = 20\n",
    "mpl.rcParams['ytick.labelsize'] = 20\n",
    "mpl.rcParams[\"axes.labelsize\"] = 20\n",
    "mpl.rcParams['legend.fontsize'] = 33\n",
    "mpl.rcParams['axes.titlesize'] = 32\n",
    "mpl.rcParams['pdf.fonttype'] = 42\n",
    "mpl.rcParams['ps.fonttype'] = 42\n",
    "\n",
    "data_color = \"dodgerblue\"\n",
    "true_color = \"lightsalmon\"\n",
    "title_size = 20"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The method can be implemented with a few lines, as seen in the `data_balance` function below."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def count_freq(X, Y, sizes):\n",
    "    \"\"\"\n",
    "    Compute the histogram/joint probability mass function of integer-valued paired data.\n",
    "    \"\"\"\n",
    "    pairs = list(zip(X, Y))\n",
    "    ind, count = np.unique(pairs, axis=0, return_counts=True)\n",
    "    cmat = np.zeros(sizes)\n",
    "    cmat[ind[:, 0], ind[:, 1]] = count\n",
    "    return cmat / len(pairs)\n",
    "\n",
    "\n",
    "def data_balance(X, Y, marginals, num_iter):\n",
    "    \"\"\"\n",
    "    Apply `num_iter` balancing iterations to empirical data and return the sequence of joint distributions.\n",
    "    \"\"\"\n",
    "    pmat = count_freq(X, Y, (len(marginals[0]), len(marginals[1])))\n",
    "    if np.sum(np.sum(pmat, axis=1) == 0) + np.sum(np.sum(pmat, axis=0) == 0) > 0:\n",
    "        raise RuntimeError(\n",
    "            \"Missing mass in this sample. Try a larger sample size.\")\n",
    "        \n",
    "    est = [pmat.copy()]\n",
    "    for i in range(1, num_iter):\n",
    "        pmat = (marginals[0] / np.sum(pmat, axis=1)).reshape(-1, 1) * pmat\n",
    "        pmat = pmat * (marginals[1] / np.sum(pmat, axis=0))\n",
    "        est.append(pmat.copy())\n",
    "    return est"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Simulate Data\n",
    "\n",
    "We first simulate a joint data distribution by specifying $P_X$ and $P_{Y|X}$."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(123)\n",
    "\n",
    "# set true distribution\n",
    "m = 50\n",
    "bias = 0.5\n",
    "transition = np.random.rand(m, m)\n",
    "transition = np.arange(m + 1)[1:][:, None] * np.arange(m + 1)[1:][None, :] + bias * m * np.ones(shape=(m, m))\n",
    "transition = transition / transition.sum(axis=0)[None, :]\n",
    "weight = 0.8\n",
    "nums = np.arange(m)\n",
    "p = 0.7\n",
    "px = weight * np.exp(np.log(scipy.special.binom(m-1, nums)) + nums * np.log(p) + (m - 1 - nums) * np.log(1 - p)) + (1 - weight) * np.ones(m) / m\n",
    "prob = transition * px[None, :]\n",
    "py = prob.sum(axis=1)\n",
    "marginals = (py, px) # each marginal is of size m"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Below, we plot the true marginal distributions $(P_X, P_Y)$ in the form of histograms."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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99xzyyoRgApz8jsAhT366KNx9tlnR61Wi4iDsyI7duyI0dHRpud27twZF154Yezbt69x77zzzouvfOUr81ovANVhxwSAwt70pjfFhg0bGtdPP/10TExMND3z97//PTZv3twUSk499dT40pe+NG91AlA9dkwAaMmf/vSn+OAHP9i4XrJkSdx7772xaNGi+M9//hMXXHBB/OUvf2n8/Pjjj48f/OAHMTIyklEuABVhxwSAlpxwwglx2mmnNa6npqbi9ttvj//+979x1VVXNYWSo446KrZu3SqUAHBYdkwAaNlDDz3UdOjiihUr4l3velfcddddjXsjIyMxMTERJ5xwQkaJAFSMYAJAWy6++OJ44IEHXvFnRxxxRGzZsiVOPfXUea4KgKrSygVAWy6//PJX/dk111wjlADQEsEEgLa8+93vjre97W0vu3/VVVfFeeedN/8FAVBpggkAbdm5c2c88sgjTfcWL14cV1xxRVJFAFSZYAJAy/72t7/FZZdd1nRWSUTEnj174p577kmqCoAqE0wAaMmePXti06ZN8a9//esVf37TTTeF96oA0CrBBIDCnnvuubj88svjsccea9w75phjYtWqVY3rRx55JO69996E6gCoMsEEgEJqtVpcffXV8dvf/rZxb/HixbF169bYvHlz07M33XTTfJcHQMUJJgAUct1118X27dsb16997WvjO9/5TrzxjW+MD3zgA3H00Uc3fvb73/8+fvnLX2aUCUBFCSYAHNatt94at912W+N6YGAgbrjhhnj7298eERELFiyITZs2Na2xawJAKwQTAGa1ffv2uP7665vufe5zn4v3ve99Tfc+/OEPx4oVKxrXv/71r+PBBx+clxoBqD7BBIBX9eCDD8bVV18dtVqtce+iiy6Kiy+++GXPDg0NxSWXXNJ077vf/W7ZJQLQIwbq3ukIwCt47LHH4vzzz489e/Y07r33ve+Nb37zmzE4+Mr/rvXss8/G6aefHk899VTj3h133BFr1qwpu1wAKs6OCQAv89RTT8Wll17aFEpOPvnk+OpXv/qqoSQiYuHChS/bTTFrAkARdkwAaLJv37648MILY+fOnY17q1atih/+8IexZMmSw66fnp6OM844oxFqBgYG4ic/+Um85S1vKatkAHqAHRMAGg4cOBCf+tSnmkLJkiVLYuvWrYVCSUTEyMhIXHjhhY3r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",
      "text/plain": [
       "<Figure size 800x500 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "sns.set_style(\"white\")\n",
    "edges=False\n",
    "color = true_color\n",
    "fig, ax = plt.subplots(2, 1, figsize=(8, 5))\n",
    "\n",
    "# sample original data\n",
    "n = 1000000\n",
    "data = np.random.choice(m ** 2, size=(n,), p=prob.reshape(-1))\n",
    "x = data // m\n",
    "y = data % m\n",
    "num_iter = 4\n",
    "\n",
    "n = len(x)\n",
    "m = len(np.unique(x))\n",
    "\n",
    "# flip X to match orientation\n",
    "if edges:\n",
    "    ax[0].hist(x, color=color, bins=m, edgecolor=\"k\")\n",
    "    ax[1].hist(-y, color=color, bins=m, edgecolor=\"k\")\n",
    "else:\n",
    "    ax[0].hist(x, color=color, bins=m)\n",
    "    ax[1].hist(-y, color=color, bins=m)\n",
    "    \n",
    "ax[0].set_xticks([])\n",
    "ax[1].set_xticks([])\n",
    "ax[0].set_yticks([])\n",
    "ax[1].set_yticks([])\n",
    "\n",
    "ax[0].set_xlabel(r\"$X$\", fontsize=30)\n",
    "ax[1].set_xlabel(r\"$Y$\", fontsize=30)\n",
    "\n",
    "fig.tight_layout(pad=0.5)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Balance the Empirical Distribution\n",
    "\n",
    "We then sample data from this distribution to generate a dataset of $(X_1, Y_1), \\ldots, (X_n, Y_n)$ of iid pairs."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# sample original data\n",
    "n = 1000\n",
    "data = np.random.choice(m ** 2, size=(n,), p=prob.reshape(-1))\n",
    "x = data // m\n",
    "y = data % m\n",
    "num_iter = 4\n",
    "\n",
    "# get empirical distribution and balanced  estimators\n",
    "est = data_balance(x, y, marginals, num_iter)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# seaborn joint plot requires data, so we generate a large dataset from a joint pmf\n",
    "def generate(jpmf, n=1000000):\n",
    "    np.random.seed(123)\n",
    "    data = np.random.choice(m ** 2, size=(n,), p=jpmf.reshape(-1))\n",
    "    x = data // m\n",
    "    y = data % m\n",
    "    return pd.DataFrame({'x': x, 'y': y})"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We then visualize each iteration of the procedure."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAh0AAAJFCAYAAAB5vUV9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAAyJElEQVR4nO3de3xU9Z3/8ffkYpBMwqWlW2sIotYoIoqhXtBCFdagKFqtkcpu1z4MN41Q8Ia/ii4iCm5XqSLKRbdW3QcCvawrQgBr09Kq1VCo4oXCglwEi0IIAcn1/P4YJwaY8z3JXL5nZvJ6Ph4+HsEz58x3zpzJvHPO+Xy+AcdxHAEAACRYht8DAAAAHQOhAwAAWEHoAAAAVhA6AACAFYQOAABgBaEDAABYQegAAABWEDoAAIAVhA7gKE1NTX4PISapPn4A6YvQgQ7j17/+tYqKilRUVKS33nrrmOX19fWaM2eOnnnmGR9GF7vPP/9ct99+u6qqqo5ZFn7djz32mA8ji81HH32kO++8U4MGDVLfvn118cUXa9y4cfrDH/7g99AAtBOhA/jSTTfdpCeeeEJ1dXV+D6Xd9u7dq8svv1yvvPKK0mlmg9WrV+u6667Tyy+/rE8//VQNDQ3as2ePXn/9dY0ePVoPPvig30ME0A6EDuBLu3fv9nsIUTt06JD279/vurywsFCFhYXq1q2bxVHFZsOGDZo8ebIaGhp01lln6fnnn9ebb76ppUuXasiQIZKk559/Xi+++KLPIwXQVll+DwBA4q1atcrvIbTb7NmzVVdXp8LCQj333HPKzc2VJHXr1k1PPvmkbrvtNq1atUqPP/64rr76agWDQZ9HDMALZzoAJJ3Nmze33LMxZsyYlsARFggENGXKFAUCAVVXV6uiosKPYQJoJ0IHOrx//dd/VVFRkXbu3ClJmjNnjoqKinTppZce89h9+/Zp9uzZuvrqq3Xuuefq7LPP1rBhwzRjxgzt2rUr4vafeOIJFRUV6Yc//KGqq6t1++23q3///jr33HN17bXX6qOPPmp5bH19vZYsWaJx48Zp0KBB6tevn8455xxdeumlmjx5st54441jtl9UVNRyuUGSfvSjH6moqEhTpkw54jGmG0m3b9+uGTNm6IorrtA555yj/v37a/jw4XrooYdcX1f4xtxBgwZJkv72t79p0qRJuvjii9W3b18NHjxY99xzjzZv3hxxfZNw4AgEAhHfB0kqKChQUVGRJOm1115r93MAsI/LK0Abvfnmm5owYcIx905s2bJFW7Zs0eLFi/XII4+opKQk4vr19fUqKyvTu+++2/L/tm3bpl69ekkKffGXlZVp69atx6y7c+dO7dy5U8uWLVN5ebluu+22uL2uJUuWaPr06cfcQLtp0yZt2rRJixYt0owZM3TVVVcZt3H//fcfUa67e/du/frXv9Yrr7yiefPmaeDAgW0e0wcffCBJ+uY3v6mvfe1rro/r06ePPvzwQ23YsKHN2wbgH0IHOrwFCxaoqalJV155pT755BONHTtWY8eOVUbGVycCN27cqLFjx+rw4cMqKCjQhAkTdMEFFygrK0vvvfeennjiCb377ruaPHmyfvnLX6q4uPiY53nvvfckSbfddptKS0u1d+9ebdmyRZ06dVJTU5PKy8u1detWde7cWRMnTtSgQYPUrVs3ffbZZ/rzn/+suXPnqrq6WnPnztWIESNawsratWu1a9cuDR8+XJI0f/58DRgwQNnZ2Z6vvaKiQlOnTpXjOCosLNTEiRN13nnnSZLeeustzZ49Wzt27NCdd96p/Px8DR48+Jht7N27V/fff79OOukk/eQnP1FxcbG++OIL/fa3v9XcuXNVX1+ve++9V6tXrz5in5p88sknkqQTTzzR+LhvfetbktRS2dKW1wzAP4QOdHidOnWSFDqVL0nZ2dnH3EPwwAMPtASOpUuXHlEFMnjwYF144YX6l3/5F61fv17Tpk3Tyy+/HPG5RowYofLycknSN77xDZ1++umSpDVr1ujDDz+UJE2bNk0jRoxoWadbt2769re/rRNPPFG33nqrmpubtWbNmpbQkZub2/Iawq/n6PFHUl9frxkzZshxHPXq1UsvvfTSEa/rqquu0kUXXaTrr79eO3bs0H333adVq1bpuOOOO2I7DQ0NKigo0OLFi4+4mbO8vFwHDx7Us88+q507d+r9999X3759PcclhS5jSVJ+fr7xcXl5eZIkx3F04MABde/evU3bB+AP7ukAPGzatElvv/22JOmWW26JWHZ63HHHadKkSZJCzazWr18fcVtul15yc3P1ox/9SMOHD9cVV1wR8THnn39+y8/hL+VYVFZW6tNPP5Uk3XXXXRFfV/fu3VvuDdm9e7def/31iNu68cYbI1aPtD4zsmPHjjaPLXypp3WYiiQnJ+eYdQAkL850AB5ady897bTTdPDgwYiPKyoqUmZmppqamlRVVaWzzz77mMecccYZEdcdMGCABgwY4DqG/fv365133mn5d2NjY1uH7+ovf/mLpNAXd6TLJmHf+973lJOTo7q6Or399tsRg1Ok1yrpiPsxDh8+3OaxZWZmSvrq7JOb1o3Q2nrpBoB/CB2Ah+3bt7f8/IMf/KBN67hVfHTt2tVz3fXr12vdunXaunWrtm3bpi1btuiTTz454gs2Hl1Hw83QevXqZbwXIjs7W7169dLGjRtb7rU4mlvTsdaXYpqbm9s8ts6dO0vyDir19fURnwtAciJ0AB5qa2vjtk7rywFHe+eddzRz5swjqlvCCgoKdPHFF2vRokXtHovXGMNf8CbHH3+8pFDn00iysuL7qyR8qebAgQPGx9XU1EgKneXo0qVLXMcAIP4IHYCH1vcV/O1vfzMGh2i9++67uummm9TQ0KDOnTtr6NChOvvss3XqqafqtNNOU/fu3dXY2BjX0BEOG25BorVwQAmHj0Tr3bu3/vKXv7ieMQoLLz/hhBO4vAKkAEIH4CFclimFboY85ZRTXB/rOI7nfQiRzJ49Ww0NDcrLy9OvfvWrlsqU1uJx82hr4df18ccfG8tN6+vrtW3bNkneJazxEm76tXPnTtXU1LhWsYT7c4SrgAAkN/40ADy0vsHT1Ply7dq1Ovvss1VSUqLly5e36znWrl0rSRo4cGDEwCHpiG6kR9/TEU3QCb+uuro6VVZWuj7u9ddfV0NDgySpf//+7X6eaIS7nDY3N+v3v/99xMds375dGzdulCR997vftTIuALEhdABfCt+XEP6CDevXr1/LX9ILFiyI2DH08OHDmjVrlurq6rRz507169evXc8drtbYsmVLxBsud+3apZ/97Gct/z56jK3vqTh6mZtLLrlEPXr0kCQ98sgjEc+k7Nu3r+V5u3fv7tqSPN569uzZ0mDtySefPObeDsdxNHPmTDmOo27duunqq6+2Mi4AsSF0AF8KV5b88Y9/1Pbt27V3796WZffff7+ysrJUU1OjG264QS+88IJ27Nihzz//XGvWrNFNN92kdevWSZJuvvnmdl+GuOiiiySFOp/ecccd+uCDD7Rv3z5t3rxZCxcu1DXXXNPSU0PSMWW7rW+iXL58uT777DPjVPdSqNrjvvvukxS6xFJaWqply5Zpz5492rNnj5YtW6bS0tKWSysPPfSQtXs6JOmee+5RRkaGtm7dqhtvvFFr1qzR3r17tWHDBt16661avXq1pFCH17bcDAvAf9zTAXzp/PPP1/r16/X+++9r6NChys7O1l//+ldlZ2fr3HPP1eOPP6477rhD1dXVmj59uqZPn37MNq6//npNmDCh3c995513qqqqquXLftmyZcc85nvf+56qq6u1bt06ffzxx0cs69Spk8455xytW7dOS5cu1dKlS3Xeeefp+eefNz7vZZddpgcffFDTpk3Ttm3bNHny5GMe07lzZz3wwAO65JJL2v26YnHWWWdpxowZmjp1qjZu3Kibb775mMf8+Mc/1qhRo6yOC0D0CB3Al8rLy/XFF19oxYoVqq6uVvfu3bV792717NlTkjRkyBCtWrVKzz//vCorK7V9+3bV1dWpW7du6t+/v2644YaWMxbtVVBQoN/85jeaN2+eKisrtWvXLgUCAX3ta19Tnz59dO2112ro0KF6+umntW7dOq1du1aff/75Ec23HnvsMU2fPl1vv/226uvr21SVIoWC0gUXXKDnnntOf/rTn7Rr1y5lZmaqoKBAl1xyiW644QadcMIJUb2uWF177bU688wz9cwzz+itt97S559/rs6dO6tv37668cYbNXToUF/GBSA6ASceXYYAAAA8cE8HAACwgtABAACsIHQAAAArCB0AAMAKQgcAALCC0AEAAKwgdAAAACsIHQAAwApCBwAAsILQAQAArGDuFQBATGrqpMONkZd1ypLyc+yOB8mL0AEAiMnhRum2FZGXPTGM0IGvcHkFAABYQegAAABWcHkFAGBkumdDkpqa7Y0FqY3QAQAwMt2zIUmPXWZvLEhtXF4BAABWEDoAAIAVhA4AAGAF93QAABImIyD942DkZTQO63gIHQCAhKlvkiatjLyMxmEdD5dXAACAFYQOAABgBaEDAABYQegAAABWEDoAAIAVhA4AAGAFoQMAAFhBnw4AiDOvWVlpitU2pv3IPkxNhA4AiDOvWVmTsSmW6Qver6nrTfsxGfchvBE6AADGL3imrke8EDoAAL4wzcsi+XeGBYlD6AAA+MI0L4vEGZZ0ROgAEBepePNkR7pR0ev94awCbCB0AIiLVLx5siPdqOj1/nBWATYQOgDAMtO9DOl2hgVojdABAJaZ7mWI5QxL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",
      "text/plain": [
       "<Figure size 600x600 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "iteration = 0\n",
    "\n",
    "jpmf = est[iteration]\n",
    "df = generate(jpmf)\n",
    "\n",
    "g = sns.jointplot(data=df, x='x', y='y', kind='hist', bins=(m, m), color=data_color)\n",
    "g.ax_joint.set_xticks([])\n",
    "g.ax_joint.set_yticks([])\n",
    "g.ax_joint.set_xlabel(r\"$Y$\")\n",
    "g.ax_joint.set_ylabel(r\"$X$\")\n",
    "\n",
    "g.fig.subplots_adjust(hspace=0.01, wspace=0.01)\n",
    "g.fig.suptitle(f\"Iteration {iteration}\", y=1.02, fontsize=title_size)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAh0AAAJFCAYAAAB5vUV9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAAzYUlEQVR4nO3deXxU1f3/8Xc2EpIBBEVlKYhbtPp1C9oKrYgsEUGqXwRRREGiLAZQql1+FSmCa/vFjeWLICiCIiBWCyKLRKqtWg2CWrUWviCbuEGUCWSf3x+YCJJzbjKZObO9no+Hj0fw5N45c2cyeefe+/mcpEAgEBAAAECYJUd6AgAAIDEQOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAE4QOAADgBKEDAAA4QegAfqSysjLSU2iQWJ8/gPhF6EDCWLp0qbKzs5Wdna233377iPGysjJNnTpVTzzxRARm13DffPONfv3rX6uwsPCIsern/dBDD0VgZqE1efJkZWdna/HixZGeCoB6InQA3xsyZIgee+wxlZaWRnoq9bZnzx716tVLy5YtUzyvbPDqq69qwYIFkZ4GgCClRnoCQLTYvXt3pKcQtP379+vbb781jrdr106S1Lx5c1dTCrmCggLdeuutqqqqivRUAASJ0AEkgNWrV0d6CkGrqqrStGnTNH36dAIHEOO4vAIgar3xxhu64oorNHXqVFVVVemMM86I9JQANABnOpDwBg8erH/+8581/546daqmTp2qNm3aaO3atYd97969e/XUU0+poKBA27dvV2VlpVq1aqVf/vKXuvHGG9WqVasj9v/YY49p6tSpOu+88zRjxgxNmjRJa9euVVJSkk444QTdd999ys7OlnTwZtYXX3xRr776qj766CMVFRUpOTlZLVq00DnnnKP+/fvrwgsvPGz/1dtWu/766yVJV155pe6///7DvmfEiBG67bbbjpjj9u3bNW/ePP3973/Xrl27lJSUpNatW6tz584aOnRorc9r6dKl+v3vf6/jjjtOf/vb3/T+++9r7ty5euedd1RUVKSjjz5anTp1Ul5enk466STP16E2w4YNkySlpaVpxIgR6tu3r3r06BHUvgBEHqEDqKO33npLY8aMOeLeiS1btmjLli1atGiRHnzwQeXm5ta6fVlZmfLy8vTBBx/U/L9t27apffv2kg7+4s/Ly9PWrVuP2Hbnzp3auXOnli9frvz8fI0ePTpkz2vx4sWaNGnSETfQbtq0SZs2bdLChQt1zz336PLLL7fuY8KECYeV6+7evVtLly7VsmXLNHPmTHXq1Knec0tKSlK3bt00btw4nXTSSdqxY0e99wEgehA6kPBmzZqlyspK9enTR7t27dLw4cM1fPhwJSf/cPXx008/1fDhw1VSUqK2bdtqzJgx+vnPf67U1FR9+OGHeuyxx/TBBx9o3LhxmjdvnnJyco54nA8//FCSNHr0aA0YMEB79uzRli1blJGRocrKSuXn52vr1q3KzMzU2LFjddFFF6l58+b6+uuv9Y9//EPTp09XUVGRpk+frr59+9aElfXr1+vzzz9X7969JUmPP/64OnbsqLS0NM/nvnLlSo0fP16BQEDt2rXT2LFjdcEFF0iS3n77bT388MPasWOH7rjjDjVt2lRdunQ5Yh979uzRhAkTdMIJJ+jWW29VTk6ODhw4oL/85S+aPn26ysrKdOedd2rNmjWHHdO6WLFihTp06FCvbQBEL0IHEl5GRoakg39VSwdP5WdlZR32PXfffXdN4FiyZMlhVSBdunTRhRdeqOuuu04bN27UxIkT9dJLL9X6WH379lV+fr4k6dhjj9Vpp50m6eC9C5988okkaeLEierbt2/NNs2bN9cpp5yiNm3a6JZbblFVVZXeeOONmtCRlZVV8xyqn8+P51+bsrIy3XPPPQoEAmrfvr2ee+65w57X5Zdfrs6dO6t///7asWOH7rrrLq1evVqNGjU6bD/l5eVq27atFi1aJJ/PV/P/8/PzVVxcrDlz5mjnzp366KOPdOaZZ3rO61AEDiC+cCMp4GHTpk165513JEmjRo2qtey0UaNGNfdK/Pvf/9bGjRtr3Zfp0ktWVpauv/569e7dW5dddlmt3/Ozn/2s5uu9e/fW6znUZt26dfriiy8kSb/5zW9qfV4tWrTQ7373O0kHL5cUFBTUuq9rr732sMBR7dAzI1waAcCZDsDDod1LTz31VBUXF9f6fdnZ2UpJSVFlZaUKCwt19tlnH/E9p59+eq3bduzYUR07djTO4dtvv9W7775b8++Kioq6Tt+o+ubZ9PT0Wi+bVLv44ouVnp6u0tJSvfPOO7UGp9qeqyQdffTRNV+XlJQ0cMYAYh2hA/Cwffv2mq+vuuqqOm3z+eef1/r/jzrqKM9tN27cqA0bNmjr1q3atm2btmzZol27dh3WaTQUXUerm6G1b9/eev9HWlqa2rdvr08//VS7du2q9XtMTccOvRRDjw0AhA7Ag9/vD9k26enpxm3effdd3X///YdVt1Rr27atfvGLX2jhwoX1novXHDMzMz2/t3HjxpIOdj6tTWoqHyUAvPFJAXg49CbN999/3xocgvXBBx9oyJAhKi8vV2Zmprp3766zzz5bJ598sk499VS1aNFCFRUVIQ0d1WHDFCQOVR1QqsMHAASD0AF4aN26dc3XO3bssDa6CgQCNVUw9fHwww+rvLxcTZo00fPPP19TmXKoUNw8eqjq5/XZZ5+pvLzceImlrKxM27ZtkyS1adMmpHMAkFioXgE8HHqD56uvvmr8vvXr1+vss89Wbm6uVqxYUa/HWL9+vSSpU6dOtQYOSXrzzTdrvv7xPR3BBJ3q51VaWqp169YZv6+goEDl5eWSpHPPPbfejwMA1QgdwPeq70uo/gVb7ayzzqrppzFr1qxaO4aWlJTogQceUGlpqXbu3KmzzjqrXo+dkpIi6WB309puuPz888/15z//uebfP57jofdU/HjMpGvXrmrZsqUk6cEHH6z1TMrevXtrHrdFixa65JJL6rRvAKgNoQP4XnVlyeuvv67t27drz549NWMTJkxQamqqvvvuO1199dWaP3++duzYoW+++UZvvPGGhgwZog0bNkg6uF5IfS9DdO7cWdLBzqe33367Pv74Y+3du1ebN2/W7NmzdcUVV9T01JB0RNlus2bNar5esWKFvv76a+tS99LBypK77rpL0sFLLAMGDNDy5cv11Vdf6auvvtLy5cs1YMCAmksr9957L/d0AGgQ7ukAvvezn/1MGzdu1EcffaTu3bsrLS1N7733ntLS0nTeeefp0Ucf1e23366ioiJNmjRJkyZNOmIf/fv315gxY+r92HfccYcKCwtrftkvX778iO+5+OKLVVRUpA0bNuizzz47bCwjI0PnnHOONmzYoCVLlmjJkiW64IIL9PTTT1sft2fPnpo8ebImTpyobdu2ady4cUd8T2Zmpu6++2517dq13s8LAA7FmQ7ge/n5+Ro8eLBatmyptLQ0tWjRoqaXhSR169ZNq1ev1ogRI3T66afL5/MpLS1Nxx57rHJzczVnzhxNnjy55lJJfbRt21YvvPCCBg8erHbt2iktLU2N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",
      "text/plain": [
       "<Figure size 600x600 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "iteration = 1\n",
    "\n",
    "joint = est[iteration]\n",
    "df = generate(joint)\n",
    "\n",
    "g = sns.jointplot(data=df, x='x', y='y', kind='hist', bins=(m, m), color=data_color)\n",
    "g.ax_joint.set_xticks([])\n",
    "g.ax_joint.set_yticks([])\n",
    "g.ax_joint.set_xlabel(r\"$Y$\")\n",
    "g.ax_joint.set_ylabel(r\"$X$\")\n",
    "\n",
    "for patch in g.ax_marg_y.patches:\n",
    "    patch.set_facecolor(true_color)\n",
    "\n",
    "g.fig.subplots_adjust(hspace=0.01, wspace=0.01)\n",
    "g.fig.suptitle(f\"Iteration {iteration}\", y=1.02, fontsize=title_size)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAh0AAAJFCAYAAAB5vUV9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8pXeV/AAAACXBIWXMAAA9hAAAPYQGoP6dpAAAz6ElEQVR4nO3deXiU1d3/8U82kpABDYoWoeCOtj4uhNoKVYSgUVCqVSKyWJAoWwRxofX5iT6ARbGKWBFEFhVBMaJ1YSeCqbZqNVS0Uhd42BcXIMgECFnm9wdNHpacM8ksZ2Yy79d1eV2RM/c9Z+6ZyXxy3/P9ngSfz+cTAABAmCVGegIAACA+EDoAAIAThA4AAOAEoQMAADhB6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAI5SWVkZ6SkEJdbnD6DhInQgbrz++utq27at2rZtq48++uiY8YMHD2ry5MmaOXNmBGYXvJ07d+ruu+9WcXHxMWPVj/uJJ56IwMyCU1RUpOHDh+uyyy7Teeedp3bt2un666/XpEmTtGvXrkhPD0A9JEd6AkC06N+/v4qLi5Wfnx/pqdTbrl27dPXVV2vPnj3Kzc2N9HRCoqKiQn/4wx/09ttvH/Hv5eXlWrNmjdasWaOCggI9/fTTuuiiiyI0SwD1wZkO4D927NgR6SkEbN++fdqzZ49xvHXr1mrdurUyMzMdzio4jz/+eE3gyM7O1ssvv6wPP/xQb731lu6++26lp6dr586dGjx4sL799tsIzxZAXXCmA4gDy5cvj/QU6uXbb7/V7NmzJUnXXnutHnvssZqxzMxMtW3bVr/61a908803q6SkRNOmTdMDDzwQqekCqCPOdACIOoWFhaqoqJAkjRw5stbbnH/++ercubMk6d1333U1NQBB4EwH4l6/fv30j3/8o+b/J0+erMmTJ6tly5ZasWLFEbfdvXu3XnjhBa1cuVKbN29WZWWlWrRooUsvvVS33nqrWrRoccz+n3rqKU2ePFnt2rXT1KlTNW7cOK1YsUIJCQk69dRT9fDDD6tt27aSDn2Z9c0339Q777yjNWvWqKSkRImJiWrWrJkuvPBC9ezZU5dccskR+6/ettott9wiSbr++uv1yCOPHHGbwYMH1/ohvnnzZs2ePVt/+9vftG3bNiUkJOiUU05Rx44dNWDAgFof1+uvv6777rtPJ598sv7617/qs88+03PPPaePP/5YJSUlOuGEE9ShQwfl5eXpjDPO8Ps8HO67775TWlqaPB6PWrZsabxd69ata24PIPoROoA6+vDDDzV8+PBjvjuxfv16rV+/XgUFBXr00UeVk5NT6/YHDx5UXl6ePv/885p/27Rpk9q0aSPp0Ad/Xl6eNmzYcMy2W7du1datW7Vw4ULl5+frjjvuCNnjevXVVzVu3DiVlZUd8e9r167V2rVrNW/ePP3xj3/Utddea93Hgw8+eES57o4dO/T6669rwYIFmjZtmjp06FDnOY0cOVIjR46U1+u13m7jxo2SpOOOO67O+wYQOYQOxL3p06ersrJS11xzjbZt26ZBgwZp0KBBSkz8v6uPX3/9tQYNGqQDBw6oVatWGj58uH71q18pOTlZ//rXv/TUU0/p888/11133aXZs2crKyvrmPv517/+JUm64447lJubq127dmn9+vVKS0tTZWWl8vPztWHDBjVu3FgjRozQZZddpszMTP3www/6+9//rilTpqikpERTpkxRjx49asLKqlWrtH37dnXv3l2S9Oyzz6p9+/ZKSUnx+9iXLl2q0aNHy+fzqXXr1hoxYoQuvvhiSdJHH32kSZMmacuWLbr33nvVtGlTderU6Zh97Nq1Sw8++KBOPfVU3XnnncrKytL+/fv1xhtvaMqUKTp48KDuv/9+FRYWHnFM68Lj8RjHtm/frqKiIklSu3bt6rVfAJFB6EDcS0tLkyQlJCRIklJSUpSRkXHEbcaOHVsTOObPn39EFUinTp10ySWXqG/fvlq9erXGjBmjt956q9b76tGjR01J7kknnaRzzjlHkvT+++/ryy+/lCSNGTNGPXr0qNkmMzNTZ511llq2bKlhw4apqqpK77//fk3oyMjIqHkM1Y/n6PnX5uDBg/rjH/8on8+nNm3a6JVXXjnicV177bXq2LGjevbsqS1btuiBBx7Q8uXL1ahRoyP2U15erlatWqmgoOCIkJCfn6/S0lLNmjVLW7du1Zo1a3Teeef5nVddVFVVafTo0SovL5ck9e7dOyT7BRBefJEU8GPt2rX6+OOPJUlDhw6ttey0UaNGNd+V+Oqrr7R69epa92W69JKRkaFbbrlF3bt3V7du3Wq9zS9/+cuan3fv3l2vx1CboqKimlLTUaNG1fq4mjVrpj/84Q+SDl0uWblyZa376t27d61nJQ4/M7Jly5ag51xt/Pjxeu+99yRJ3bt3P+Z7LgCiE2c6AD8O71569tlnq7S0tNbbtW3bVklJSaqsrFRxcbEuuOCCY25z7rnn1rpt+/bt1b59e+Mc9uzZo08++aTm/6srO4JR/eXZ1NTUWi+bVLv88suVmpqqsrIyffzxx7UGp9oeqySdcMIJNT8fOHAgyBlLPp9P48eP14svvijp0PMxbty4oPcLwA1CB+DH5s2ba36+8cYb67TN9u3ba/33448/3u+2q1ev1qeffqoNGzZo06ZNWr9+vbZt2yafz1dzm8N/DlR1M7Q2bdpYv/+RkpKiNm3a6Ouvv9a2bdtqvY2p6djhl2KqqqqCmO2hy0H33XefFixYIEk644wzNGvWrDpdSgIQHQgdgB/+Kijqs01qaqpxm08++USPPPLIEdUt1Vq1aqVf//rXmjdvXr3n4m+OjRs39nvb9PR0SYc6n9YmOTm8v0p27dqlYcOGadWqVZKkn//855oxY4aaNWsW1vsFEFqEDsCPw7+k+dlnn1mDQ6A+//xz9e/fX+Xl5WrcuLG6du2qCy64QGeeeabOPvtsNWvWTBUVFSENHdVhwxQkDlcdUKrDh0sbNmzQ7bffXlMee+mll+rJJ5/kDAcQgwgdgB+nnHJKzc9btmyxNrry+Xw1VTD1MWnSJJWXl6tJkyZ67bXXaipTDheKL48ervpxbdy4UeXl5cZLLAcPHtSmTZskydqoKxy+/PJL9e/fv+ax5+bm6sEHHwz7mRUA4UH1CuDH4V/wfOedd4y3W7VqlS644ALl5ORo8eLF9bqP6ssGHTp0qDVwSNIHH3xQ8/PR3+kIJOhUP66ysrKafhe1WblyZU1pqsvVXDdu3Khbb721JnCMGDFC48aNI3AAMYzQAfxH9YdZ9QdstfPPP7+mn8b06dNr7Rh64MABTZgwQWVlZdq6davOP//8et13UlKSpEPdTWv7wuX27duPWPTs6Dke/kF89JhJ586d1bx5c0nSo48+WuuZlN27d9fcb7NmzdSlS5c67TtYBw8e1MiRI7Vz505J0n333aehQ4c6uW8A4UPoAP6jurLkvffe0+bNm7Vr166asepT+j/++KNuuukmzZkzR1u2bNHOnTv1/vvvq3///vr0008lSQMHDqz3ZYiOHTtKOtT59J577tG///1v7d69W+vWrdOMGTN03XXXHbF8+9Flu4e3AV+8eLF++OEH61L30qHKkuqVWTdu3Kjc3FwtXLhQ33//vb7//nstXLhQubm5NZdWxo8f7+w7HQUFBfriiy8kSVdddZV69uyp0tJS638Aoh/nKYH/+OUvf6nVq1drzZo16tq1q1JSUvTPf/5TKSkpateunf785z/rnnvuUUlJicaNG1drf4iePXtq+PDh9b7ve++9V8XFxTUf9gsXLjzmNpdffrlKSkr06aef1nypslpaWpouvPBCffrpp5o/f77mz5+viy++uKafhcmVV16phx56SGPGjNGmTZt01113HXObxo0ba+zYsTUrurpQvay9JC1ZskRLlizxu81XX30VzikBCAHOdAD/kZ+fr379+ql58+ZKSUlRs2bNanpZSFJ2draWL1+uwYMH69xzz5XH41FKSopOOukk5eTkaNasWXrooYdqLpXUR6tWrfSXv/xF/fr1U+vWrZWSkqJGjRqpRYsWys7O1tNPP61p06bVfPCvWrWq5tJDtSeeeEJdunRRkyZNlJqaWqeqFOlQUFq8eLH69eun008/Xenp6fJ4PDrnnHM0ZMgQLVq0yLrYW6jt3r37mFAFoGFI8IWiyxAAAIAfnOkAAABOEDoAAIAThA4AAOAEoQMAADhB6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAEyz4BgCITgdKpfKy2sdSUqW0DLfzQdAIHQCAyLCFCkny+aS/TKx97Ld3EzpiEKEDABAZ5WXS64+bx68b6W4ucILQAQAIH9vZDBY5jzuEDgBAcPwFC9MlEs5kxB1CBwAgOLbLJAQLHIbQAQCwq8sXPoE6IHQAAAK/RCJxNgN1RugAAHCJBE4QOgAgHnCJBFGA0AEADQWXSBDlCB0A0FBwiQRRjtABANHEdrYiKVmqrDBvyyUSRDlCBwBEE39nK954wrwtZzMQ5VjaHgAAOMGZDgBwjfVIEKcIHQAQasEs2c4lEjRghA4ACATlqUC9EToAIBCUpwL1xhdJAQCAE5zpAIDa0DYcCDlCB4D4xfcyAKcIHQDiF9/LAJwidABo2OiJAUQNQgeA2EZPDCBmEDoAxDbbJRKJYAFEEUIHgOjHJRKgQSB0AIg8LpEAcYHQAcANylOBuEfoABAanK0A4AehA0Bo8IVOAH6w9goAAHCCMx0A6o4qEgBBIHQAqDvahgMIAqEDwJE4mwEgTAgdAI7E2QwAYULoAOJNXUpbASAMCB1AQ0QjLgBRiNABxCIacQGIQYQOIBbRiAtADCJ0ANGKKhIADQyhA4hWVJEAaGAIHUCkUEUCIM4QOoBwoooEAGoQOoBw4hIJANQgdADB4gufAFAnhA7AH3piAEBIEDoAie9eAIADhA5A4rsXAOAAoQPxgfJUAIg4QgfiA23DASDiCB1oOKgiAYCoRuhA7KCKBABiGqEDsYNLJAAQ0wgdiC5cIgGABovQgehC6SoANFiEDrjH2QwAiEuEDoQeX/gEANSC0IHQ4wufAIBaEDoQGC6RAADqidCB2nGJBAAQYoSOeMbKqgAAhwgd8YzyVACAQ4SOhoyVVQEAUYTQ0ZBRRQIAiCKEjlhHFQkAIEYQOmId38sAAMQIQkcs4GwGAKABIHREA3piAADiAKEjGvCFTwBAHCB0uMIlEgBAnCN0hAqXSAAAsCJ0hAqXSAAAsCJ01AeXSAAACBih42iBLoLGmQwAAKwIHUej2RYAAGERf6GDRdAAAIiI+AsdfOETAICIiInQ4fP5VFpaGpqdeUulsvLAxgMdi9S20TinYLaNxjkFs200zimYbaNxTsFsy5zCv22w+01oZN62njIyMpSQkBCy/aF2CT5f9F9P2Lt3r9q3bx/paQAAGqiioiKdfPLJBI8wi4kzHdWKiork8XgiPQ0AQAPi9XrVqVMnFRcX8xkTZjEROqqTp8fj4QUBAECMSoz0BAAAQHwgdAAAACcIHQAAwAlCBwAAcILQAQAAnCB0AAAAJwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcIHQAAwAlCBwAAcILQAQAAnEiO9AQANAyNnvBZxw+OTHA0EyAwP5RK+xOk1GSpaWqkZ9MwcaYDAABJ9xRK+UuksopIz6ThInQAAAAnCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcomQUQEv5KYuesLDWO9e2cYd12xrJ9xrHGKVXWbXt39hjH5r3rNY55D9r/Jsu7srF1PJ4UFJmPY24n8/GPNhOypQyPlMyf42HDoQUAQNLv35FGLJUq7DkWQSB0AAAAJwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMCJBJ/PZ1+POgp4vV5lZWWpuLhYHk/s1HwDsejFFeZ+Gv262PtpxJNnl5p7h9yeE5keHlMW7zeODb063eFM6maqZb6SNMTRnKs/YxYVFSvD41FqkpQZfYerQeBMBwAAok+HC4QOAADgBKEDAAA4QegAAABOEDoAAIAThA4AAOAEoQMAADiRHOkJAIgu9OKoG1svjucKzT08JGlA18D6eNj6cEjR2YtjxjLLsUhIcDeROpiQLWV4pGT+HA8bDi0AAKJPhwuEDgAA4AShAwAAOEHoAAAAThA6AACAE4QOAADgBCWzgKQH5pcbx8bemOJwJgi1u16uNI5NvDkpLPcZaEmsP9FYEutP3pXhORbhQMls+HFoAQAQJbMuEDoAAIAThA4AAOAEoQMAADhB6AAAAE4QOgAAgBOEDgAA4AR9OgBFphfHUwsPWMfv6J7maCaxbcxr5h4rkjTxZvfP7Z/eKrOO39sj1dFMUB/06Qg/Di0AAKJPhwuEDgAA4AShAwAAOEHoAAAAThA6AACAE4QOAADgRILP5/NFehL+eL1eZWVlqbi4WB6PJ9LTcW7m8n3GsYFXxM6y0QidWCy3tc05GueL0Ji2dL9xrKwiwbrtcEevi+rPmEVFxcrweJSSKKUkSU2pbA45+nQAAKBDJbNJ/8k5k6+SROgIOS6vAAAAJwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJSmZjAL046mbES5XGsSd7JzmcSfiFs6/FnJWlxrG+nTMC3q9tzs8sMfdykKTBV6UHfL/RyNZ7x9Y5qdJn72sxKCf6jlOjJPMD8jdfW4+PcDzW6qXtJSkxQfq+VEpNpl9HKBE6AADQkX06qtGvI7S4vAIAAJwgdAAAACcIHQAAwAlCBwAAcILQAQAAnCB0AAAAJyiZjQLz3vVax3td7nE0k7qbuthcPz/k6sj0CmhovTjCZcYyc48IScq7MvBeHIGqsvSm8Ge6n8dTWWXubfHDfvNr5v7rGwU8p+cK7XMKtPfOVD/9TGxGvVJhHT/9uHLjWIql14ZkfzwDugbeZ6iqKuBNA3J4n45q1f06DkfvjsAROgAAUO19OmpD747AcXkFAAA4QegAAABOEDoAAIAThA4AAOAEoQMAADhB9UoU8FcSG66lxm1e9lPGO+Tq6CvjtXnhHfMxlKTfZbsvE7WVHUtSeoq5XrDCUgYqSQmW4eREe/njSyvNz33vzubn3badJK3bbS5BHf1be5n15EUHjGP53ewlmePfOGgcs5XFPrnQfJ+SdHxapXEsOYg/554vNL9Wh1xlf53ePMv83Ga3Mh8HSfpwh7lso1OrwEt1g7HzgNsy+NpKZmtTWxltNcpp7QgdAACo7iWzNpTT2nF5BQAAOEHoAAAAThA6AACAE4QOAADgBKEDAAA4QegAAABOJPh8viAWlXbD6/UqKytLxcXF8nhiqz9EPPLX32BEd3NNmq1HQZKfiLyv3HyDQTn2PhC2XihlFfaeGAct47Z+Gidm2Jca/3i7ec4Tbw68f4HtGEvS+j3m3hXb95nvt/1J9uf99pzAlzgPxpA55n4nZ1mWc0/y08/E9jp+YkGZdduR15hrKm09Zb7ZZa/F/PpH8/NzfjPzY5XsPUtGvWJ/rTayvDf/12ue01lN7fsdc0OKdTxUqj9jFhUVKyPIz5iURKmqjp+q8djTgz4dAAAoNH066iMee3pweQUAADhB6AAAAE4QOgAAgBOEDgAA4AShAwAAOEH1iiO2MsW0ZHt9VamlFHTgFYGXIc5cvi8s+7XNV5Jum20uYZx+i3np7unLzPOVpL9tN3/tfJB1S6lv58CXts94zLxk+CUecxli3zPtpavBlMUOeMF8jJ/7nf2xDrWUmD7bz/bc2l8zTy8yL49e5bOXJe+1vKbe/dZ8jCVp2SDbnAMvHXhmifnxvLTRXqKdZtk2wfK34M4y+3uraYr5d8kJaZXWbTtNNY8VDbF/VDz0F/N74K97zNsen2J/jU9ZbD5OQ6+2H+NA1HVp+1BJTJC+t/8aqFUsl9oSOgAAkPuS2UDFcqktl1cAAIAThA4AAOAEoQMAADhB6AAAAE4QOgAAgBOEDgAA4ARL24fIq3/dax3/zmuuTh7WLfT15uE2y9Lj49YgenzYzLYs+S3Z+4OsK7H3crjAsiz75h/ty2ufkG7uf2Bb9n7dHvt+W3rMy37f2yPwernhL9n7NZRVmuc8zdKnY9pSc08FSUqxLBW/cqv9PeCx9LLZss/+t1PrDHPfkU9KzNvusK+6rryfmntTjNtof24Hn2ye0ymNzc/PCY3tk7rrC/N773w/L5nPS83P3wUZ9ufnHI/58ZyTWW4cm7bePqnmlqYOfxtq3bReQrm0vQspiVJVgJ/cke7xQZ8OAAAUO306ghHpHh9cXgEAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAE3FXMltQ5LWO53YKrEb7h332Q2nrIfHMEnt/g8FXBdbH49Ip9vH3LHXu/npi3HpFhnHsymn2++1ysrm/Qaal58UOr73XxoM3mHsjTF9m7isiSWWWfhr/fZ39fp9YUGYcG3mNuTZthJ9+GU1Tzb0P8ufatz37eHNvhPVee01gx5PMz49kPhaDcuyv06mLza9zWx8OSZra1/z+8fd6e7qPedufjt1mHDv1xFOs+135nflYPP1z++vtm93mbY9LMz+3D32wy7rfRseZ+3Qc9NPXofFB83v+63L776iBZ5tf548v+8I41vm/2lv3++837zcPDh1v3TYQE7KljOhv0xGUxATpe/uv9yOEuq9H3IUOAABqEw99Ouor1H09uLwCAACcIHQAAAAnCB0AAMAJQgcAAHCC0AEAAJxgafsQmbvSXorbp7N53mNeM5c3SlILj3l81GrzdiWj7EvMP7vUXNZ3e054lqf3505LGem5J9hKOf2XbNrYlmW3ldNK0vDugX3dfdZye1nlZ9+bvzL+7DZzOa0kdcs0F6Z1OeWAddsf9pu33b4vyTj23QH7cWpiqZW74ET7cztpnflYdDvBfiwSLdM67wRzufM2r724783t5hLts9Ltv1Y/3PGdcezG0080js3YZi+VLr3HXIr7k0nWTdUt03wc95Tbn9vvC83lq2XNfmYcS6y0vxa/Pvdm49jOO62b1kusLW3vUkqiVFXLyznQUlpKZgEAECWz9RFoKS2XVwAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAE4QOAADgBCWzR5m0wFw3Xukz16rffa29tvv5QvNawg/eYF4m/hBzP4Dbc/xsavHxt+basB/fMvcvkKT0ZHNN/7Bu9n4Ztv4Uk3qb+4OMeS3wl6utJ4kkvbvVPOeT0+x9IPo9b+7J8O6m7caxsR2Ot+43JdG835O85v1KUvqJPzWOvb3Z/vwcsLSC8FoORacT7D0kCv/xd+NYVvYvrNve1trcx2PxdnNvCklKtfxp1SzV/Joad6P5fSdJ46yj9r4WXZ852Tj2taXlT9+T7ftNnWh+zUw8x94TY3zxHuu4zW09RxnHPtppPo5fm9vjSJJaOv6zOB6Wtg+VxATpx7L69+ogdAAAIPp01FcgvTq4vAIAAJwgdAAAACcIHQAAwAlCBwAAcILQAQAAnCB0AAAAJxJ8Pp+5sDtKeL1eZWVlqbi4WB5P5Iqo/2TpXdHCU2Hdtm8Xf704osvQOfbeFOdmmvsm3NHdXnP2P6+VG8dKy819CM7MNG8nSYNyzP0nfv+K/flJtsTvl77Yat12WLufGMf2VZh3vH1fknW/U/uat01/3N4TY//d9n3bzLT0URn7913GMW9qU+t+H/wvcz+NT763192d3sT8/DVPtz+3s9ebXxefHTBvO+pU+6/GNz742Dg29IqLrNuu2Wk+Fq+sLzGO/eEC++8R23H8ZK+9x8eXw81j6Y/bj/FZO4qNYzlZWcaxln5+byZZpuzv90x9VH/GLCoqVkYEP2OiUYIO9eSoTWoyfToAAAgIfTqONfkqqXkI/2bm8goAAHCC0AEAAJwgdAAAACcIHQAAwAlCBwAAcCLuqldeXGFeYl6S+llKW+/tYa4NKiiyl3Pa2JZ6l6RbrzAv9x4M27GY0tf+deUZy+wltTZnWMpt/77NXN7YOMV+n12eMY9d+RP7tsmWZeT7/9cp1m3v6WFeuvvWF8z3+1OPvex1yBzz2JPnmcu3JeniYe8Yx67NzrFue9rx5mPRqU1L49gZlrJWSRre3XycnlliX+N8X7n576Pv99t/jV3Twvx6G9LU/L6945/2suO9j3Uwjt04015u+/1Bcy3o+IvN74GKKvvruEW6+TW1/0f7cTrjSfPYtAvtr7fCzRcbx17+Zodx7MFfHmff7xbzsbjDumVg4mVpe1sZ7NFSQ5wS4i50AABQm3gpmQ11GWx9cHkFAAA4QegAAABOEDoAAIAThA4AAOAEoQMAADhB6AAAAE7EXcmsv14PgcrtFHhxd7j6cPhj60ny0kqvddvEBHNenbzogHXb/G7m+/W9Y+4d4rO3PlCv08z9Tn4ss/dcuMfSg8VfHxXJ3H+iXXNzj4hmfpZkX/WN+TWV5afG/orLrzaOJSXa3wPrdpuXXZ/d33zHf3rLvt8xr5l7Ynzzo71O8Vcnm/tEbNtnf25Pt/QPWbHF/N7be6/9ID/yhvm57XGqvW/Pqm/Nj/cHy+NZY3luJOmSFub3Xutd9l/3V/7E/HhuX23ulyFJj7Y1Pz/ZP21q3m6N/Xdf71bmOUn2YxGIWOvTUZ9+G4cLde+N+oi70AEAQG1irU9HJPttBIrLKwAAwAlCBwAAcILQAQAAnCB0AAAAJwgdAADAibirXrnh0iaRnsIx5r1rL0/dX2HOhgO6Bl5uO/Z1c1lferK5DFSSMiylx6WWZcj9uSXb/FVsW8mlJB2fZl7W21YS64+/kuZbnjfX8s7ub/4q/GxLebAkvX27bdQ+J1vZcuFW+9fzrzvNPK+pi83lpy089pLZvpYS7SFz7NvmdzPPeeri/dZth1xtLvfs/Zz5ubO9PyTpgd+aSzbbTbaXc+acZD6Oo35jfq0+tdBejj4ox/xYq4I4Ti2W2cvG564zvx5vPt287Rd+1qefutj8ng6HSJbMBlL+GsnS10DF4JQBAAi9SJbMxmL5ayC4vAIAAJwgdAAAACcIHQAAwAlCBwAAcILQAQAAnCB0AAAAJxJ8Pn8Lhkee1+tVVlaWiouL5fHE0LrDIfKCpZ/D7yx9LYIx18/S9vstvTj2HrQvNW4z8hpzjwLbcZDsx+L5Qvu2Nv27hucYz1lhn5Otr0WkzFxu7rkw0E8/k2C2DcYMS4+JvCvN99vX0sNDki5vae57kZps3/Z/LUvUt2xi7g9im69k71mSnGSf022WfV833bqpzmxi7qeRmWoea5Zu78ORkmies79jUR/VnzGLioqVEcLPmPr03khNlpoG3k4oZtCnAwAAhb5PR7z03qgPLq8AAAAnCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcIHQAAwAlKZkPklSJ7X4ubOgVe+x1oL47H3iqzjt/Tw1wU3qdz9PVDCaajTLh6bQSljvX70aSyyjz29CJzjwhJGtbN3Fdh6hL7tra+MHdda29uYOvnMMvSO2TOAHsfiAfmpxjHxt5oHvNn7OsBb6ohV6cHvO10Sz+TN26zH4t75pnH/t/15p4kv3+lwrrf044z9ywJhwnZUkY9f/XZenGk8gl7DA4JAAAKrE8HvTjqh8srAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJlrZHQB5/21yOe7efEsaG5n9eM5f1/c8NgZdOxhpbyaVkXzq9oRn+kn3J9j/3TjKOPbHA/N4aeU3g7y3be1ayl0M3TrF/TOR3C+HSrBFQ16XtayuPjZcl6UOFklkAAOS/ZJby2OBxeQUAADhB6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAATtCnowEbY+kfIUkPxlEPCX9sPSbC1V/i+cJS63hZpflvgkE5gS9hDoTazOXm98/AK8zvn8mLDlj3a/t0uqN76HqD2Pp0HN6bg54cwaNPBwAAqr1PB705QovLKwAAwAlCBwAAcILQAQAAnCB0AAAAJ/yGjnXr1rmYBwAAaOD8lsy2b99eEydO1GWXXeZqTsegZBZwx1bGGOtLmB/Nttz73dfaayP/+JeDxrH0ZHsngrv87Btu1VYyW10qS5lsaPk90+H1ejVkyBA9//zzDqYDAEBk/P4dacTSQ/8lJhwqlSVwhJbf0NGkSRNVVlZqwoQJuv/++1VRURHUHe7bZ24iAwAAGi6/oaOgoEBt2rSRz+fTa6+9pgEDBmj37t31vqMffvhBEydOVOfOnQOaKAAAiG1+Q8dpp52m+fPnq0OHDvL5fPrkk0+Um5urtWvX1ukO1q9fr9GjRys7O1vTp0/Xjz/+GPSkAQBA7KlTyWyTJk00Y8YM9enTRz6fT1u2bFGvXr1UVFRk3GbVqlUaNmyYunfvrvnz56usrEw+n09ZWVkhmzwAAIgdde7TkZiYqNGjR2vMmDFKSkqS1+vV0KFD9dxzzx1xu8LCQt18883q06ePVqxYoaqqKklSdna25s2bpzlz5oT2EQAAgJhQ7wXfbrrpJp166qkaMWKESkpK9Oijj+qbb77RRRddpFmzZmnDhg2SJJ/Pp0aNGqlHjx4aOHCgTjvttFDPHQAAxJCAl7bfvHmzBg8erHXr1ikh4dC6v9W7atKkiXr16qXf/e53OvHEE4OeJH06EKxH3zT3Y5CkUb+Jvrq4CW+a+0D8/jeNHM4EoTb+DfNz+9/X8dy6dnSfjgRJ6SmUy4ZDQEvbf/vtt3rppZe0Y8eOIwJHQkKCcnNzNWrUKGVksBYwACB2VC9tP/kqAke41Ct0fP3115o1a5YWLlyoioqKmjMbrVq10pYtWyRJixcvVk5Ojjp06BD62QIAgJhVpy+Sfvjhh7rtttv0m9/8Rm+++abKy8vl8/l0ySWXaObMmSosLNS4ceOUlJSkvXv36vbbb9fcuXPDPXcAABBD/J7puOGGG7RmzRpJhy6hJCUlKScnR3l5efrZz35Wc7uePXuqTZs2Gj58uEpKSvTQQw9p7dq1Gj16tBITWcwWAIB45zcNfPHFF/L5fEpLS1OfPn20dOlSTZw48YjAUe3iiy9WQUGBTjvtNPl8Ps2bN095eXnau3dvWCYPAABih9/QkZmZqfz8fK1YsUKjR49Wq1atrLdv3bq1Xn31VXXs2FE+n08ffPCBcnNztXHjxpBNGgAAxB6/oWPlypXKz89XZmZmnXfq8Xg0ffp09e7dWz6fTxs2bFBubq4+/PDDoCYLAABiV8B9Oupq7ty5evjhh1VRUaHk5GTdf//96tWrV732QZ8OIHRmLbev9HzrFY0D2u+8d73W8V6XB/7e/fPCA8ax4d3TAt5vMJ4rNB/HAV0DO4Y40oxl5mOcd2XojvHRfTpSk6TM9JDtHocJ+zc8+/Tpo2nTpqlp06aqqKjQmDFjwn2XAADU2+/fkUYslSqqIj2ThstJWUnHjh01b948tW7d2sXdAQCAKOSslvX0009XQUGBfvGLX7i6SwAAEEWcNtA4/vjjNWvWLJd3CQAAooTzrl3JyQEt9wIAAGIcrUIBAIATYS+ZDQVKZgEE6vevVBjHJtwUfWde7yswz/fh3Oibb0NAyaw7nOkAAECUzLpA6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAExR9A3AiUv0ywrXv+181P56HegZ+n/TiiJwJ2VKGR0rmz/Gw4dACACD6dLhA6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAExSEx4C5K73GsT6dPQ5nEhovWR5PpS/BONavS0Y4phN3Zr9Tah2/JTs8xzmcvTgiwdaLY85K+zGuspRkhuv4wz/6dIQfhxYAANGnwwVCBwAAcILQAQAAnCB0AAAAJwgdAADACUIHAABwIsHn8/kiPQl/vF6vsrKyVFxcLI8n9kpEAQRn6uL9xrEhV6eH5T6fXmS+T0ka1i089xuMmcv3GccGXtHY4UxC47lC8+MZ0DV0j6f6M2ZRUbEyPB6lJkmZ0ff0Ngic6QAAQJTMukDoAAAAThA6AACAE4QOAADgBKEDAAA4QegAAABOEDoAAIAT9OmIcXe+VGkcm9Q7yeFMQmOuZdn7Pp157iPNtmR7384syd5QzXvX/L6UpF6Xx/Z7kz4d7nCmAwAA0afDBUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHCCklkAiBFpE+1lFQfu4u/IQFAy6w6vUAAARMmsC4QOAADgBKEDAAA4QegAAABOEDoAAIAThA4AAOAEoQMAADiRHOkJwL/57+01jt14aRPj2IsrzMuQS1K/Lg1rKfKXVpqX3+7dOfr6u7xmeV4l6QbLcxuNXn/f/nh+++vYejz+vPpX8+PteVngj7WgyPw6PnCX/XVsW4I+mOXnbY9VCu7xRpMJ2VKGR0rmz/Gw4dACACD6dLhA6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAEwk+n88X6Un44/V6lZWVpeLiYnk80ddvAQBCpdET5l/JB0cmOJxJ/Kj+jFlUVKwMj0epSVJmeqRn1TBxpgMAANGnwwVCBwAAcILQAQAAnCB0AAAAJwgdAADACUIHAABwgqXtAT+mL9tnHb/tysaOZhJ505buN44NyrHXGD6zxLzt4KvCV584c7n5+Rt4RfQ9d5TFRg5L24cfhxYAAFEy6wKhAwAAOEHoAAAAThA6AACAE4QOAADgBKEDAAA4QegAAABO0KcD8COe+nD4468Xh01yonnJ9mg0acEB6/id16Q5msn/oWdMeNGnI/w4tAAAiD4dLhA6AACAE4QOAADgBKEDAAA4QegAAABOEDoAAIATlMwCcCIvQuWcgS5f/1VJoxDPpG4efbPMODbqN5TEhhMls+HHoQUAQJTMukDoAAAAThA6AACAE4QOAADgBKEDAAA4QegAAABOEDoAAIAT9OlAXHi+sNQ63r9rhqOZIBC2ZebDtcT81L6R+ZvsxMaVAW87bel+49ignPSA9xsv6NMRfhxaAABEnw4XCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcIHQAAwAlCBwAAcII+HYgL/vpwzF3pNY716ewJ9XTqZM5Kc2+Rvp0bVl+RF96x91G58xrz452zwr5t3y6xdawaJfsC3jaYXhzx9HozoU9H+HFoAQAQfTpcIHQAAAAnCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcomT3KrOX7jGO3XtHY4UzgUqTKYm3ipUxRkn6XHfhjjbWSWCnw8tTnCs2/nyRpQNfAf0dVRqBi4/lCe7mzv1L3UKNkNvw4tAAAiJJZFwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHCCPh1HoRcHQsnWVyGYngrBmPeu1zre63Jzz5KXVpq37R3GXicNrX9OoD1YwvmaSU70BbztK0Xm18VNncyvC9d9OPyhT0f4cWgBABB9OlwgdAAAACcIHQAAwAlCBwAAcILQAQAAnCB0AAAAJxJ8Pl/gdVKOeL1eZWVlqbi4WB5P9C1BDiA4E98us47fdW2qo5nU3dOL9hvHhnVLdziT2PXEAvvzPvIaN8979WfMoqJiZXg8Sk2SMnkKw4IzHQAAiJJZFwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHCCpe3j2GNvmWvk7+kRfX0RwulPlmNxb5wdi0BNXWLuWyFJQ64yNz6Ixj4c/tCLI3iu+nDUFUvbhx+HFgAA0afDBUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHCC0AEAAJygT0cci8ZeHH9eeMA4Nrx7Wtjul14cwbP14cCRIvE6n/i2uReNFJu9UkKNPh3hx6EFAED06XCB0AEAAJwgdAAAACcIHQAAwAlCBwAAcILQAQAAnKBkFrV6cUWpdbxfl4yw3G84y2IRvGlLzcvXD8oJvGT26UXm/UoNbxn5SLzO/ZXEPrPE/BwMjpNyaEpmw49DCwCAKJl1gdABAACcIHQAAAAnCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcSfD6fL9KT8Mfr9SorK0vFxcXyeDyRng4A6P5XK4xjD/WkBVIo3FdgPsYP54buGFd/xiwqKlaGx6PUJCkzPlqTOMeZDgAARJ8OFwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHCCYnLU6uV3vdbxmy+nX0pDZXvug3nenyvcZxwb0LWxddunF+03jg3rFpmGCtHYi2PmcvMxHniF/RhHo1D24qiLCdlShkdK5s/xsOHQAgAg+nS4QOgAAABOEDoAAIAThA4AAOAEoQMAADhB6AAAAE5EX80XokI4S2JnWcr6bg1TWd+zS833KUm358ReOWG4hOu591cWaxOpsthYs788PH9Hzlhmf//kXdkw3j+UzIYfhxYAAFEy6wKhAwAAOEHoAAAAThA6AACAE4QOAADgBKEDAAA4QegAAABO0KcDzlX53N9nMH04nlp4wDp+R/e0gPeNunlgfrlxbOyNKQ5nEt3yu4XntdhQ+nD4Q5+O8OPQAgAg+nS4QOgAAABOEDoAAIAThA4AAOAEoQMAADhB6AAAAE5QMgvnYq38jpLYyKMsFi481vVQyWwqn4xhw6EFAEDSiRmSJyPSs2jYuLwCAACcIHQAAAAnCB0AAMAJQgcAAHCC0AEAAJwgdAAAACcIHQAAwAlCBwAAcILQAQAAnCB0AAAAJwgdAADACUIHAABwgtABAACcIHQAAAAnCB0AAMAJQgcAAHAiOdITqAufzydJ8nq9EZ4JAKCh4bPFnZgIHaWlpZKkTp06RXgmAAAgUAm+6tMIUayqqkrfffedMjIylJCQEOnpAAAakOqPQY/Hw2dMmMVE6AAAALGPL5ICAAAnCB0AAMAJQgcQY/bs2aOLLrpIbdu2VadOnVRRUeF3m8rKSuXl5alt27Zq27atFixY4GCmAHAkQgcQY4477jjl5uZKknbs2KGlS5f63eaRRx7Re++9J0kaMmSIrrnmmrDOEQBqQ+gAYlD//v2VkpIiSXrhhResty0oKNDs2bMlSTk5ORoxYkTY5wcAtSF0ADGoRYsW6t69uyRp9erV+uc//1nr7T766CONHTtWkvSzn/1MEyZMoCQQQMQQOoAYlZeXVxMgajvbsWnTJg0fPlzl5eVq3ry5pk6dqvT0dNfTBIAahA4gRp111lk1XXqXLVumbdu21Yzt3btXgwcPVklJiVJTUzVlyhT95Cc/idRUAUASoQOIaXl5eZIOVafMmTOn5ueRI0dq3bp1kqTx48fr/PPPj9gcAaAaoQOIYb/4xS904YUXSpJeffVV7du3Tw8//HBNpcrQoUOpVAEQNWiDDsS45cuXKz8/X5J0ySWX6IMPPpB0qFLlySef5IujAKIGoQOIcT6fT1dffbXWr19f828///nPNXfuXL44CiCqcHkFiHEJCQkaOHBgzf83b95cU6ZMIXAAiDqEDqABaN26dc3PvXr1olIFQFQidAANwL///e+an88555wIzgQAzAgdQAPw1Vdf1fx87rnnRnAmAGBG6AAagOozHccdd5xatmwZ4dkAQO0IHUCMKy8v19q1ayVJbdu2jfBsAMCM0AHEuHXr1qm8vFwSl1YARDdCBxDjvvzyy5qfCR0AohmhA4hxhA4AsYLQAcS46i+RpqSk6IwzzojwbADAjNABxLjqMx1nnnmmUlJSIjwbADBj7RUAAOAEZzoAAIAThA4AAOAEoQMAADhB6AAAAE4QOgAAgBOEDgAA4AShAwAAOEHoAAAAThA6AACAE4QOAADgBKEDAAA4QegAAABOEDoAAIAThA4AAOAEoQMAADjx/wFoMn1dB4OdNgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 600x600 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "iteration = 2\n",
    "title_size = 20\n",
    "\n",
    "joint = est[iteration]\n",
    "df = generate(joint)\n",
    "\n",
    "g = sns.jointplot(data=df, x='x', y='y', kind='hist', bins=(m, m), color=data_color)\n",
    "g.ax_joint.set_xticks([])\n",
    "g.ax_joint.set_yticks([])\n",
    "g.ax_joint.set_xlabel(r\"$Y$\")\n",
    "g.ax_joint.set_ylabel(r\"$X$\")\n",
    "\n",
    "for patch in g.ax_marg_x.patches:\n",
    "    patch.set_facecolor(true_color)\n",
    "\n",
    "g.fig.subplots_adjust(hspace=0.01, wspace=0.01)\n",
    "g.fig.suptitle(f\"Iteration {iteration}\", y=1.02, fontsize=title_size)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "dl",
   "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.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
