{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "3b2940b0",
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "import os\n",
    "from pathlib import Path\n",
    "import numpy as np\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "plt.rcParams.update({'font.size': 6})\n",
    "\n",
    "import sys\n",
    "sys.path.append('..')\n",
    "\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "from omegaconf import OmegaConf\n",
    "\n",
    "from probssl.data.base import prepare_data\n",
    "from probssl.methods import METHODS\n",
    "from probssl.utils.offline_model_eval import get_states_and_decodings\n",
    "\n",
    "from probssl.data.dot_motion import get_video_frames, save_videos_as_gifs_and_pdfs\n",
    "\n",
    "# seed everything\n",
    "from lightning.pytorch import seed_everything"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "65d7c5cb",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_loader(model_name):\n",
    "    seed_everything(3, workers=True)\n",
    "\n",
    "    folder_name = \"../trained_models/\" + model_name + \"/\"\n",
    "    # get name of the most recent model by folder creation time\n",
    "    name = sorted(os.listdir(folder_name), key=lambda x: os.path.getctime(os.path.join(folder_name, x)))[-1]\n",
    "\n",
    "    pretrained_checkpoint_dir = folder_name + name\n",
    "    args_file = os.path.join(pretrained_checkpoint_dir, \"args.json\")\n",
    "    args = json.load(open(args_file, \"r\"))\n",
    "    cfg = OmegaConf.create(args)\n",
    "\n",
    "    # prepare data\n",
    "    _, val_loader = prepare_data(cfg)\n",
    "\n",
    "    return val_loader\n",
    "\n",
    "def load_model_and_data(model_name, val_loader):\n",
    "    seed_everything(3, workers=True)\n",
    "\n",
    "    folder_name = \"../trained_models/\" + model_name + \"/\"\n",
    "    # get name of the most recent model by folder creation time\n",
    "    name = sorted(os.listdir(folder_name), key=lambda x: os.path.getctime(os.path.join(folder_name, x)))[-1]\n",
    "\n",
    "    pretrained_checkpoint_dir = folder_name + name\n",
    "    args_file = os.path.join(pretrained_checkpoint_dir, \"args.json\")\n",
    "    args = json.load(open(args_file, \"r\"))\n",
    "    cfg = OmegaConf.create(args)\n",
    "\n",
    "    # build paths\n",
    "    ckpt_dir = Path(pretrained_checkpoint_dir)  \n",
    "    args_path = ckpt_dir / \"args.json\"\n",
    "    ckpt_path = [ckpt_dir / ckpt for ckpt in os.listdir(ckpt_dir) if ckpt.endswith(\".ckpt\")][0]\n",
    "\n",
    "    # load arguments\n",
    "    with open(args_path) as f:\n",
    "        method_args = json.load(f)\n",
    "    cfg = OmegaConf.create(method_args)\n",
    "    cfg.optimizer.batch_size = 8 # set batch size to 8 for evaluation\n",
    "\n",
    "    # build the model\n",
    "    model = (\n",
    "        METHODS[method_args[\"method\"]]\n",
    "        .load_from_checkpoint(ckpt_path, strict=False, cfg=cfg)\n",
    "    )\n",
    "\n",
    "    # move model to the gpu\n",
    "    device = \"cuda:0\"\n",
    "    model = model.to(device)\n",
    "\n",
    "    # get images and features\n",
    "    data, labels, latents, decodings = get_states_and_decodings(device, model, val_loader)\n",
    "\n",
    "    return data, labels, latents, decodings, model, cfg\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "e552cabd",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Seed set to 3\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Pre-generating videos...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "8it [00:00, 48.65it/s]\n",
      "Seed set to 3\n",
      "Collecting features: 100%|██████████| 1/1 [00:00<00:00,  3.94it/s]\n",
      "Seed set to 3\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Pre-generating videos...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "8it [00:00, 47.62it/s]\n",
      "Seed set to 3\n",
      "Collecting features: 100%|██████████| 1/1 [00:00<00:00,  4.80it/s]\n",
      "Seed set to 3\n",
      "Collecting features: 100%|██████████| 1/1 [00:00<00:00,  4.90it/s]\n",
      "Seed set to 3\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Pre-generating videos...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "8it [00:00, 46.42it/s]\n",
      "Seed set to 3\n",
      "Collecting features: 100%|██████████| 1/1 [00:00<00:00,  4.63it/s]\n",
      "Seed set to 3\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Pre-generating videos...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "8it [00:00, 37.86it/s]\n",
      "Seed set to 3\n",
      "Collecting features: 100%|██████████| 1/1 [00:00<00:00,  4.39it/s]\n"
     ]
    }
   ],
   "source": [
    "# model name is \n",
    "model_name = \"kalmanSSL_inferred_sigmas\"\n",
    "loader = get_loader(model_name)\n",
    "data_inferred, labels_inferred, latents_inferred, decodings_inferred, model_inferred, cfg_inferred = load_model_and_data(model_name, loader)\n",
    "\n",
    "model_name = \"kalmanSSL_figure8_procrustes\"\n",
    "loader = get_loader(model_name)\n",
    "data_8_ent, labels_8_ent, latents_8_ent, decodings_8_ent, model_8_ent, cfg_8_ent = load_model_and_data(model_name, loader)\n",
    "\n",
    "model_name = \"kalmanSSL_figure8\"\n",
    "data_8_stpgr, labels_8_stpgr, latents_8_stpgr, decodings_8_stpgr, model_8_stpgr, cfg_8_stpgr = load_model_and_data(model_name, loader)\n",
    "\n",
    "model_name = \"square_kalmanSSL_procrustes\"\n",
    "loader = get_loader(model_name)\n",
    "data_square_ent, labels_square_ent, latents_square_ent, decodings_square_ent, model_square_ent, cfg_square_ent = load_model_and_data(model_name, loader)\n",
    "\n",
    "model_name = \"square_kalmanSSL\"\n",
    "loader = get_loader(model_name)\n",
    "data_square_stpgr, labels_square_stpgr, latents_square_stpgr, decodings_square_stpgr, model_square_stpgr, cfg_square_stpgr = load_model_and_data(model_name, loader)\n",
    "\n",
    "model_dict = {\n",
    "    \"kalmanSSL_inferred_sigmas\": (data_inferred, labels_inferred, latents_inferred, decodings_inferred, model_inferred, cfg_inferred),\n",
    "    \"kalmanSSL_figure8_procrustes\": (data_8_ent, labels_8_ent, latents_8_ent, decodings_8_ent, model_8_ent, cfg_8_ent),\n",
    "    \"kalmanSSL_figure8\": (data_8_stpgr, labels_8_stpgr, latents_8_stpgr, decodings_8_stpgr, model_8_stpgr, cfg_8_stpgr),\n",
    "    \"square_kalmanSSL_procrustes\": (data_square_ent, labels_square_ent, latents_square_ent, decodings_square_ent, model_square_ent, cfg_square_ent),\n",
    "    \"square_kalmanSSL\": (data_square_stpgr, labels_square_stpgr, latents_square_stpgr, decodings_square_stpgr, model_square_stpgr, cfg_square_stpgr),\n",
    "}   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "851aef1f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# plot settings\n",
    "P_height = 0.7\n",
    "P_width = 3\n",
    "plot_folder = \"plots/paper/\"\n",
    "Path(plot_folder).mkdir(parents=True, exist_ok=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "c9dee44f",
   "metadata": {},
   "outputs": [],
   "source": [
    "batch_ind = 0\n",
    "video_ind = 2\n",
    "model_examples = {}\n",
    "for model_name, tpl in model_dict.items():\n",
    "    data, labels, latents, decodings, model, cfg = tpl\n",
    "    data_i = data[batch_ind][video_ind]\n",
    "    labels_i = {key: labels[batch_ind][key][video_ind] for key in labels[batch_ind].keys()}\n",
    "    latents_i = {key: latents[batch_ind][key][video_ind] for key in latents[batch_ind].keys()}\n",
    "    decodings_i = {key: decodings[batch_ind][key][video_ind] for key in decodings[batch_ind].keys()}\n",
    "    model_examples[model_name] = (data_i, labels_i, latents_i, decodings_i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "53fbc78b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAfAAAAEuCAYAAABxmEC5AAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAac1JREFUeJzt/XmQZMd1349+8+61dlXvPdPds68AZgAQAEFIIMVFEkVIsqwgqZ/9C+lHPYlU+FmhH73RZsi0GeFQ6OegTfrFe8GnUIQoWZZkP8myZVpPlGmRJkVIAAWCJAgOBrN39/S+VXXtt+6Svz9qeqa7uuvk7a7qpQbnE4Ege/JWZt48mXnu9j1HSCklGIZhGIbpKrSD7gDDMAzDMDuHHTjDMAzDdCHswBmGYRimC2EHzjAMwzBdCDtwhmEYhulC2IEzDMMwTBfCDpxhGIZhuhB24AzDMAzThbADZxiGYZgu5FA4cF3X8fjjj9//b2Ji4kD68eqrr+Kxxx7D6dOn8Su/8it4qwWpOyx2qNfr+NjHPoazZ8/i/Pnz+OM//uMD6cd+cljG/ld/9VcxNjaGZDK56d8/+9nP4uLFi7h06RLe+973YnJy8kD6t9ccdjtMTU3h3e9+N5544glcunQJf/Znf3Yg/dtrDoMdKpUKXnjhBZw/fx6PPPII/tk/+2dbjvnjP/5jCCHwrW99a9/7BwCQh4BEItGyLAxDGQTBvvTj6aefli+99JIMw1C+//3vl3/2Z3+2L+0eFg6LHf7Fv/gX8ld/9VellFIGQSCXlpb2pd2D5LCM/UsvvSRnZ2e39OerX/2qLJfLUkopP//5z8sPf/jD+9Kf/eaw2+GjH/2o/PznPy+llPLKlSvy2LFj+9Kf/eYw2KFcLsuvfvWrUkopXdeVP/iDP7jJJxQKBfn888/Lt7/97fKVV17Z8/5sx6G4A29mYmIC586dw8/93M/h0Ucfxd27d/H3/t7fw1NPPYVHHnkE//Jf/sv7xx4/fhyf/OQn8fjjj+Opp57Ct7/9bfzoj/4oTp06hd/4jd+4f9xnPvMZPP3007h06dKm368zNzeHQqGAZ599FkII/NzP/Rz+5E/+ZD9O99ByEHYAgC984Qv45Cc/CQDQNA39/f17e6KHkIMa+2effRYjIyNb/v3d73434vH4/WOmp6c7fMaHk8NmByEECoUCAGBtbQ1Hjhzp8BkfTg7CDvF4HO9+97sBAJZl4cknn9w07z/1qU/hn/7TfwrHcfbwzBUcyGVDE5qmycuXL8vLly/Ln/qpn5J37tyRQgj50ksv3T9mZWVFSiml7/vyXe96l3zttdeklFIeO3bs/hXpxz/+cfnYY4/JQqEgFxcX5eDgoJRSyv/xP/6H/OhHP3r/yu2FF16QX//61zf14ZVXXpHvfe977//9l3/5l/KFF17Y0/M+bBwGO+RyOTk6Oir/wT/4B/KJJ56QH/zgB+X8/Px+nP6BchjGfiPUHdDf//t/X/6rf/Wv2j7nw8hht8Ps7Kx89NFH5dGjR2Umk5Hf+ta3Onr+h4XDZodcLidPnDghb926JaWU8tVXX5U//dM/LaWU8l3veteB3YEbB3fp8IBYLIbvfve79/+emJjAsWPH8Oyzz97/tz/8wz/Eb/7mb8L3fczNzeGNN97ApUuXAAA/+ZM/CQB47LHHUCqVkEqlkEqlYNs28vk8vvzlL+PLX/4ynnjiCQBAqVTCjRs38M53vnP/TrILOAx28H0f09PTeO655/DZz34Wn/3sZ/GP//E/xn/4D/9hH0bg4DgMYx+F3/u938O3vvUtfP3rX2/zjA8nh90O//E//kd85CMfwT/6R/8IL730En72Z38W3//+96Fph/Jh6q45THbwfR9/5+/8HfzKr/wKTp48iTAM8Q//4T/E7/zO7+zdAETkUDjw7UgkEvf//507d/Bv/s2/wSuvvIJsNouPfOQjqNVq98tt2wbQeNy6/v/X//Z9H1JKfPKTn8Qv/dIvtWzv6NGjmx6PTE9P4+jRo508pa5kv+3Q19eHeDyOn/7pnwYAfOhDH8Jv/dZvdfq0uoL9HnsVf/EXf4Ff+7Vfw9e//vVNbTzsHCY7/NZv/Rb+/M//HADwjne8A7VaDcvLyxgcHNxVfd3EQdnhYx/7GM6cOYOPf/zjAIBisYjvf//7+KEf+iEAwPz8PH7yJ38SX/ziF/HUU0914Eyj0xWXbYVCAYlEAj09PVhYWMCXvvSlHf3+R3/0R/GFL3wBpVIJADAzM4PFxcVNx4yMjCCdTuPll1+GlBK/+7u/i7/1t/5Wx87hYWA/7CCEwE/8xE/ga1/7GgDgK1/5Ci5evNiR/ncz+zH2FN/5znfwS7/0S/jiF7/4lnAWrThoO4yPj+MrX/kKAODq1auo1WoYGBjYUR8eBvbLDv/8n/9zrK2t4d/9u393/996enqwvLyMiYkJTExM4Nlnnz0Q5w0c4jvwjVy+fBlPPPEEzp8/j7GxMfzAD/zAjn7/Iz/yI7h69Sre8Y53AACSySR+7/d+b8tG9PnPfx4f+chHUK1W8WM/9mP4sR/7sY6dw8PAftnhX//rf42f/dmfxcc//nEMDAzgt3/7tzt2Dt3Kfo39Jz7xCfzBH/wBKpUKRkdH8Yu/+Iv49Kc/jX/yT/4JSqUSPvShDwFoOJIvfvGLnTm5LuKg7fBv/+2/xUc/+lF87nOfgxACv/M7vwMhRMfOr1vYDztMT0/j137t13D+/Hk8+eSTAIBf/uVfxi/+4i927kTaREj5FhM7MwzDMMxDQFc8QmcYhmEYZjPswBmGYRimC2EHzjAMwzBdCDtwhmEYhulC2IEzDMMwTBfCDpxhGIZhuhB24AzDMAzThUQO5PLzP//zm/6u1+tkvuxarXY/yk0rzp8/TwYhyOfzm8LjNROLxZTRb65du4Z6vd6yvLe3F2NjYy3LpZSbYvJux3povlaMjIzg13/918k6otJcz8zMDHzfb3l8T08PmbFISonr16+TbR4/fpzMCFav1zE1NUXWcefOHXie17JcZYcwDJXRlk6dOkVGCRscHLyf5axdPvGJT2z6e35+nrRDX18fTpw40bJcSom1tTWyTVXIhjAMUa1WyWNc1yXrcRwHqVSqrX6GYUi2MTQ0hE996lNkHTuheU3Mzs6StqhWq/czerXi2LFj5N40MTGB1dXVluWO49yPy90KVT+Xl5eVedcvXrxI9vPkyZNkpLa+vj587GMfI9uISrOP0HWd7JsQArqutyyXUiqj1MXjcViW1bLcsiycPHmSrGNpaQlBELQsr1arKBaLZD9V6y6dTiMWi7UsHxoawqc//Wmyju3gO3CGYRiG6ULYgTMMwzBMFxL5EXpz9qGbN2+Sj7dHR0fxzDPPkHW++OKL5KO20dFRZaD+mzdvkuVBEJCp9srlsvIxVSKRIPupaRr5qCiTyZD174SFhYVNf9+6dYu0QzabRblcJuvM5/Pk+VmWpXyERD2CAhqPJCksy4JpmmQb69l/WmHbNllHNpslf78TXnrppU1/X7hwAY7jtDze931cuXKlZbkQAo888gjZ5iuvvELOVdu2ceHCBbIO1WNN13XJV19CCGVymevXr5OPPg2jsykY3njjjU1/j4+Pk49Vq9Uq+TgTgPIReyKRIDOySSlx69Ytsg6VLTKZDJLJJFkH9WoLAFZWVkhbUK/Xdkrz/J2YmCBfX6piuGuahsuXL5PH1et1cu/RdV0536LMR9d1yfKRkRFlP6mxoF6lUEReSc1O0PM8skNSyk3p35oJwxC1Wo10HFJKcnCllGQf1qEGVkpJvpsFtp77duXUMZ3M1dtsaNd1yTFQlUdxvp7nkWMUJZy+aZrkOKg2MwDKzUxlB+p9206pVCpb2qbmqud5yk1ANQb1ep10rkEQKG0hhFCuB9VmYlmW0lZUHbvdrFrRPK66rpMXcp7nKS8WqYtioGFvqo4gCJQXzo7jkOMYxfmo5kytViPnXZT9MyrNF7CquaYqBxrfObUz11Q3Vxv70goppXJdmaZJ1uF5ntLX7QZ+hM4wDMMwXQg7cIZhGIbpQtiBMwzDMEwXwg6cYRiGYboQduAMwzAM04WwA2cYhmGYLiSyjKxZN3nmzBny833HcTA7O9uyfF1mRn0+r+s6wjBsWR4EgTKso0ry4jgOKXeTUirlICr5TjweJ3+/E5rlXNlslpSBpVIpZWjMXC6nbJOSokgpSTsBaq1lGIZKOZ8qfG8ikSD1v7uVamzHdvEJqPXguq5SW7yyskKWa5qGnp6eluXxeLxtrbvruqRueF36Q62pVCqFvr6+luWd1OMD2CLnWl5eVkoGVXOBGmegoROnwmdKKcm5CDTGieqn7/tKmZdKAqrrOqlXV/VxJzTr3mu1GrkvRJGRLS4uKucaJS+NGqJYJSNrVwIaRcq2GyI78OZFd/bsWVIHeevWLXzzm99U1kmdmGVZ5ASo1+tbAps009PTQy4SwzBIByulVG6spVKJXGjUOO2U5ok0OjpKnp/jOORmJKXE7du3yTZVcX6jaMljsRg5Dq7rKjdEVT90Xe/ohkSxXXxlyoGXSiVlXOfJyUlyPRiGgZGRkZbl8Xgco6OjZBuq2AvLy8tkAJL1+NWU3n5gYIDcVCnnvhua1+/ExARpi1QqhaGhIbLO4eFh0haVSoXc1IUQ5I0B0Bgnak1UKhXk8/mW5VHiYFiWRQat6eTNxcsvv7zp7+PHjysvqFXaaNXe+/jjj5PBvqrVqjJQl+pCKoqmX6UD72QskE317kmtDMMwDMPsKezAGYZhGKYLYQfOMAzDMF0IO3CGYRiG6ULYgTMMwzBMF8IOnGEYhmG6kMgysu0+91dpKanyKBrAKG1E+X07/YzS/l5p/DpB1L51UiO92zai9KETdXSCnbajssN6uWrN7DWqdbkuh1Gtmf2yg6ovu/mNECLSPOuErfYixeTDQlT/0AkfQcmVo6ToPShbRXbgzUEoNE0jtXP5fF6pUVQFSxBCKPNQU7pYAJienibrKBQKZIANIQSOHz9OtuE4Dpk/OJPJkL9vh3Q6TWpJo+TZvnbtGjkBL126hP7+/pblmqYp9aSqvOSe55Fa8ihac9d1ybFQ5XjeCc1zStW3dDqNp59+umW5lBJLS0tkHSMjI+RcsiwL6XSarCOZTJK27u/vx3PPPUf2k9KJA42135wvfSOdzEENAKurq5v+Vs1FTdOUwYuWl5fJ8vn5eXLfsG0b586dI+tYWVlRBhChEEKQQZqAxh5L6cCpdb1Tmue3KvBSKpUiNdyA2g7FYhFvvPFGy3LLsvD444+TdXzhC1/YMoc2Mjw8jPPnz5N1qOaTEILUxO82VkhkB75XtHtX0e7vo97RRb2DOghUEYDa2SSitBGlnOnMXO+EHdqpo1uekrQz1vv5FGev12bUJ52doLmdKPNMdf6qOjrxhDTKXfx+7KG7gd+BMwzDMEwXwg6cYRiGYboQduAMwzAM04WwA2cYhmGYLoQdOMMwDMN0IezAGYZhGKYLiSwja9ZsSynJnLuO4+Do0aMty4UQypy7KysrZB5WTdOUOXcHBwfJftq2TdaxLnWg+pnP58m8tZ2UDzTrjZeXl8lcs6ZpwnGcluVSSjz//POkFELTNDIXtxBCqYOu1WpksAQhBHkeQgicOXOGHMvZ2Vky53YndeDNY+p5HjmGnucpdd5R4gVQ6yEejytjK2QyGXKc5+fn8eabb5J1qLTHxWKxrcAYO6VZ953JZMgYFfV6XZlbXhX74dy5c+Q5uq6LmzdvknWo9oX+/n5lPyYnJ5XjSWmMDaNzSuLm80mlUuQ5+r6P6elpsk5VzADKBkBj71LtTb29vWTsgEQiQcY1ANRr1zAMct1RWn2y3qgHNjdQrVbJiWNZFrLZLFlnT0+P0jGqBp9yTkBjElFGNgxDWYfKgVcqFeTz+Zblqk11JzSPebFYJPtmWZZyDE+fPk3WMTc3Rwat2K5fzajmSxQ7DA0Nkf2cmpoiAz/sdpFsR/OmqNKS1mo1lEolsk7VBa2UkgxKFASBMohJf38/uaHncjllUIqxsTGyn7qu72uEseYAGargRpVKhbyoBxpBc6hzjMViZGCOfD6PK1eukG2o1uXAwAAGBwdblkcJqiOEIC9mqLKd0uygbNsm6y8UCuS+Caj3XpUDB9QXAYlEgtwbNE0j1x3Q2L+ofpqmSY4FNZco+BE6wzAMw3Qh7MAZhmEYpgthB84wDMMwXQg7cIZhGIbpQtiBMwzDMEwXwg6cYRiGYbqQyDKyZslDGIbKhPaqHKe2bSslUJRO0TCMSJ/fR9EKUkTRD+9FrtcoBEGglGep9KYqiYSmaeQ5RNG5W5al7Ccls4jShq7ryvmyV6jWg6ZpsG27ZbkQAvV6vS0piqZpSslMPp8n6ygWi0qJlWq+RJH2dJJmu2qaRo6jpmnkXBBCKNeNEEJ5nqq9yXVdcs74vo9isdiyPEq6zfXj9oNmH+F5HjmXVOUAyDUDNOYaJcdTyegAdVwDgB7DdRtQx1SrVXK+7LkOvFnDqhp40zRx5MgR8pizZ8+SEzAIAtJxmKaJY8eOkW2srKyQm5pt26QBpZR49dVXyTZSqRQZcGFkZIT8/U5oHq/V1VVyAvf29qKvr69luZQSMzMzZJu9vb3o7+9vWe77vlLPqQquEQSBUher0oRmMhlyXg4MDJD174TmeVkoFMj+JxIJjI2NtSyXUuL1118n2zx16hRpS13Xlbb86le/Ctd1W5ZHCXKichzlclnp/DpJ8z5jWRZ5UW4YhnLDHBgYUJ4jta8IIXDq1CmyjampKfJiaHl5GV/72tfIOo4ePdrWhYZqze2E5ouNpaUl0qlFmWvj4+Pk+a2urmJtba1luWmaGB0dJdt43/veR/qAQqFABmGSUmJycpJs4/XXX8fs7GzL8pMnT5K/bwU/QmcYhmGYLoQdOMMwDMN0IezAGYZhGKYLYQfOMAzDMF0IO3CGYRiG6ULYgTMMwzBMF8IOnGEYhmG6kMg68Pn5+U1/+75PavySySSpHQag1M6ptOTxeBzPPPMMWUcYhmQyds/zSF0sADz22GNkeS6XQ7lcblneSa1ls/5YpbXP5/N48cUXW5YLIfDhD3+Y1Fpev34dV69ebVmeSCTw7LPPkv3o6+sjg2csLi4qcxt/4xvfIMvDMCQ1ryrN6U5YXFzc9LfneeR6CMNQGQBFpRvWNA2rq6tkueocXdcl56OUUhn4YnZ2Vrn29zOwUXN/V1ZWyHPs6enB8PAwWefCwgJZXq/XletapT/O5/OkvWq1GgqFQsvydY03tXZv3LhB7n+qPu6EZDK56e8oMTgymQxZJ9V3QB3nwnEc5f5dLpfJdnK5HBlfQUqJb3zjG+SaGB0dJX2Vah9vRWQH3hwgQ7UZqSKEAVBGnlJFc7MsC4lEgmwjHo+T5bVajTwXKaUy6AMVSGC9jk7RHKBCFXVKSkleXAghYNs2GfgiCAJyozEMQxkxybZt0oHruk46X9V5rNdBOZ9O2mG7yIQUYRgqgx9ZlkXa0vM80mlEuUiIMgaq+RSlDWo+dTqQS3N9vu8rL1KofUVKqbRVEARtR2LTNE0ZfZCyVxRbuq5LOqdOXtQ2n4vqgjrKxaLqHHVdJ21p27bSB1SrVbIdz/PIOR+G4ZZAZ81omkb6qt1GYuNH6AzDMAzThbADZxiGYZguhB04wzAMw3Qh7MAZhmEYpgthB84wDMMwXQg7cIZhGIbpQiLLyJolQD09PaRUxPM8zM3NkXXG4/G2JCW+72NiYoI8plqtkhIAXdeVWsT5+XlSZtDf34/e3t6W5YODg2T9O6HZDo7jkHaIx+OkxEsIgYWFBaWcr6enh2xDpaWvVqtKyUg6nW5ZJqVU5lVX2ZqSse2U5r4uLy+T0qN4PK6UPKqkKIZhkLbWNE0pXapUKkrZpErfHEU+RZ2rStazU5rX5uLiojLnuUq2o5qrUXTgKglUuVwm+xmGITmOQggcP36cbMPzPHJe7la+tB3N88JxHHIcY7EY2b4QQrlmpZTKcVbtTX19fWQ/TdNUrpnx8XGyH/F4nJwvu40VEnlHax7oY8eOkY7h9u3beO2118g6BwYGSMcRhiE5KFECV6h00ul0GkNDQy3LpZRkInagMRZUMAEqWfxOcRxn09+ZTIac5NVqdUuAhWZu3rxJjlEqlSKdp2EYyoALKs2rlFJpB6ocAO7evYtcLteyvHns2qF5PObm5lCr1Voen0gkyDkipcT169fJNjOZDLmhq7SmQCNmAbVmhBDkRUKUDVN1rqoL5p3S3J9bt26hWCy2PH5oaEh545BOp8ljarUa6RjDMFQ6jpWVFbIO1TgKIfDkk0+S9nJdl3Q+ndybmu2QSqXIiz3HccgbA6ARHIayQy6XI+ezlFK5N12+fJlcNysrK+TeIaUkA+4AjYs1yg67deD8CJ1hGIZhuhB24AzDMAzThbADZxiGYZguhB04wzAMw3Qh7MAZhmEYpgthB84wDMMwXUhkGdl20pF20jOqUuV1ElVKPlXKu3brUOlm24XqmxBCKZmJosVXnb/KllHGOUodh4XdrAfVGKrWRLt2inJMlLnSbhudtmNzfSrpaCfOIco4qdZ9u/aM0sZhJuq+1O582eu9KYqP2Ct2nQ98dnaW1B/quo7nnnuuZbmUEnfu3CHbVOmrAfUkKBQKpHFmZmbwV3/1V2QdKo3e1atXSY3f6Ogo3v/+95N1RKX5fFdXV8nj+/v78TM/8zPkMQsLC2T5zMwMmdBe0zSsrKyQdSwvL5PjeOTIEVy8eLFluZQSk5OTZBu6ru9bAJErV65s+nt+fp6cA2tra5ienm5ZLoTAO9/5TnI+qzS96+1QlMtlUjc7ODiI06dPK+ugEEKQ87KTAXWArevzAx/4AHl8LpcjbQE0NNiULU6cOEEGb1pcXMTnPvc5so0XXniBDF5ULpfJuAZSSnzmM58h2xgaGiK13p2MjdA8HrZtk3vvwMAATp482bJcSokvfelLpGO0bZucT2EYol6vE70GXnzxRbK8VCpheXmZPGZxcZEsz2QypOZ9t3r8Xa+kKFcb7URZi1pHJ9po924h6hVaJ9jN+aoCc7Q7hodpjDoxH/aCKHcBnXpa0i5UG52YL3tNlHHsVDu7Kdt4zF6vvf20V3M7qvNTlbf7lCQqqjHq1JPBvbADvwNnGIZhmC6EHTjDMAzDdCHswBmGYRimC2EHzjAMwzBdCDtwhmEYhulC2IEzDMMwTBcSWUbWrKH1fZ/MZWuaJpkvHGjk3KU+wXddF0tLS2QdqkAGlUqFPKZQKJAaPiEELly4QLaRyWTINlR5rHdC85jHYjGlHr9UKrUsVwUxWG+DslMYhsqcu5lMhpRRZLNZUgsppcTc3BzZj5GREVJr2UnNa/N6cByHzHc+NDSEs2fPtiyXUuLu3btkm77vKwNOqGwZBAHZTwBKrblKx51IJMi13+l84M1rYnJykjwHTdOUMQFUY6CKL1GtVnH58mWyjmw2i1gs1rLctm1lfnfVnHYcB5ZltSw3TZP8/U5oHrPZ2VlSg53L5ch4AVJK6Lqu9BFUXAPLsshyACgWi2SMiijzRaU1TyQSpK1UvrIVkR148yQol8vkBDYMg1zoUkrl5PN9n1xIUkplkJVqtUpOgFKphGKxSNahSiqv0gGqktbvhOYxN02THGchBFzXbVkeJQKaaZrkRuP7vjKhfSqVIvupmuBhGKJQKJB9HRoaIjerTgYQaV5wVLtAw2lRQSvCMMSNGzfIOsIwVF5IqTYSKSV5wRclupcq0pnjOPsWUAfYuv6WlpZQq9VaHp9KpTA4OEjWGQQBeY7VapUs9zwPY2NjZBuJRIKcN5ZlkWMVZf9TobqY2wnN82ZtbY20Q6VSUQYFSiaT5Hyt1WrknA/DkLzRXO8ndUwikSDns5RSOadt2yYvlna7N/EjdIZhGIbpQtiBMwzDMEwXwg6cYRiGYboQduAMwzAM04WwA2cYhmGYLoQdOMMwDMN0IZG/XR8YGNj0dzKZJOUmtm0jmUySdao0iCrZDKDO1e26LllHKpUiJVJCCAwMDLQlI6PyBu+U4eHhTX+n02mlDlwlFanX6+Q5JJNJUqoRBIFSRtXT00P2o6+vj8yNLKXE+Pg42c/h4WEyf3wn9cfNdlBJHkdGRsj2pZQYHR1VplxVlUeRkVH09PQoJVbUfFuvg5IEUnbeDc170/j4OClfSiaTyjWpkvWkUinyHH3fV6aP7O/vJ/fAKLr+dmVg1HrZKc12GBsbIyWsKpkcAKUOvlqtknPesqwt/WpGCEH6kVgsppQCq87DcRzS1tlslvx9K4TsZLJqhmEYhmH2BX6EzjAMwzBdCDtwhmEYhulC2IEzDMMwTBfCDpxhGIZhuhB24AzDMAzThbADZxiGYZguhB04wzAMw3Qh7MAZhmEYpgthB84wDMMwXQg7cIZhGIbpQtiBMwzDMEwXEjmZyc///M9v+luVAKNWq6FUKpF1nj9/ngz4n8/nyaQEsVgMTz31FNnGtWvXyGD3vb29GBsba1kupcR3v/tdsg3f98mxGBkZwa//+q+TdUSluZ6ZmRn4vt/y+J6eHhw5cqRluZQS169fJ9s8fvw4+vv7W5bX63VMTU2Rddy5c4dM9qGyQxiG+NKXvkS2cerUKTIRx+DgID75yU+SdUTlE5/4xKa/5+fnSTv09fXhxIkTLcullFhbWyPbVKUtCMMQ1WqVPEaV3MdxHKRSqbb6qUpCNDQ0hE996lNkHTuheU3Mzs6StqhWqygUCmSdx44dI/emiYkJrK6utix3HAeXLl0i21D1c3l5GZOTk2QdFy9eJPt58uRJMplHX18fPvaxj5FtRKXZR+i6TvZNCEEmY5FSYnFxkWwzHo+TiZQsy8LJkyfJOpaWlshkJtVqFcVikeynat2l02kyadbQ0BA+/elPk3VsB9+BMwzDMEwXwg6cYRiGYboQduAMwzAM04VEfgdu2/amv2/evEm+nx4dHcUzzzxD1vniiy+S78pGR0eVydhv3rxJlgdBAE1rfZ1SLpeV75kSiQTZT03TyHc9mUyGrH8nLCwsbPr71q1bpB2y2SzK5TJZZz6fJ8/PsizlOyDqHRLQeKdIYVkWmfBeSokf+qEfIuuwbZusI5vNkr/fCS+99NKmvy9cuADHcVoe7/s+rly50rJcCIFHHnmEbPOVV14h56pt27hw4QJZh+q9pOu65LcrQghcvHiRbOP69evku0vDiLztROKNN97Y9Pf4+Dj5XrRarZLvIwEo35EnEokte+JGpJS4desWWYfKFplMBslkkqyD+jYFAFZWVkhbUN/H7JTm+TsxMUF+f0SdO9DYVy9fvkweV6/Xyb1H13XlfIsyH13XJctHRkaU/aTGgvoWgiLySmp2gp7nkR2SUiKRSLQsD8MQtVqNdBxSSnJwpZRkH9ahBlZKSX5cBWw99+3KqWNUv98JzYZ2XZccA1V5FOfreR45RqqPqwDANE1yHFSbGQDlZqayA/XBzE6pVCpb2qbmqud5yk1ANQb1ep10rkEQKG0hhFCuB9VmYlmW0lZUHbvdrFrRPK66rpMXcp7nKS8WqYtioGFvqo4gCJQXzo7jkOMYxfmo5kytViPnXZT9MyrNF7CquaYqBxofKrcz11Q3Vxv70goppXJdmaZJ1uF5ntLX7QZ+hM4wDMMwXQg7cIZhGIbpQtiBMwzDMEwXwg6cYRiGYboQduAMwzAM04WwA2cYhmGYLiSyjKxZN3nmzBny833HcTA7O9uyfF1mRn0+r+s6wjBsWR4EgTIus0ry4jgOKXeTUirlICr5TjweJ3+/E5rlXNlslpSBpVIpZWzrXC6nbJOSokgpSTsBaq1lGIZKOZ8q/n4ikSD1v7uVamzHdvEJqPXguq5SW7yyskKWa5qGnp6eluXxeLxtrbvruqRueF36Q62pVCqFvr6+luWd1OMD2CLnWl5eVkoGVXOBGmegoROn4l9LKcm5CDTGieqn7/tKmZdKAqrrOqlXV/VxJzTr3mu1GrkvRJGRLS4uKucaJS+NmmNAJSNrVwIaRcq2GyI78OZFd/bsWVIHeevWLXzzm99U1kmdmGVZ5ASo1+tbAps009PTQy4SwzBIByulVG6spVKJXGjUOO2U5ok0OjpKnp/jOORmJKXE7du3yTZVgfqjaMljsRg5Dq7rKjdEVT90Xe/ohkSxXYIEyoGXSiVlYobJyUlyPRiGgZGRkZbl8Xgco6OjZBuq2AvLy8tkAJL1BBSU3n5gYIDcVCnnvhua1+/ExARpi1QqhaGhIbLO4eFh0haVSoXc1IUQ5I0B0Bgnak1UKhXk8/mW5VHiYFiWRQat6eTNxcsvv7zp7+PHjysvqFXaaNXe+/jjj5PBvqrVqjJQl+pCKoqmX6UD72QskE317kmtDMMwDMPsKezAGYZhGKYLYQfOMAzDMF0IO3CGYRiG6ULYgTMMwzBMF8IOnGEYhmG6kMgysu0+91dpKanyKBrAKG1E+X07/YzS/l5p/DpB1L51UiO92zai9KETdXSCnbajssN6uWrN7DWqdbkuh1Gtmf2yg6ovu/mNECLSPOuErfYixeTDQlT/0AkfQcmVo6ToPShbRXbgzUEoNE0jtXP5fF6pUVQFSxBCKPNQU7pYAJienibrKBQKZIANIQSOHz9OtuE4Dpk/OJPJkL9vh3Q6TWpJo+TZvnbtGjkBL126hP7+/pblmqYp9aSqvOSe55Fa8ihac9d1ybFQ5XjeCc1zStW3dDqNp59+umW5lBJLS0tkHSMjI+RcsiwL6XT6/t+alDC9AFoYwvQDSCHgpmz4moBnbL92+/v78dxzz5H9pHTiQGPtN+dL30gnc1ADwOrq6qa/VXNR0zRl8KLl5WWyfH5+ntw3bNvGuXPnyDpWVlaUAUQohBBkkCagscdSOnBqXe+U5vmtCryUSqVIDTegtkOxWMQbb7zRstyyLDz++ONkHV/4whe2zKGNDA8P4/z582QdqvkkhCA18buNFRLZge8V7d5VtPv7qHd0Ue+gDgJVBKB2NokobUQpZzoz16PawfR89K8WMDa7CggB1zKgByFMz0fFNnHnaB8KCQdymwAT7c6Xw/CUpJ2x3s+nOHu9NqM+6ewEze1Emauq81fV0YknpFHu4vdjD90NB+7AGYbpLFbdw+jcKmI1D7PDWaz2JFGJWdDCEKm1MnrXyhibz2GpN4XFbApS44svhulG2IEzzEOEFoQ4spCDHoS4dWwQNdsE7l35h7qO1Z4EcqkY+tYqOLKUR80ysJaM3T+GYZjugb9CZ5iHiHSpioGVIlayqU3OeyNS05BLxVA3DYwu5mH6dCIahmEOJ3wHzjAPEVbdR802UY7bD5y3lNBDCSEltCDA+hu7QsLB0EoRdt2DvHeoAO7/LtAEpBB8d74BAwImgISmI9Bbb5+2bsABPW6W0CCIY6TQEMrGxZUHCfozybcQEtDRcF5SAtQbaCuUMH165BK6gbrR+iOyhKbfs6WEBxwqO7ADZ5iHBCEBww9QTMTgWg+Wtl33cOLuElKl2qbNTg9DOK6HSzdnETR9zBYKgbmBNGaGOpv6s5sxIPAuO4ULRgzB0SyZ8UzTNMQtxdfwQoPCx0PqEgEkvhe6eEXSKS3fKugAnnSBE76AgEZ6cE0L4dyiM1aOHXsE3pHWSiXbthFHAiEkvoc6vg067fF+wg6cYR4mBCCadjQ9kLDqPvQwRLjBYYh7d+VCSmgy3PTvJoCh1SI78A2YAC4YMaQ0HVURQBApIjVNg6Hwzo1fU8dIaBDQoOGCsNmB38NAw3nHAASKO3AhBbSQ/krcFBppS1MI2EDDDrC604E3a7allOQVqOM4OHr0aMtyIYQy5+7KygqZh1XTNGXO3cHBQbKftm2TdaxLHah+5vN5Mm9tJ+UDzXrj5eVlMtesaZpwHKdluZQSzz//PCmF0DSNzMUthFDqoGu1GhksQQhBnocQAmfOnCHHcnZ2lsy53UkdePOYep5HjqHneUqdd5R4AWRe4ngc9lAamdkl9NsxeDEbAKClJFbTaZQ8H9mezP34DfrdOYj5ZeDRM0CscT5zc3OovX4Voz7gBj7uTk9vaUalPS4Wi20FxtgpzbrvTCZDxqio1+vK3PLbxX6wQ4l0rg5TAiujQygk7Ja/r1Zr+N73XiPbkIrnv/0D/Xi2bxgDpRrSmsCpwa2a6cnJSeV4Uhpjw+jcPVzz2kylUuR69X0f09vMr41sFzPAgUAYxlEH8FpQw6xs7VCF0GBJWmO94hXgE/uXAQOPmwJndBuQISr1rTEOVGvXMAxyf6O0+mS9UQ9sbqBarZITx7IsZLP01XtPT4/SMaocA+WcgMYkojYTwzCUdagceKVSQT6fb1muClizE5rHvFgskn2zLEs5hqdPnybrmJubI4NWbNevZlTzJYodhoaGyH5OTU2RgR92u0i2o3lTVGlJa7UaSqUSWafqglZKSQYl8oMAWk8KiYk5ZHyJcuzB1+VBMoEqgOyRI9BME3DrMGcWIEcGgPOnoDkNZ1SUday95mM4EKiHPnKr9S03iWNjY2Q/dV3f1whjzQEyVMGNKpUKeVEPNILmNJ+j6QewKwvQQolgsA9ef+t1Xcrn8b2X6TWjWpfHBzJ4WzIOoxbAMnQMDgxs+h4hSlAdIQR5MUOV7ZRmB2XbNll/oVAg902g1d4rEBoOAghMB3VcD+knE4ak3VxB1hGitY/QwgBH/TpOCguhDLddg4ZhkGvCNE1yLKggLxT8FTrDPERUsmmsHR1Aem4Juus1vvJpJgihL+cgKjV4F88A9u42D4ZhDhZ24AzzMKFrWD05ChGGOPLaNaRml2CWq9A8H7pbh76cg/XtK7D/5nvwzhxDMNzPX5kzTJfCH7ExzENGaJmYu3QGfbdmMHT1DmqpONx0ApofIFF2oZkmak8/hmB0mJ03w3Qx7MAZ5iFEGgaWz46jNJiF4dahez6k0OA8MgLRn4VM0R9/Mgxz+GEHzjAPK0Kglk1v+qfkkaFdZz5iGOZwwe/AGYZhGKYLiXwH3ix5CMNQmdBedaVv27ZSAkXpFA3DiPT5PSUjA7bKH5qJoh/ei1yvUQiCQCnPUunQKWkS0Bgf6hyi6Nwty1L2k5JZRGlD13XlfNkrVOtB0zTYdmvdsBAC9Xq9LSmKpmnKXNv5fJ6sY13DLaUGCbmtPlY1X1TrrdM021XTNHIcNU0j54IQYtt1Y0BAQKCRvE0qz1O1N7muS84Z3/dRq7kIwgB+IFFskiFGSbe5ftx+0OwjPM8j5XqqcgDbrhltg64xCAJ4PqUDp2V0gDquAQBYmgVNaNABxJwY5AYB/7oNqHGuVqvkfNlzHXizhlU18KZp4siRI+QxZ8+eJSdgEASk4zBNE8eOHSPbWFlZITc127ZJA0op8eqrr5JtpFKpbQM/rDMyMkL+fic0j9fq6iqpJ+3t7UVfX1/LciklZmZmyDZ7e3vR39/fstz3faWeUxVcIwgCpS5WpcfPZDLkvBwY2BoIY7c0z8tCoUD2P5FIYGxsrGW5lBKvv/462eapU6dIW+q6rrTlV7/6Vbhua91svV7HEy4QaA48GSJXXdu2r5QdyuWyMkd1J2neZyzLIi/KDcNQbpgDAwP3+6kFARJrZThuDUnZCFnbky/DlRJrydY51U+dOkW2MTU1RV4MLS8v43rZx0Asg2Lo42s3v7vlmKNHjyrHmnIcqjW3E4rF4qa/l5aWSKcWJaDO+Pj4lvOzJaBJDRqAtbU1zBVaB9EyTROjo6NkG+973/tIH1AoFNB7fRKJtSp0XcOl0fEtevzJyUmyjddffx2zs7Mty0+ePEn+vhX8DpxhGKYFplvHwPQiehdy8GwTIgwhpER/voS+fAnzvUncHcrCNzoXEIVhosLvwBmGYbbBqHsYmlyAWfcxdXYMU2fH4cYc1B0bd0d6cW18AJYX4OzUIvQO3skyTFTYgTMMwzQhQonsYg521cXc8REU+npQdxrvtCUA1zSwmo7j9tE+GEGIsYX89lHvGGYPYQfOMAzThOF5GJ6cRzGbajju7d4zCwHP0DHfm8aR5QJSFc4Wxuwv/A6cOVikhAbADCU0v/GxjScE6lojJzXDHARaEEIKgUJvmo5WJwSKCRtlx4Jd91Hk+DjMPsIOnDk4pEQiCHG25CHmBXAhISEQkxIFXeCmo2PNEBzuk9l3jLoPzzZRS2zOkCekvOfcH/ybZ+goxW0Ywf7K5xiGHThzYAy6AS4XXAQQmLJ1LN/7kLfXCzHuBuj1QryeMDBvaezEmX0l1AS0IIQWhAjvfWEuNYH8QA+smoe680CfLKSEfu+OnWH2k8gOfH5+ftPfvu+TGr9kMklqhwEotXMqLXk8HsczzzxD1hGGISqVrQnY1/E8j9TFAsBjjz1GludyOZTL5ZblndRaNuuPVVr7fD6PF198sWW5EAIf/vCHSS3p9evXcfXq1ZbliUQCzz77LNmPvr6+TcEznFIFp65OYvHRsyimYlhdXsKt27cBALcl8H1NYjwQOFXwsWKFWNWAb3zjG2QbYRiSmleV5nQnLC4ubvrb8zxyPYTh9nmEN6LSDWuahtXVVbJcdY6u65LzUUoJTdMbIUtaBMGYnZ1Vrv39DGzU3MeVlRXyHHt6ejA8PEzWubCWx/FyBfWJu1hOPrgLX4jpELZAtVaBX2m00eN6SKwVMWFI5PMP2lXpj/P5PGmvWq0Gz/MQ2iGCIEClXMbGUV/XeFNr98aNG+T+p+rjTkgmk5v+jhKDI5PJkHVu1/dQaAhMGwICAwP9OEnkZXccR7l/l8tlcoxyuRzkygpGAx0VSExOrm2yg5QS3/jGN8g1MTo6Svoq1T7eisgOvDlAhmozUkUIA6CMPKWK5mZZFhIJ+qVTPB4ny9cXSSuklMqgD2trW4NdNNfRKZoDVKiiTkkpyYsLIQRs2yYDXwRBQG40hmGQUcaAxmJdd+C65+PE3SXUBrIoDmQgNQ3CMDY53zKAG7rE0QC47AJ/aQXkeQCNjZwKFtNJO2wXmZAiDENl8CPLskhbep5HOqYoFwmRx0AAkFuDrkgpI7VBzadOB3Jprs/3feVFCrWvSClRq7komTp6cyUsWDpkI/wafADQBHzPbzhPKTFUqMGXwJqhbZoHqkhsmqYpow9KGULei/sVNtkuii1d1yWdUycvapvPRXVBLaVURknb7hxD+aBOwzBgE3PNtm2lD6hWq+RYrq87KTWECFGve9gQDA5hGG4JdNaMpmmkr9ptJDb+Cp3Zd+KFMuyKi9xQb+vHjgLwBTChS4wHGmzJjyeZ/cPTNVwbSCHm+Rgs1Rrvt5s2eSMI0Vf1MFSt42pvHFWTg7kw+0v3vwMPQ2hrRWhrJYha41G4dGyEPUmEPeoYt8z+Y3g+ipkk3Ji96d12JgSyocCqJrEmAAhgWpOoCQmLJbbMfiIE8jEbcz0hxnNlDJVqWE7YWHMad+7Jcg0DpRqcIMSdtIPFuDonA3N4Mf0APTUPVhBCDyV8TcA19Pv2Pqx0twP3A5g3JmG/fg3SMuEPNeJEG9dXIOoe3EvnAI6QdOjQghCebSFsenx2ItBw0ReY0SS+ZjUek9VE4z924Mx+IzWBuXQMNUPH8dUSxvJlDOsaIAHDD1DVgKvZOFYdkz+y7GJSrodzc3kkPB+rcRs1Q0ei7iNbcbGUdPA3+uH1Id3rwKXEyPwK7JUa3POnEBwZRJhovEfwylXoswuwv38Dw0EVt7MJXmCHiMDQ4VRq0H3//he+AKADsCAwsMFZJyWQkECVzcccAFIIrMYtFO1MI1ZB2Jic9TBADRKexjLHrkVKZEpVnJ5ZxbJj4eZAChXTQKAJGGGIpOtjNF/BE2sVFA/pK7zudOBSor/k4lzZR/mD74BM3nPQ9xZSaJkIMyn4YyM495//HGumjpWkwwvtkOBbBhJrJTjlGjz7QZSrENj0dSckcCwQ0KWAz6ZjDop7Edc2frrnefufNpXpIFLCcX2cnlnBtb4klpIxhAL39yJf11AzdORjFo4bAqfnAwQ4fI8Bu/IjNjMI8dT0CmaySch0CtCadMJCAJoG2ZPCbF8ST04tc5CFQ0QllUChL42+uWXofuvHUzEJjIUC14wQNXH4Fg/DMN2JkMDo8hoCTUM+7iDc7kmKEPB1DTOWBg8SNg7fXUTkO/CNGl6goaWkpCKe52Fubo6sMx6P70pSMuA2ckdf0XxUJibIY2cdHYNhiFi5itX4VqmTrutKLeL8/DwpM+jv70dvb2/L8sHBQbL+ndBsB8dxSDvE43FS4iWEwMLCglLO19PTWmsZj8eVWvpqtbpJMnJ7KIPjt2YxMDmH1Z4k9CBEzHGgVwNYAjhuGjjqhhBxG3djGuJCKvOqV6tVUuLUPHbtkE6nN/29vLxMysTi8bhS8qiSohiGQdpa0zSldKlSqShlk55wEBomQhmi5la33Heo7jxV8k6VrGenNK/NxcVFZc5zlWxHJW+q1+vK+A4qmVe5XCb7GYYhLNuGrukwILeMqRACx48fJ9vwPI+cl7uVL21H87xwHIccx1gsRrYvhNh2zcYgYPgGDACa0NB4dtea7cbY9gNYVRezCQvpoQFoxN5QMi2UVkrIVH04hoHBvs1rX0qJ8fFx0t7xeJycL7uNFRJ5R2se6GPHjpGO4fbt23jttdfIOgcGBkjHEYbhtoPS70rc1SUmVpex+uqr9/896zYmas5+cFqapmHeMWCUa6hpW+tKp9MYGhpq2QcpJZmIHWiMRV9fX8tyKln8TnGczaEdM5kM6Ziq1eqWAAvN3Lx5k7RDKpW67zxtCZz1BZIb3glpmobk9PKW34UCCO7Va6+UoWmb26iFIYbuzKJH09AXM2FLAykhkZbAD/sGlnpsTGaTiNsGYgBpJwC4e/cucrlcy/LmsWuH5ouJubk51Gq1lscnEglyjkgpcf36dbLNTCZzfxO3JXDGF0hu2Lu0QEPS3WrHEIB/zw59NYl6vfWGJ4TAsCkQigB+GKDYdFEhpVQ6JtW5qi6Yd0pzf27duoVisdjy+KGhIeWNQzqdJo+p1Wr3HaMjgfPSQGrDA00p5baOMwTg3bskStQFQknEubBtnEtmYAkDptS2jKkQAk8++SR5Uee6LnnB1sm9qdkOqVSKvNhzHIe8MQAawWGa7WCFEvZyFWYInA0l+sLG+Qs0vqHZiF7XkJ3aHHQJaOReSFTq8D0PJ9dcWOtfyYYhRBhC6jqkZUICqMCEYTkw6zUkEg5Onzq26W5dSolCoUCeR7lcJu2w5w78MCGBbd9G1LTtF5w8hI8+upWEBJ7wNZgbxzQAzNLWiEsb7aTp/rZWCDQByw8wVPAbgUMgUTN0XB1KYzVuw9PvvR7hVI2bSITAE3Wx2Q4AzOLWuw2JB/cow2YSoU7fsTSeJgp4kl87RSEBDU9Ja5PzkFJCalu3Vwl53xZuspe8GNI1HZYwIAQQ8PzfRADAAnAi1DG+4WZCb1oPIhCwilsvrLVQwgnCxn93ph88KQhDIJSArkHqjQsDy/Mhqy4gGiF2DxNd6cCLOnC8AsTCzZO6amy9EnX8EEerdVzPdO7u662MByAngN4NG4oQAtvN6xBofKULQDc0iBYXUjUDACQgJUIhMNGbwEIqxh8dEngCyGlAbxjFDgL1ewUlGSAI6av9def9Uo2+q2AaeJBYRYj+pk+KmiOnAUAAidq9y9pa4G97zDoGJEypIZAS3wpbP915q+EL4FbcwLFqgCCUCO5936RBoPmBvACgb7MmpCaAANDlPWe9/hRjw9OM8F68+0CrwyiW4ZoGpgcynT+hNuhKB57XgLoAHnGB61I+2OibN3wpcbrowhMCBY6S1BGKAvjv9mYHYJgGBgcUj8JSKejEo75QSoQbHyOx8yYpCuBPnc13yIahY0CxwfzNzJuoEOEzhRDkI1lmKwVI/Bdts4MNZQjXo78LmV2cgUe8n1a9inirEgJ4M2HizYSJXL6GaqX1xY1lWzh+bGuccU0CF1fKyNQ9mO9/Hnq69asEf3Ia8b98BbdHB7CcTR6qvakrV6qrCXwjJjDuSWTrAcQ2V7FCNsqOlOt4pT/ReBTLtI/Y5j/ggYyv1X+qY5rLGZpt7CABpR34Qewe0GpNKGBb7JIt+wbxX/Px9/4LNYHJHqeRvnh+GfCDbfckUXORXFhFLh1H/hA+FezKO3AAmLCABCQeXashb+qYiZnIWY277N56gCNVDxkvwJ2Ug9kEhzlkGIZhHlA0dVzLxvHsnWmkF1fhjg6jPtwP6djQShXYswuw5pZQqNcxM5iFbxy+p7hd68AlgGuOBmFYOFtwcbxcR/neO/CEH6JsaLiesrGQpLNkMQzDMG9BhMByzMTaxbMYuDOD1N+8DulYCG0LWqUKEYSonhpDrj+Nysryobv7BnbgwLf7WrKd9IyNVHltPEQSAoEQmI6bWHQM6KGEc++Dnpom4GsCniYeSBBatCWlVKa82/i/u6ljryM2UX0TYsMYEMe000YUaVGUcY5Sx2FhN+tBNYaqNdGunaIcE2WutNtGp+3YXJ8qxW4nziHKOKnWfbv2jNLGYSbqvtTufFH93utJYu3px6Bd9iA8D8LzIW0L0jAQ2ib8tTXIFZA+pBP93A27zgc+OztLfuyi6zqee+65luVSSty5c4dsU6WvBhpG9tHI07suZLovXQJQKBTIST4zM4O/+qu/IttQafSuXr1KavxGR0fx/ve/n6wjKs2TfnV1lTy+v78fP/MzP0Mes7CwQJbPzMxgZmamZbmmaVhZWSHrWF5eJsfxyJEjuHjxYstyKSUmJyfJNnRd37cAIleuXNn09/z8PDkH1tbWMD093bJcCIF3vvOd5Kam0vSut0NRLpfJHNCDg4M4ffq0sg4KIQQ5LzsZUAfYuj4/8IEPkMfncjnSFkDjAzLKFidOnCCDNy0uLuJzn/sc2cYLL7ywJSDQRsrlMhnXQEqJz3zmM2QbQ0NDpNa7k7ERmsfDtm1y7x0YGMDJkydblksp8aUvfYl0jLZtk/MpDEPU61slrht58cUXyfJSqYTl5a1xLjayuLhVa76RTCZDat53q8ff9UqKcrWxmyhrO61ju/Kdttru3YLq7rGTV2a7GVPqQmv9zq8dDtMYdWLO7QVR7s479bSkXag2OjFf9poo49ipdnZTtvGYvV57+2mv5nZU56cqb/cpSVRUY9SpJ4N7YQf+NJthGIZhuhB24AzDMAzThbADZxiGYZguhB04wzAMw3Qh7MAZhmEYpgthB84wDMMwXUhkGVmzhtb3fTJRvGmaZL5woJFzl/oE33VdLC0tkXWoAhlUKhXymEKhQGr4hBC4cOEC2UYmkyHbUOWx3gnNYx6LxZR6/FJTTueNqAKsrLdB2SkMQ1QqFbKOTCZDyiiy2SyphZRSYm5ujuzHyMgIqbXspOa1eT04jvMgJeE2DA0N4ezZsy3LpZS4e/cu2abv+8pgOCpbBkFA9hOAUmuu0nEnEgly7Xc6H3jzmpicnCTPQdM0ZUwA1Rio4ktUq1VcvnyZrCObzSIWa86f9QDbtsm4BoB6TjuOA8tqHUraNFvnI98pzWM2OztLarBzuRwZL0BKCV3XlT6CimtgWRZZDgDFYpGMURFlvqi05olEgrSVyle2IrIDb54E5XKZnMCGYZALXUqpnHy+75MLSUqpDLJSrVbJCVAqlVAsFsk6tksq39wPClXS+p3QPOamaZLjLISA67bOihQlApppmuRG4/u+MqF9KpUi+6ma4GEYolAokH0dGhoiN6tOBhBpXnBUu0DDaVFBK8IwxI0bN8g6wjBUXkipNhIpJXnBFyW6lyrSmeM4+xZQB9i6/paWllCrtc5QlUqlMDg4SNYZBAF5jtVqlSz3PA9jY2NkG4lEgpw3lmWRYxVl/1OhupjbCc3zZm1tjbRDpVJRBgVKJpPkfK3VauScD8OQvNFc7yd1TCKRIOezlFI5p23bJi+Wdrs38SN0hmEYhulC2IEzDMMwTBfCDpxhGIZhuhB24AzDMAzThbADZxiGYZguhB04wzAMw3Qhkb9dHxgY2PR3Mpkk5Sa2bSOZTJJ1qjSIKtkMoM7V7bouWUcqlSIlUkIIDAwMtCUjo/IG75Th4eFNf6fTaaUOXCUVqdfr5Dkkk0lSqhEEgVJG1dPTQ/ajr6+PzI0spcT4+DjZz+HhYTJ/fCf1x812UEkeR0ZGyPallBgdHVWmXFWVR5GRUfT09CglVtR8W6+DkgRSdt4NzXvT+Pg4KV9KJpPKNamS9aRSKfIcfd9Xpo/s7+8n98Aouv52ZWDUetkpzXYYGxsjJawqmRwApQ6+Wq2Sc96yrC39akYIQfqRWCymlAKrzsNxHNLW2WyW/H0rhOxksmqGYRiGYfYFfoTOMAzDMF0IO3CGYRiG6ULYgTMMwzBMF8IOnGEYhmG6EHbgDMMwDNOFsANnGIZhmC6EHTjDMAzDdCHswBmGYRimC2EHzjAMwzBdCDtwhmEYhulC2IEzDMMwTBcSOZlJc9B7VZB9AG0lAFn/vaqOdvsRpTwKVB1nzpzByy+/3HYbwM7tIIQgk09IKeH7flt1RKWdce5EyP4zZ87gm9/8Ztv1AFuTD0TpX5QxpMYoylxXJbdQ1dGJcVat206uB6CRFGQjh+EcO7U3tct+2mI3PqLdvUnX9bb3pqi+qN3fq+zw0ksvKetphu/AGYZhGKYLYQfOMAzDMF1I5EfozbT72CHKo6Moj6FU/djrR1TrfVDlae5kWzupO8oYqnIfB0FAPsoSQkTKS9yOrYQQkc51v9jNvIpy/tQxqseFQgjYtk22QeU9BhqPPVXHUHnP19mv9bBdfUEQkG104pVQlMfDKvbj1Z5q/XfiPHZLlEfkVA5toGFraj5GsfV++IgwDMk5qVpzrdi1A+8U7bxb6NRG0O77DRXdkHL9oN9Pq/pw2Oyw0/GK0raUUnkRoypXXUhFWVMqO+zHhtcO+/FOs1PstRPvhr2HQjXnVY4v6vnv9ZzeKzvwI3SGYRiG6ULYgTMMwzBMF8IOnGEYhmG6EHbgDMMwDNOFsANnGIZhmC6EHTjDMAzDdCGRZWTNn8F3QperqkOlr+4Uex3y8KBlN6ox7EToyU7MhU70o93YA7ulE2Ok0qs6jgPLslqWG4aBwcFBso56vU72o1qtIp/Pk3WotLsHLV2KYmfVXOsWnXe3E2XfaDfWR5Rw03ttK03TlLEJdsOudeBRYgF3QnMapR/t0Ik2DnIhqjb9KE4xShAB1Tm2e5HQieAaqjo6Ec99t0S9GKXGOZVKIZ1Otyy3bRuPPPIIWX+1WiX7sbi4SDpoKSUqlQrZxn4GNtoOlZ2j9E91kQKo14RKkx/FcUSh3dgBB0WUeRAEQVvn14n9LcrepGlaW34iSiCsbdvd1a8YhmEYhjlQ2IEzDMMwTBfCDpxhGIZhuhB24AzDMAzThbADZxiGYZgu5MCzkTEM03kMCRyXOnqhISkF6kJiDRJXIVBAd2eoYhimwZ468HZ1r/tFu3r1g6Tdfu3nual0kN10LtvRif63w7qUJSWB90gbZ2HipghwR/ORgY53SQM/aJr4I28Nd7G9tCaqFOYwrYfD1Jf9hrXknaETe9Ne72/bsWcOvBP6ak3T9jzhfBiGSp2gruttaxE7RXM7+7F5qTSOUXSSlUpFaUtVHYfJDjutO0rbKi3o6dOncfLkyZblvdksPvKTfxvWX78KOA78px7F2UwaZwAIAMKto/Dyq/iFN+9g6cw41kYHIZvG63vf+x5qtRp5HgsLC2Q/D7tDVW2mmqaRAXPWaTcwRyd04mEYKs9FVX5QqOwQZV9R7QlSSuW+o9L8B0EQKbYAhaZpZB279XMH/ghdNfh8hbmZ3YxHJ5zaYQ5mcxDs1fnu+gJFAprnw3r9GtCXgffMZcA0G457/ZiYg9UTR5FPxjD63WvwEw4qfRlgQ52dCi5ykEQJOhTlHNvdm9ptI0ofohxzmO25H2O0H+d/UON8OJ5hMwzTNvFiFfrdWQTHRwHTbHlcPRlHJZtG78QcNF8dpYphmMPJgd+BMwzTPgJArxQIsz0IB/oa/ygl4AfQZhcA00DY39v4Z11DcbgP2YlZ6HUPocnbQKcRAHpMC2nLRtIwkTAsJEwTtq5jvlLG9cIq1MFamXbRhECPad+3Q9I0kTAt2LqOuUoJ19dW4XneQXdz1/DKZZiHhIxhITwyBGxwyNr8Euyv/DWkYUD2pJCJGcgPZFGPO/BiNnTPR/duX4eTE8ke/O3jZ3G+pxe6pkEXApoQ0IUGAcALQyzVKvhubgl/vTSL6UrpoLv8UHI8kcZPjp3CqWQahtDu2eCeHUTDDovVCl5ZnMFfzt7F3XLhoLu8Y9iBM8xDQh0B4NY3/VsYcxDGYxDVGsTKKnrdOjK3puFbJgLTACQad+qH+D1pN2BpOkYTSfz42Gn8wPBRWJreeAAiGx+ZhXjwzjpmGDiVzuJ0TxY/NX4GN4s5/On0bXxndRHeHn+0+7BjaRqOxJN44ehJvGNgBJamwQ/DlnY43ZPFqXQGHzx5Htfzq/hvE9fxraU51LvEDuzAGeYhQALIex70qVn4ly8AiVjDKfdl4P74u6HfnYM+PQ9vbgFaxYVTKEGEEuX+DLy4jcAy2YnvAgHgWLIH7z1yDO8aGUOPZaMeBJgqFfDm2iruFNdQ9OooeXUU/TpkKHEu04u39Q/jaCKNXtvBxUwfRuMp/Pe7t/CV+SkUvLqyXWYzAsB4Io13D4/i+cHRB3YoF3FlZQl3SvkHdvA8hDLE+Uwf3tY/jJFYAr22g0d7BzGWTOO/3rmGL0/fwVrdPejTUsIOnGEeEnKWDoQh9Ok5BGdPNByyEEAygeDCaQRnTmDxjTdhzS1i+Pu3Gu/NJ+eg133kjo/AS8QO+hS6ClvT8ezgEfz4+Cmc6emFFwS4ll/Bd1cW8dLSLCZL24fMmSgX8JXZKZzMZHEx04e39Q7hkUw/Pnj8LAacGP779G3MVcv7fj7diq3peKZ/GB84egKn0xl4QYhrhVV8L7eElxfncH11aXs7lAr4yswkjsQSeCTbh2cGjuCxvkH8b6cfwVAsgf965zpmKsV9P5+dsGsHHiX4ieqzesNQNx8lj3Q7RNGaR2ljv7SvzZrFKBIL1TEqHXyUnMIqTWs2m21bdlOv03cmQRDsmx2a50wn5ohqPfi+j2q12rI8nzKwfKQfiTdvomAI1JPxLXfVs7UyhitVrDominEL/UUXyclZiEIJ0yNZ5HK5tvN9HzQqWziOg3g8Tv5+dHSUrCfmBXibmcBz/cPIWA5ygYdXvRJeDyqYMkOIgT7EJb2uJspFTJSLeG11GT8xdhI/MHgU7ztyDMOxBP7z1A28kV9Rrs0o8Qf2Kzd7c187EQtExaAdwwfGTuL5oVFkTBsL1TL+YnYK315dxN1yAX4YknEHPRliolzAZLmA1/Mr+Kn6Wfzg8CjeP34aRxJp/P/uXMWV3HJHJIOUn9mtHfb0Dlwlflc5hjAMIwV7aYdOaEL3k+ZJoApkANBjJKVs206qNgAgkUjsOmk9EG2Cu66r3PD2iijaYxWqgDlhGJJBJ9wgwPJQL4JKFT1feRnl/gxWTx6Fl4xD83ykZ5Zw8dodVDTg9dFeuIaOslHE6EoJznIOg+UytMIa+VVuVOd9kOtFNV9N00QqlSLrGB8fb1nHoNDx9tDCqDBgQGAmcPEXbh7TgQsv4aA/4cB1XaytrZFt+L4PKSVmqiX8xztvIl938YGjJ3ApOwDHMPAb117DRJGuQ0UnAgx1qq3taGeenEz24O+euoDHsgMwNA03ijn83q2ruF7IoR4+2Aei3uTMVMv4/dtvIF938ePjp/B4/xDipon/75vfwWS59Z14J8bwUDpwhmH2l9A0sHJ6HF48hsErt9A7MQs3EYNR96D5Ae70xHBtOINAb1xwTQz0oGqaODuXQ6pWx1ihClMIeIf4DvsgcSDwmHBwQjfhhyG+4xbx5fIqqrK9j57WvDr+aPI6FqoV/MKZR3EqmcEPjxzDvy9/v2s+qNpPkoaJHz5yHE/0DSGQIb4+fxf//uYVlHyvrUj/+bqL/3TnKuarZXzs3GWcSmfwI0dP4N/fPJx2YAfOMA8TQgC6QGFsCIXRQWi+D6NWR2jo8G0LU9PTCDbcYUshMJ9NIOnWMbpSwmkzhrPxNN4or3HKk20YFDrOaBbCIMTXKqv4q+oaOrWtB1Lify3cxbFkCi8cPYkn+gbx6soAvrOywLbYgABwMpXBOwaPIAgl/nDiOv7b3ZsIOnTRGUiJr8xN4niyBz8+fgpP9g3h2ysL+PYhtANHYmOYhxUhEJom6qkE/JgDEK857valUXZM9Js2nk73IaXztX0zFoDHNAdxoWHKr+F7bqljznsj//3ubUyVixh2EnjXUOOLauYBMd3Ajxw9jh7LxrXCKv5yYbpjznsjfzJ1A1PlAobjCbxreOxQ2oEdOMO81RECrqljqi+FQNfwSKIH5xM9vDk0cVSYOCksVGWI77llFMO9+d4iV6/hT+7eRDXw8bb+ITzeOwjtkHyDcxi4kOnDU33DKHp1fG3+LnL11sl32iHn1vBfJq6j6vt4W/8wnugdOnR24DXKMAwAYDUZw3Iqhphu4PnMICxt9x8dPoycFhZMITAnfUyF9T25+waAEMD3ckv41soC4oaJHz16HA7b4j7PDozA1nVcL6zi+7nlPbn7BoAQEt9dXcTfLM0hYZh4/+jhswM7cIZhACHgGRqmHR0F38MxJ4ERi3XhGxnTTIRSYkUGKO6Z+26w5tXx8tIs8q6LM+ksxhP0V/NvFQSAR7MDCGSIu6UilmutZZWdYK3u4qXFGeTrNZxJ92I8ebjssGcvulS5XIUQsG36nYJhGEqNtko2pMqXG0UidVgkZHtFFE2/qlylYU6lUuQxqny5gFrnHQQBOV/20o6dkpJQ9dTrdVIHbhgGlpeXyTby+TwpEyvkVxDWyjif6MEz6V7cLG+WMa3LLg+7DpyytWEYcByH/H1fX9+mf0v6IbKeBU8AoZVARs/CdVtH6qpWqzCJjHDr7VDzdaJUwGRpDZd6B/CukXFcXVsl6+s2oq7HjXNtOJZAn+2g7HmYqRQRQkKgdT1R9ibDMMi+LPh1zFQryNoxvPfoCUzUKls+ZlMlRInih3ZDZAe+0wZs20YymSSPUQVL0HWd1A57nofZ2VmyjbW1NaWAnlpoURLCR0nm3imiBL/ZKe1uxoZhYGhoiDzmPe95D6m9dRwH6XS6ZXkYhvjSl75E9nViYgIrKystyy3LIvvYDlF13hSu65LrYWpqinTQhmFgamqKbKNcLpPz2a1W8cOJPpyw4zjvJBGUyigFD7TnUkqlnl+1We01lmWRYz04OIjHHnusZbkQAh/5yEc21WHfnkbqW68jSMTw/DOXcFaTKJdbR0tbWVkh5yIAFAoF8gZkrlTAjbVVXOjpxVO9g3AAFJscRZQgS/t1A9K8N0UJkEXtZ1LKLXEPzvf0QoNAyfdwt9RIPkLZWtf1LRdjzQwMDJD9EGGIcsKG4dj4oWOn8G1borLh3KSUmJ6eJtuoVCqkk99tjAx+hM4wzH0CKXGtlMeaV0fKMHEpQ29+Dz1hCD1fgDMxDa3qQqvVYawVYZQqjSQwe0ggJd7ILSNfd5E2bTzZP7yn7XUDFzP9EEKg6LmY2acsbgGA27UKSoGPhG7gnHN4HqOzA2cYZhM3S2so+HVoQuB0ouegu3Nw+AGcm1PIfOVlOBPTEEEIEQSIv3ELoy+/jvRibs+d+LX8CtZcF5oQuJDp39O2uoGxZAoSEiu1Gor7mPRlqt5w4ALAcbt1GN79hh04wzCbKPoe3HuPdnvMw6d93Q9EECD+5i3Er91B+dGzcI8OI4zb8AZ6sfZDz6B4chSj16fQP724p058zaujeu8VRvYQ6pD3GzcIICBgaGJfg6qUwwD1e9H2kocoRsLh6QnDMIeGyj2nkTToD7EeVszZRTh3ZrD2zqcQpJMwc2sw8gUgDBEkYlg5cQRz6RjOffMK3JiNYl/PnqVjLd8LD5oy9+4bjm5hxW18xJk0LRhC21cnXr333jt+iKRkfAfOHBq0MITlBzD9ACKUe/54kmlNw2lIJPbgo8nDjiklev76u3BHhxH0pBoR7WKNu18RBNDqXkN251hYHh3A2NUJGHX6K+R2KHl1QEok2YFj1W0EbbE0fd8vaKqhDwkgdogc+FtvdTKHCykRDyTO5KvoLXr3RCESgMBazMJCOgbPODwL5q1CyfchJWBqGiyh3X98+FYgAw2a56F6cuz+vwUxBxCi8R7cfeCsV48MoG9uBU65ipK9Nw6l5Nch0XBatqbD3aMIcN3ASq3aUA5pGlKmiUKwdxdOzVTupSY1hYAJgfbSpnQGvgNnDg4pkfVC/EDexbGKh7qhYykVw3KyodE9sVzA+fk8nHrrFJrM3uCGASQATQg4baSB7Ub6oKM+1I8wnbj/b2HMgfB9CLcOsS4HEgKebaHQm4ZZ2ztHUvV9SEhoQiD2FnwispFVtwoJwNYNZKzWWv69oHHhJKFBwO6gNLgddj0bLMsi9YWZTAZHjx5tWS6EwIc+9CGyjlgs9kC7K+WWd0yFQgH/83/+T7Kf3/ve91CrtY6V6/s+qc+Lmqh9v3SvzbpRlS58Oy3lTlFp4aWUSo315cuX0d+/+StavVrDsb9+DeUnH4M4NY7TR0YQ3htvPZSw3Dr6vnsVR6sVBM+9DVeuXCH7kcvlUCq1lpZ0UgfebO8o+cBVdlDpZnO5HIrF1nmJhRBYWloi61Cxbuuk0AEp4fkBcpUyvA19ixLYZy+CVkQlSlAgaqxDSEhNIJQPXuPUe3sQxByEjg3fsRDWqgjDENq9uzIpw011rge8oTBNk+zHeuCiPicGTWgIpEQtDDadW5Sx3K+9qXl+R9E21+vqL8k3nuPt4hoCKZG1HFzI9OF2vQKTWNfxeBxPPPEEWf973vMeMmZJvV5HpVLB2ellDBQqSFom/va7n76/V4VhiD/6oz8ix3l2dhb5fL5luSroTyt27cBVkdYMwyA3TCEEMplM6zqkRNJ2YOs6dM+HXvfg2xZCQ0doGpBCQNd1xGJ0uEdd18l+appGLoIoC/GgUfW/nd9HRbVhxmIxxOMP5BfC93HkW1dRHz+C4omjyPSkkehpkixJCTz/NJL/6U8RTszANE3yfFS23ms7tmuHMAzJOlRRB9froIgS2MPQNAzH4hAAZqol1De0G3U97Oea2a4tVXATihX4sKcXoJUqCJONORs6FpZfeBdEKCFNA6hVASmhex4yS3msDvXuuB3VHhqGIQxNw9FECpoQmKmW4G0z/u3Mu7127lTfVHN1OxZrFby0OIN3DY/hsewAXqusYZl4jB7FR2SzWTKIlOu6cEwTvbN5GLqBIJ1ETzZ7/4YyDENYlnUge9PhfB4jJWKVGo7emkUmX0JoGPBsE1bVBaTE6rER5EfpyF/M4SaxsAonX8TiE+c3P1kJQwCiEfRYCCAeA86dAL7xChBKEFETmQ7Rb8eQ0A1ISEyU1tQ/eMjIQyK0LcRvTaF06VxjHgoBGMaWt579M8sIDQ1ufG8kXv1ODAnDgoTE7UJ+T9roNv7/d2/h7QNHcCqdwdliFqu5xT2OTA9YdQ+637iQrcSdPVMc7JTD58ClRLJYwZGZZWiOjbmLJ1EayCI0dJhVF6nFVSSX87DLVawNZw66t8wuMcpVlEb64cc3vMcqVYBbk4BtAedOPvj3i2eAl74NSwLu4Vg3DzVDTgwxw4CUwJ174SrfSngCKDz9KBI3pmCMDsPv3UYiJiXihTKyi6uYPjUKz9kbBz4cSyC+botifk/a6DamykW8ujyPHxwexZPZAVy5FzmwLaSEUavDrNSg31MUBLYJz7HhaoBd86AHIaQAKon9ffdOcbgcuJQw6z5G7y6i0JNA8eQY9NiDqx0v7mD12AjWRvoxdH0KI29OwgSwf98hMp1Cr3uo9fZArm+MYQjcnAC+8TdAPA6cPfFg08ykgVQC8TLQOn0E0wk0IXAymUHatBBIiYm3oAMHgNrYCHTXQ+o7V+H1ZVA7fhR+JtXY6FfXMHx1AtryKhZHh5AfzO5JHzQhcDqdRdq0EcgQt4tvvach2+EGPr4+fxdP9g/hYiqL9wwcwZ/OTcJr43VAcmEF/TfuQgqg1tN4Hx7LFyGFhrnRAYhiBYYfQEKwA6c4MrMEp1rHzTNjsAwdevOVrxAIbAuLp0Zx7PY0HtUcfCfcm4TuzN4RWAasQglCovFYcq0ETEw33nsP9m2+4ylXgUIJe5s4kAGAISeORzN9sDUdr64uIld/a64taeionjmG0LGQfO0akm/cQr0vAyFDGKsFlG0Td08eQSm7d3Gxh2MJXMoOwNF1/M3y3H0N9FsdCeBGIYdvL8/j+SPH8KND43B0A3949xaCXUi7em9PYzBXRmFkAKXhvntPUyTMmovU3ApG7sxCVF1oMkQum4ZnHZ7gRofOgacLZawM9MA3ddz/BG79ymrDpu47FspjQzh6LY/v7Hsv35pYmgZDaHDDAEGbH7/4MQe9b05g9dwJBDEbYmERmF8CdAN426MPDpQSuHoDEBrcw6HcOHA6aYeNCADn0lmcSWcQQuJP7t7a83eLhxlp6KgdH4U7NgLh1qEXK4CQCFIJLJdKKNWqEKGEFgQINR3QOvd+RwC40NOLcz19CKTEH09cR3gIdMeHhVy9ht+/dRVx28bb+kfw3oFRxDQd/2n61v0oglE4AwOD86uYeudT8B3rwfcOAALbgptKIGMaGH7tOjzLwPyRw5Xc59BtibbrYWkgs/kOTALOWmlzZC4hUBgbRurwncJDhy4ExuIp/MNHnsb/+9n34bnB1vLAqFQGsvCScWRu34VZLCM+swT4PnBqHBjofWD/ag148zbw+IW22+x2DE3DWDyFf3DxKfy/3v4ePDdwpKP1H40n8Y7+ETi6gb9ZXsBUubVk7S2BEIAmIE0DYTIOb6Qf3vAAwkQc0DTYlRqOvTmBc996E5mlXEebPhpP4h0DRxAzDLy8NIvJt+DHhCpmqyX8/sQ1LLoVGJrAuwaO4v958hGMxhIwhdovZCDwPGxUzxyDH7MBTdvyrYNZceHkCpACgJRwY/ah+YAN2MEd+E7lBiqJhKZpsG170+fzAo33PqZpIrQs2LbdOCYMkVkpIOds/vrPNMvQNA2m2Pp16MZ+U/KbMAyVcobdyB020klJTXNdUWQi7fQ/YZh428AwPnjiPI4ne5Cv16Bj85iEYQjXpd9Or6ysbOnr2tFenH9zClauCGutiGrMQbE3hWBpCUJKxCouYpOzKGkS1WMjqHy1Sp6L53mk1rpdO1LstRQvbhh4auAIPnjiHI4l01iru9A1bcuYRtH8N/dVg8Cjvf14YfQkHsv0o1B38dXZSVS9ekML3YRqHPc7F3hzf1T9q1QqZF51IQSuXLnS0qYWBDJrZQxNLcCuVFGSEpMTE1gsPMj/XSwWUalUyH4EQbClr5oQuNjThw8cPYFL2X4UPBdfnZtEfY/m7l7uTUBnZGytjokbJo4l0nA0A+KePOWRdC9+6cRF/NXKPF5fW8GaEC3lzCOhBhnqqI+OIB6PN+I/BiG0ShVS06AXK0hNzCCeK8CLO7CqLmzDhNb0CF1KCV3XyXm3V7LKXTtwlZ5U0zRSnC6EQCKR2LxpSYkgEcPIWgXLQwNIJpNwnMYHA0Emg01KvTBEYuE65gwNjtFa5xeGIenAfd8ny6PqqPdLf7xdO1QfpZS73lAzlo0PjJ3CDx89gX4nhqVqGf/5zjW8ND+9acx830e1Sr+hnpqawtrag7sIMwgxWq7DdetIT81BEwJFTWB+YhLh5BQyuQL0Ug25ZAxzR/pRnZpALpcjz6VSqZAOrN2ANhtR6d53Q6s1lTFtvH/sJN535Bj67RiWa1X88eR1vLI0t+k41cUqsFVLnjRNvOfIcfzY2CmMJ9PI1138+cwdvLG6hLrfHZ+HbncRQ9mnUChgcnKSrPPLX/7ytraIQ+Cy5mAIFgIIzMoAL9WLeGO+itqG2wjXdVEo0B8A+r6/adNPGibeNTSKHzlyHKOJFPJuDX82fRs3CnlACGg7vHgH1Pv0Xu9NFFGce6tj1vem9x87jYxpYcWt4btryzifymIslsQHj57CE5l+XPcqSDsJFDU07qI3kHWBuRA4kU0jlkzBWFyBObcIY2kV0jCgVarQi2XUHRu54T5kp+aR0g248fimG8kwDGEYxoEELzp078DLfRlkZhaROzYCtA6OA63qwrwxgXnxVn5Lt3dYmoYfHz+Nnzh2BgnDxNXcMn7n+uu4vrYKr5242FIi7gU4U6hhpOpBA7BqG1hLx5FwHPQur0EKINQ0LA73IteXhmfSi+NhxtQ0fGDsFH5i/BTihok311bwuzffwI1CDn6b8clH4kl8+OQFvGPoKJKGiflKCX9w6w18O7eEcpc47/1EB/C45uAJzYEpgWnfxVfdAubCOtqNTj7sxPHTx87g7f0jSBgm5qtl/Kc7V/HK0jzboomNe1PadnCzvIY/nL6F6WoZQ3YMz/UN4wf7hnEumcEpLQujCuQ1YMZo/OcKwJZAXAKxELDvzCCZK0IrFKFVXQi3DmgapK4hSMRRODOOXNxCem750CVYOnQOfOXkKJxCGem5ZfjZ7NYP2O6FN0xeuYG1bBpvdPjdE9NACIF+J4a4bsAPA/z59G28ubay44+mBAABAT2U6KvWMVLxMVjzEPdDCACrto7XeuPQMj3o7+uFuFe9FKIRqnA9oMshWzj7hXbPDjHdgB+G+PLMBK4XVndlB00IGLqO8z19ePvgEbxtYBjDsSSEAK7klvH/ufItzNcqEG+x2OdREQCS0GBBwIfEd7wyZsP6jj/0W18TtqbjZCqDp/tH8GTvIAadOIQQuLq2it+49hpmKyX4b+HEJa3YtDfJEP9raRa3SgUEkLhTKWK6WsbLqwv46SMncDHTj54Q6AmBMR/wReMrdu2eA49LwLxys/GkYr0BQ4cUAv5ALyrPXEbNELAmp2FWXQR7lLBmtxwuBy4EAtvE4rljGH7jNsJAwjs1Dj8ZB3QdwvNg5AqI3ZqCUaoi/9zjCP4b/SiM2R1uEOC/Td7AWCKN0z1ZfPT848jXXXxnZWHb96IbMYRATDOQMAyMxRJ4LJXFO1frSBmN30kAniZwN2nhzR4HoRBIagIhO44tuEGAL07dwFgihVPpDH7h7CXk6y5eyy0p7WBqGuKGiZRp4WQqg7f1D+GJviH0mI2gI6GUKHp1fGV2Ar9/4wrce3G2D9emcHjwAXwnrKJP6BiEjh92elCuhrgTuMrvwzeuifFYAo+lenHmTBLpe/nWQylR8j38r/m7+MOJN+GGIcK3UAa4nbBxbzrXO4D/ffwMCl4dVwo5hJDwZIhb5QL+zY3X8EjfIP7us89jxAdiEnDW7wfv/WdKQHM9wDYBISCFgLQt1E+MoXrpHGDoEOUyUvMrCGwTob71Q7eD5PCtVSFQy6Qw/8gpjNycRuLFb8Md7kcQc2CuFWCurMEdGcDqey4hrFY4tOYeMlFcw7+/8To+cvYSTqUz+JVHn8JvXv0OJopr99/pCDQC8Q/bcRiaQFo3cTQWx6l4D47Hk8jcy9kbExqqukDV0FAydCzETSw6JsIOSm8eViZLBfzuze/j504/ilPpLH75whP4reuvY6K00Q4CgEQYNOJnZy0bx1I9uJDpw5l0Fr1O4zsRPwixWKtgqVrBbKWIby7O4tvLC6jznV4klhDixbCCHxAxDELHC04W/7OWx2LYyBG+Pps9PUR1fU0YJo46CZxOpHE8nrq/Jtx6HUtuFcu1KuaqJXxreQHfzS3u2QdrDxPre9MvXnwCZ3r68AvHL+D3p67jbrW0aW+qC+CWCczqwLgPnPIeOD1fADUBJKo1BKk4gkwaYSoJ7+gQvCODgNFI6GMVyrDXSlg6PY7wkGWDO1y9WeeeE195+2NI1jxolRo030ctPYLy+ZPwsj2Nwa3SX3ky7SEBfH91Gf/hxvfx/zh3CceSPfjY+ScwXS5AQDQkkxDQdQ2JREO60WfZSOgGhBDwwhAz1TLu1so4crQfXiqOoqmjamgPIrAxSiSAK7kV/MHtq/jImUcxnkjjF84+huly6b4N1v8XMoSp6xh0EkiZJoQQqIcBJoprmCiu4U4hj6nSGqZKBSzXKvDfoq8m2mFa+viGX8I79TgGNRM/7PRgOfTv39UJCIRmgOKoBVPbuiamq2VMV8u4tryAyVIB0+Uilt1qRzX9Dzvre9MfTt7A/3HGxmgsif99/CzmauVN68EyTRytAboEkmHjOwYACAAUNOCOAZw7MYpUoYwwlUTt3AmEmTQgAC1XgHV3DphdQK4/g9JQb0e1/p3gcDrwe4QxB/Vs5qC78ZYmhMRrq4v4zavfwf/52DMYjiXQ5zR99S8epOr0wxAT1RKuldZwu1LEct3Fml/HDz/1KNIx81A9fuomQkh8L7eE37r+On75wpMYisXRa2+2g9zwINcLAtxYy+H7uSW8ubaKhWoZObeGXK2qfPTO0EgAE74L1/fw404WGc1AStu8lYZhCDd5LwT0xjVRLmDp3pqYWliAt4OgI8xmQkhcKazi9+/ewC8ev4ABO4astTkmvaZpSN97uBQAWNGAeQNY1oGiBlQFMPjUI9BzRcS/+yasiWn4A72NC+GlHEJdx+q548j3xBEah+8VX1sOvF2NXxT9tSoPdTs6wk6Vq/Ifd5Lt2tnr1I2hlPh+bhm/+srX8Fh2EKamIYREKBv/abqObG8vPBliwa2i4HtwwxCefJAnPQjDTfmVt7QR0dYHnR5xL9tR2TGUElfyy/iX33kRj2b7YWjafRuEEghkCD8I4IUBZssl5Oou3MBHPXwQYDKK8243Be1hUAy0M0/W1zN1TCBDTAUefq+8iGOGAx0Nxx6iMcXrXh2z83PwQnlvTdRRD0PUN6yJELuXeEY5j43nclhpt2+BlLhWzOP/uvZtXEj3whBiw5qQcOJxPD74BALRuOOuisaj8wC4//o1MA3UTo2jfmQQolKDVioDEChfPo8wHkM1COAX1u5/QN1MGIYHtjft2oHX63VyEa+srMDzWssfhBD47d/+bbIO0zTJpPCu6+LmzZtkP4vFIqn/DYKgIxcinc572ylUGvUorP9+1avj60sz27Zh5RbJOv70T/8UBvH+yLIsMm+vlBJ3794lbVGtVsk5R5XtlOY5pcorr2kaOZeB6BdiS/Ua/tfC9JZ/b9aB3x+rjV/YKujEJqPazPZ6Pfi+T45llCArVKAXYPM5bteShDruQJT52O7FuSp/+0E69yjnptq7PM9DuVJBuVLBVH6rIskyTbzhN2JUtDrT+fn5lsFegIYd6/XW2c6klLh27Ro5lqVSaU9iVER24J2+y4t6xaIKUrLXHOar16h0wnbt1rF+lbrb8nYC0hwWoozhXkfGehjZi/PciePbblZ2+1zdDTu1Q9Qxase+Eq0d9zrt7j0HuTdxIHGGYRiG6ULYgTMMwzBMF8IOnGEYhmG6EHbgDMMwDNOFsANnGIZhmC6EHTjDMAzDdCG7zgeu+nTe8zyUSiWyzmvXrim1syp99cYc0636ocqRrMqXe9jlbnsdDEJ1jBCC1EkCQC6XIzWduq6TWkwAypzjvu/vmx2a50wUqYtK/7wfQTc6pUVvp469lrep1mOU9lV7xn5Ih6Lk8lbNKSmlMkBSp2hup1OySaqPzTnVtytfWFgg669Wq0q9uWpfUfm6er1OzinVfGvFrgO5qHSSUYTpCwsLbS121eSM2g/KeFGd20Hr1SnaDeQQRROrskOlUlFuRlSglyht7Oc4bzdnOhF0Q0W7tlRdFHeCKI6n0+3thCjrNUqUSBWqi6UoqPYm1bnv55rYTZRI1fmp+h8EAWmLIAhQLBbJOsrlsnK+qvrpui7ZRhSt+W7gR+gMwzAM04WwA2cYhmGYLoQdOMMwDMN0IezAGYZhGKYLYQfOMAzDMF0IO3CGYRiG6UIiy8jOnj276W+Vbk7TNOUxUWQWnZCRqTShKqlDu5KSY8eOkeU7odkOnaBd+VIUVLaOMl/alZHtpR32I3Vnp7TmB83x48c7Wt9O10SUNR9FB65CNZ+j/F5Vh6qf+7k3nTt3btPf7Wq8o5Sr2hFCKONLqKSVUeZLuzKy3a4JIQ9apMwwDMMwzI7hR+gMwzAM04WwA2cYhmGYLoQdOMMwDMN0IezAGYZhGKYLYQfOMAzDMF0IO3CGYRiG6ULYgTMMwzBMF8IOnGEYhmG6EHbgDMMwDNOFsANnGIZhmC6EHTjDMAzDdCHswBmGYRimC2EHzjAMwzBdyP8NAqx65HN/lEsAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 500x300 with 15 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "data_i, labels_i, latents_i, decodings_i = model_examples[\"square_kalmanSSL\"]\n",
    "\n",
    "video = data_i.cpu().numpy()\n",
    "pos = labels_i[\"pos\"].cpu().numpy()\n",
    "noise_level = labels_i[\"noise_level\"].cpu().numpy()\n",
    "frames = get_video_frames(video, pos, pointer=False, colormap=\"gray\")\n",
    "frames_pointer = get_video_frames(video, pos, pointer=True, colormap=\"gray\")\n",
    "\n",
    "image_decoding = decodings_i[\"observations\"].cpu().numpy()\n",
    "position_decoding = decodings_i[\"pos\"].cpu().numpy()\n",
    "frames_model = get_video_frames(image_decoding, position_decoding, pointer=True, colormap=\"gray\")\n",
    "\n",
    "# plot the video\n",
    "n_frames = 5\n",
    "skipframes = 6\n",
    "fig, axes = plt.subplots(3, n_frames, figsize=(n_frames * 1, 3))\n",
    "for i in range(n_frames):\n",
    "    axes[0,i].imshow(frames[i * skipframes])\n",
    "    axes[0,i].axis(\"off\")\n",
    "    axes[0,i].set_title(f\"Frame {i * skipframes}\")\n",
    "    axes[1,i].imshow(frames_pointer[i * skipframes])\n",
    "    axes[1,i].axis(\"off\")\n",
    "    axes[2,i].imshow(frames_model[i * skipframes])\n",
    "    axes[2,i].axis(\"off\")\n",
    "plt.tight_layout()\n",
    "plt.savefig(plot_folder + f\"circular_motion_videos_{model_name}_example{video_ind}.pdf\", dpi=300)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "f9e661e3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 100x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot single example trajectori\n",
    "plt.figure(figsize=(1, 1))\n",
    "plt.imshow(frames_pointer[i * skipframes])\n",
    "plt.axis(\"off\")\n",
    "plt.savefig(plot_folder + f\"circular_motion_trajectory_{model_name}_example{video_ind}.pdf\", dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "fca6601b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Eigenvalues of A: [0.9522043+0.31183642j 0.9522043-0.31183642j]\n",
      "Spectral radius of A: 1.0019655227661133\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAS0AAACOCAYAAABkHF4aAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAANgdJREFUeJztnXlc1NX+/58zw77LooCCyo6ioigCiuIGgpYb5patt7xl99c1raR7rdttsa7dupWlXW9pVmruWyrmvgCCJLIpmyIgg7Lv68zn9wdfCAUTcGAG/Twfj3kwzHzOOe/zmTmvOcv7fY5EEAQBERERkR6CVN0GiIiIiHQEUbRERER6FKJoiYiI9ChE0RIREelRiKIlIiLSoxBFS0REpEchipaIiEiPQhQtERGRHsVDIVqCIFBWVoboJysi8vCjpW4DVEF5eTlPmD7BwqiFlKWWIQ+Tk/BCAvmV+fh874PlB5YYeRgxxXEKg6wGqdvcbiM9PB2nICd1m9Fl/FH9Hva6P8o8FD2tJvIq83gz+01i7WI5wAFsj9hikm9C4teJVNdXE5EVQVF1kbrN7DYywjPUbUKX8kf1e9jr/iijcaJVWlqKt7c3RkZGJCYmtjudE058t/07quqrMNUzZdGQRTgvcabMqozTo06zNmYtOWU5pBSkdKH1XUd6eHqb/yuUCnWYo3aUgpI6RR01DTUoBaW6zRHpRjRueGhgYMAvv/zC66+/3qF0Flgw+NxgKp+vxK+fHw6DHRAGCaRFplHvVE9BeQGb4zdjaWDJIKtBmOqZdlENVEfLIU5GeAYOgQ5klWaRWphK6pZUzGzMaFA2YKZnRl/jvpjrm9PHqA9GOkbkxuaSfjidIYuGYOtlq+aaqI7Eg4kwCi6fPE7mvkJyvZLRG9IHIQY8pnlgbWxNb8Pe6jZTpAvRONHS1tbGysrqD6+pra2ltra2+f+ysjIKKSRhTAKr/FfRoNNARlEGCkFBdUM1b/q9yT/P/BPpRSk7TXcyqu8oRvcb3dVVeWAywjOaRauiroIjaUc4lXmKC7kXGJA7gItnL1LdUI2FvgW2xrbYm9njnO7M0OlDyf44m8LUQqI+i2L2j7PVXJMHp7q+gktpq4nbfJ0B+rso+3Y6DVkeKH+IpvfK3SQfCmaLxUc0GExkcNEwKg9W4rHQg74j+6rbdBEVo3Gi1R5Wr17Nu+++e8drQQTx7NxnqairoJdfL1xNXelr0perIVfJkGTw4ogXSdyfyEnXk0TmRDLIahDGusZqqkH7UQpKfsv9jcTbiRw4dYDf5L9hnWvNkLghFDsUI7eVI70o5aLTRQCm/zqdHYY78Bnng2WsJUOXDlVzDR4MpVLJ1ezvyLnxNjpKOTpCEAayOqznRFKSYoNtaCT6MjDVVuJlkUJeTQpRX87BLN2Dve/t5antT3Hr1C1xUv4hokeKVlhYGK+99lrz/2VlZfzJ7k+Y6JigratNiHMIfYz6AOD8pDPns89zIecC2XrZ1Cnq2Ht1L/72/njZeqmrCu1CoVRw5sYZ1v+0HsuTluT2ykWwFQiJC8Gy0JIFVxZQGFyIXpQeBn4GZJZkUq+o51z2Oc5xjrl2czHWMsaw3BAbYxt1V6fDlFdnE3t5NtRcRAfITbLh2jkn9jhlY+DpzDBfaxJcX6Cu7gY69Qp6N2hhrdfApGcjiMuxZofbPmp/q8X+gL0oWg8RPVK0dHV10dXVbfV6cW0x05ynNQsWgEwqw8/Oj7KEMkrSSrBxsSFSFsn57PO4WLhobG/r2oVrJBxI4KTxSQz3GWJRaEHgb4FYzLbA28Ob7DezCfhbAJbDLYn5JYZJ4yZxK/YWedl51NbWcl7/PGW1Zaw8tpIbJTeYWjyVCU9MaFWOproGZOefJuVKKFrKAmoVsPMm1Hw3HsdblgQmvkjQ8iBSzqcwxnsMN2Nucjz5OJLKWRRa78LIeQc2vun8ZWIc70cnMjnnr4woHoFDLweNra9I+9G41UOAkJAQjh49ygsvvMCmTZvalSaddJzNnfHo7dHqPS2pFjVba9DJ1WFS7CTqFHXsurKLmJsxGrnyVNtQy4H3D1B9vRrDfYZE+UZRa1PL5Dcms9R7KXNnz2XI40MICArAPNucqnNVjKkZQ+8jvdHO1WZO0hxWjVuFgY4B5XXl/Dvy3+zZvIeEWwkIgnDHSqQmugbEX19PSmIgWsoCblbDn3+TcJ1JBK4MwczZjNB3QnGzdENPSw9LA0syNmRQkVFB/Q54bORG7N0P0SAxwEIX3tG1Qnahmn/+959cLbiqkfUV6Rga2dM6dOhQh9NkkMEo21Foy7TbfN/vNT+yY7IxXmgM5RCVE8X57PP0M+2Hi4XLg5r8QLT89W9QNnAq8xQ/uP6AT5QPMWNiCJ4azGBhMPNmz2tVv6jPopon3McuH0v+pXxmvT0LbTdt9i/YT4aQQfzteFILU3nj1zd4f8L75B/J18jehiAIxKb9i7KbK9GSwG/F8OV1c54a+QpzBs3B3dKdE5EnmifXHYMcAfBZ5oM8Vo7PMh90ZDq420ymbpEJBfoFyHdZoX/LAqPdcpb3W86iqkXqrKJKWbZsGYsXL2bEiBHqNqVb0cieVmexNLS853u2Xra4T3NnSvAU+pv2p05Rx9GMo0TmRFJSU9J9RrZB06+/UlByIecCb/z6BrHGsWS6ZLJgzgJWjFlBb6PebQqyzzIfLFws8Fnmg62XLU7BTth62WJlaMWiPy3i8+DPmS2djVO6E5fPXmbFryvIq8jTuB6mIAjEpP6bspsrkUrgsBx+vDWUjwK/ZkbZDIb2Gdqq/k3C27LeTQyb4U3AqJM4PGeCvl0hU56LRC/xEnH744g+Ed2tdesqkpKScHNzU7cZ3c5DJVr3QyKR4GntyXSX6QCcyz7HiWsnuJBzQe1Omk2rhG8df4v42/HIJDI8rT15eeTLuFm6Nfcqmmj6v60G24Selh7j+o9j2uVpWBZa4hvpS0pkCkkHkzi0dxMX9q0kae9pYva9RVVVWrfUsy0EQSAq9VOy9n3TKFh5kCqdwdqQdcxyn0X+6fxO5SuTygic/x19xmlh6ipn0kVfDPL02fLOFnLLc1s57PY08vPzCQsLY/DgwXz55ZfqNqfbUIloVVRUqCKbLscxyJH+Zv2Z6jiVCQMaJ6V3X93N+azzxMpjmwOuu/vLLAgCUTlRvHvmXc5knQHgWc9nmbZoGv3N+gO0Gs61d3gnlUiZGjYVEycTej3VC99IXwxvGZD19S/8tjaN8kxtLq5NJTraDbl8k0rr1V4uXf8flTdfpyTGkWO3IF/vKf4qXYavnS86Mp07rr1bvO/3ukQioY/lVIwsFmAXGolBv0ISfbbzz9P/bHRUVTGCIFBZV6myx702Abh9+zbl5eW89957nDlzhsOHD6u8LppKp+e04uPjgcYPacuWLXz88ccqM+rNN98kIiKCAQMG8N1336Gt3fY8VUdpauhDrYcy3WU6tZG1RNhEsPPKTowuG6E1T4vhNsPvcOrsKprmsbKis4g/EE+0eTQHlQdxTHdkyLQhvD7m9U7NtbXVeG29bBk8fTAznpvBN1rvYfBZIXahkQCUp9rQLzQCUJKS8jympmMxMOi++a5U+WHyM19CVypwsxoKDP/EP/xXceXdK0jmSlpdf6/P5Y8+L4lEgpfHj1xUzuHW6HReC5Kz7PJ/6S3vzcTaiZjomqhsVbGqvgqj1UYPnE8TFWEVGOoYtno9ISGBefPmYWJiQnJyMgMGDKC+vp7//e9/vPTSSyorXxPpdE/rnXfeIS4ujri4OHJyclRm0OXLl7l58yZnz57Fzc2NnTt3qizvJuxM7PCy9WJC/gR0ZbpcKbhC5rFMzmadJeF2gsrLa4uM8AwalA0c/OAgVdeqEHY0/qJOyp/EqnGrOiRYLYXqjxqeo7kjCwINMPdOx9hVjrGrvPl5IxLy8r7tVH06Q05RLGlX5qArVXCpBKq1hvD3ce9gb2rffE1TOFJubG6ny3EMckQikeI15Gdk2tYYaMGHHgLJty/y7W/fUtNQ0+NWFRMSEpon4C9fvsywYcPYunUraWlpJCcnq9m6rqXTPa2///3veHk1OmeOHTtWZQZFREQQGBgIwNSpU9m4cSMLFiy445q2wng6gkQiYXTf0SQYJTDTdSY/J//MtaJryDPlKFOUCAeE5pi9rvLrEQSBiOwI9gzZg2uEK5G+kcxwmYFPjg+eNp4dyqsj9ulSDLTuwfyfVdTUZHao7M6SX36NuMuBGEmruVYBiSnzGXKlL9I0Kbn8HjfZcnW0s+FITfdHKtXBuvdsJDrn0EmpYvh1B77duhYLAwuslH8cOtZeDLQNqAhT3XSJgbZBm68nJCTw2GOPAY2iNWvWLH777TdeffVV+vfvr7LyNZEO97S++uorpk+fzsaNG9mwYQMxMTE4OrY9n9AZiouLMTExAcDU1JSiotZbyaxevRpTU9Pmh52dXYfL0dPSo69JX6Y4TmGe1jyc0p24cPIC1zdcpzyjnPDV4YBq/Zia5spuXrxJ4sFE1mxcwzHtY2Q4ZxAwJYCXTV+m8lwleb/lqazMJpp6Y3p6A+543XxUy/k7Sav3u4LCiptExgZgJCnidg3sKhzFuItTKUsvI+qzqDuEquXqqCpwDh6E9/AT5OwKoOGmBQsve7Ly1+VcL7muksUYiUSCoY6hyh4SSds/MN9++21zu/voo48YPXo0UqmU48ePP3AdNJ0Oi9a6devYvHkzS5cuxdjYmD179qjUIDMzs+aeU2lpKebm5q2uCQsLo7S0tPmRnZ3d4XJyY3PJPZaLe7E7Q88ObV5d2zl4J9XW1ZQ9VqbybWyaBPDoR0epyKhAZ3fjJLObhRuvj3mdii0VFKUWEfVZlErLhd97G9bWz2E+6nchNvf+/bkgKNAz+7030xULEsXVBZy+OAkTsimvh3XZA3g/8BumrJzSLE73cuNQBU5BTujr2zP5by+gb1fMkIWRPF+vTdrhNA4eOIhSUPbYVcUlS5bw3HPPqduMLqfdohUXF0dNTQ2TJ09GV1cXd3d35s+fz4cffqhSg/z8/Dh27BgA4eHhjBkzptU1urq6mJiY3PHoKE2/5nkb8xj5/0Yis5dhvMgYua2cuP5xHJAc4Pi14xRXFz9wnVqSXpROtF80BRYFRPpGMt15OtMWTcPT2lPlvYq2MDBwxnfxWzR+9DJAivL/FqiKYxw5lhDWvFGiqud5CqsK+TU6GHNSqFXAl5k2rJr8LZ7WnneIk6qFqi2cx0/AOXgQRq63sD3hi2m+CXve30NUdhTpR3qmaD0qtFu0Nm/ezIwZMzhz5gzDhw9n1apV7Nmzh+vXr6vUIE9PT/r06YO/vz9JSUnMmTNHpfk30VIgpgRPYUDQAEYEjGCBR+P82fHrxzn0yyHSDqcRcTwCeLCeR25sLlcOXmH9lvX8r/J/pDul4zDagVXjVjFp3iQkEkm3NFYAG5tn8PZOwd7+dXr3fgIDi+c4dNucohgn7IXjHIxZ+MA7vN59r3JKczgcPZ3ewkUUAnyT1ZtlE76l79W+9xwCNXEvd4YHxcBgIA5O6+gXGomBXSEEH+XPv/yZW5W3xPMGNJh2T8R/+umnzc/z8/NJSEggISGBQ4cOsWHDBpUatWbNGpXm1xZ3C0Rvw954OXihLdUmxTKFoxxFd48uiiwFp9ecxmyoGdnh2R2elG+ayD/5r5OUpJdwY8MNGmY3YGlgyZsT3mRk35H3bbRdgYGBEw4Oq4HGRYFyw/kc27QWJ8BeGc6BmMWY1r/Q6fxbuo2kFqRy5lIoTtqNK7PfZ5vzgv8mpjhM4fjXx3ENdgXuLU5d5X7iGORI/35B1EzO5daxKJ6ZKKfhOCQcSCB8RjhTgqdw/dfrGhny9CjTqdVDKysrJk6cyMSJE1Vtj1oZZDUIXZkuiiAFT/Z9khPyE9jIbUgcnohOmg59KvtQr6hHW6bd7lXFjPAMtEZrkTklk6KTRZz3OY+zuTNj7Mcw1n4sUon6gxIkEgkBAwJI653GbvlpZtuUYn7lEvH7IxgQOgDP8Z6dyrdB2UCc/BJnNi5jxNgElAL8lNuH+b7fMcVhClrSO79+Le9nV/Wu2irP1fEfxBr9DQhnTJQvt24ZsuPdHUjdpCgOK0TR0jDU32LUSMuG0fTc0dyRmU/NZJLDJAKDA7nmco3TeqfZsWcHGUcy2Ll3J7nlue2a7xEEgYKqAjZd2sQnhZ+Q6piKlpsW7094n/Gh49uMJeyOxtoW+XH5KKOU+OmsZ1+eKTk7fWnI1uf4e5+TXpDcoaFxbmwuKYdS2Ln7RyJ+m4rJZSOUAuzKdySk4SsCHQPRkmr9oQ9WdwvFqHnPYGG9tMVQ8TB/PriEjOIMSmtKu9UWkT9Go0Srs4dadJaWDaPl834m/Qh2CibQMRAPKw8MtA3oG94XIVsgfm08+67u42bZTQqqClrl2dS4S2pKiMyJJD4vnjWRayipKUFfW5/3JrxHiEsIg6a1fZSZun7Voz6LoiitCK3dWkwc/gPx46+i368Ql7nh/BY3jnM7fqWqvgq499xeeng6giBw7ONjFKcWkfv1QYYaFyEIEF4ygufG/IxhnGFzD6ula4O6cZ7qjIfrl9iPexxz73QWT5CzqLqW9CNprPtpHamFqeI8l4agUaLVdKhFaGiouk2ht2FvpjpNZcj0ISwZsYSk8UkUWBTwvcv37Nq3ixtHb7B111aOXzvO+V3nyS3PRV4u59L+S5y8fpLtSdtZ/9N6Ks9XYnrDlP6m/fGz82OG2wyMdFQX5qEqmhYm/F7zY6rTVGbN/4ycQYUIDnJ6axVSW7KfQzELuXI7jrTDbQdXJ/2SxPH0g8jHfI9evwKc50WQlWhLbvRwZpqsxauv1x3zd92xWtoRJBIJw9zXY2QyDiVSbI/70ivfjJSvL7Ly2Ep2bNpBWmEaBVUFZMdks/vJ3WqzNTMzEysrKwICAhg1ahRHjhxp87pnnnmGxMRETp06xYoVKzpURlxcHNHRmrcjhkbtp9WeQy26EzM9M0KfDsUm2wZTPVNy0nK41fcWvjt9IQcSv0ok1SIVmx02xPSNAaCysJL9l/YTnRuNxxYPBhYOJDgumPEvj8djoAdmembqrdQ9uHthYpLDJHLtcvmpIJsAw/NoSwQs6/aRkXCUG7f+xJmUG9hajEdH25DS6jyKy69yPe8MzlkbGT1MSfpoba7bFFO97m+Qo0C+UU6u2Z0nBHXXamlH6WXqhcug/ZTPfZXyVBtmvhhBNHIO/Shjn8k+XCxdMPjYgJpfa9R6aMj48ePZuXMnOTk5zJo1i6lTp6o0/7i4OCoqKvD29lZpvg+KRolWe3nQMJ6OoK+tT8CAAIx0jDhjfoZlPsuIro6mUF7IgWEHkMfImS2fzeXoy5hlmeF3wo/jpseR28qp9KvEqcCJKW9MIcQtBJNhHfcnUxcyqYwxc8bgNdyLb7d9i/45JVrj4hkwRI5uQypK+Zdk5spQoI2upIbyFBvqLsxGOakPOXZyMmvMGeu+C+d3XDmSeQSfZT4qCcnpDhyDHOnX2wnDBY7kn/o3pq5ypgBp5uWgE066HHJG2xNs/lmrtIIgoFRWqcwWqdTgvqvLJSUlCILAyZMnCQsLA+Dll1/mqaeeavP6bdu28dlnnyGRSHj33XcJCgoiICCAgwcPYmRkRGhoKJ988gnr1q2jqKiI/fv3c/ToUZXV6UFRi2jl5eUxf/78Vq9v27YNa2vr+6Zv6zSerkRLqoVPPx9yzXPxdPSkv2l/5ClyXHxd0L6gjU2yDZFukQyLHEavgl6Mix5H2fIyRo0bRb/6fsycMRM9Lb1us7ez3L0I4DzVGQCvCC+u3bpG3M+hnLi9F8cIJwwD4rEZLKcsegC63hlk7vClOtuCE99PpPdH7rj3HcAUxyBkzrLm3lTLHUY1maZ5xV7GbvTrMwcrpxAyMlYiAZyMwdkYsM4i4J3WwqtUVnH2rOqG//7+FchkrXd5ADh9+jRjx44lLi6O3bt3ExYWxsGDBzE1NcXX15e5c+e2SqNQKFi9ejUXLlygrq6OiRMnEhQU1Gb+L730EhUVFbzyyisqq48qUItoWVtbc+rUqU6nb+s0ns7EH3aUoY8NxdHBkUFWgzhkdogBQwYg/0mOUCjwVOpTVD5XifLfSkb8ZQROXk4MshrEdZPrPUKw4N6LAJPemETp5VImvTOJiM/6USWv4tQPjxG5cAfDDruTbKKFrX8dw1IUjHtzDgGjAiisLkQmld2RT1vDQXWtlrYXiUTC4H6z0Mn2IPXSLm7dHI50QDx1NerfFaJpeLh161ZOnjyJQqHA0rJx914nJydyc1uvyubn52Nvb4+enh56enpoa2vT0NBwR29O0xccNG54GBISQlxcHCkpKSxZsoRnnnmm1TX3Oo2nq2lq1P1M+tHfrD+jB4/m8uuXiboWxZBXh2A61JSsxCwmz5lMX+O+yKQytKZq3C3uMLZetjiHODN2ylhsTWzZl74P71Xe+Nv5czv6NvMmLqG3QW9u3L7BzMdnIpFI6BXUq115a7oPVJOoJnyVQNX1OrQPjyJ449v33KJbKjXA3191uzxIpW3v8tCSBQsW8Omnn1JSUkJBQQGmpqakpaVha9t6rtDKyoobN25QU1NDXV0ddXV1aGlp0atXL3JycnByciIpKQlonGNWKNS7o29baFyL6syhFurAMcgRMz0zxgeOp2ZGDUEzG7vY4Sbhd+wHpemNsqM4jHZg0PRBBE0LIicmh72xexlRPgJbF1uKdYvbnH/R9N7UH9H0+bUc2upr66Ovrd/m9RKJ5J7Dua7k2Wefpbi4mGnTpiGRSHjllVfQ129to0wmY+XKlYwbNw6pVMr7778PNM6BzZ07l6FDh9KnT+MRfL6+vjz11FNcuHCBLVu2dGt9/hDhIaC0tFQAhNLSUrWUf2TZkTafP0ykHUlrft5Ux12LdgnvSt8Vdi3adcfr7c2np/GwfrY9DYkgaPgAth2UlZVhampKaWlpp3Z8EOkcubG5zXteaZrbgsjDy0MhWoIgUF5ejrGxsVqCj0VERLqPh0K0REREHh00KoxHRERE5H6IoiUiItKjEEVLRKQbqapK49q1MJKTF3DtWlinTvZub7A0dE/AdElJCdu3b293vpmZmQ8UFiSKlohINyGXbyQ62o2srDXcvr2drKw1nT7Ze/z48Zw6dYo9e/awatUq1RtL14iWUqkURUtERJMQBAGForLVo6LiMikpfwKUgOKOvykpz1NREd9muvutkzUFSwOcPHkSHx8ffHx82Lx58z3TbNu2jdGjR+Pj40N4eONReQEBAVRUNHryh4aGkpmZybp16/j888+bzyFtYv369Xh7ezNx4kT27NnDunXrOH36NAEBASQnJ/Paa68xfvx4vL29iYuLa87/jTfeICgoiHXr1vHzzz8TEBDQ5hGB90PjPOJFRHoynQuYVnLx4rA237lXwPTdwdJAtwVMb9++nWPHjmFiYoJSqWT48OFkZGQ0nwb//vvvY2BgwKVLl1izZg0//fQTAEFBQfzrX//i1KlT2NnZ8cknn7T/FrVAFC0RkR7I3cHSgYGB3RYw/dFHH/Hqq68iCAJhYWGt4oDXrFnTfAygltbvEjNq1KhO17cl4vBQRESFNAVM3/3o1285jedMtoWMfv2Wt5nufgHTCxYs4NixYxQWFiKVSikoKKC+vr5dAdNlZWWtAqYbGhruGzA9ZMgQNm7cyIsvvsjHH398x3WFhYX8+uuvnD17lv/85z93CKBUKv3DfNuLKFoiIiqkKWD67oet7RLgXj0YAVvbP7eZrj0RHs8++ywbNmzgww8/ZNq0afj7+7crYDowMLBVwPTTTz99R8D0jh07WLhw4R15vPTSSwQEBLB8+XKefPJJbGxsqK6uJjQ0lIKCAszNzQkICGDHjh1t2jtkyBBiY2OZO3cuJSUl963f3Yge8SIi3YRcvomUlOcBCY0C1vjX1fVbbGyeUattPQlRtEREupGqqnTy8r6lpiYTPb0BWFs/j4HBw7V9UVcjipaIiEiPQpzTEhER6VGIoiUiItKjEEVLRESkRyGKloiIGkgPT1e3CT0WUbRERNRARrj6jyDrqYiiJSLSAykpKSEgIICAgADs7OxYu3atuk3qNsTYQxGRHoiZmRmnTp0iMjKS77//npdfflndJnUbYk9LRKSbyY3NJf1wOrmxrQOaO0J0dDTfffcdX331FQqFgnXr1qnIQs1GFC0RkW4m6rMoClMLifosqtN5xMbGsn79etatW4dMJmPr1q2kpaWRnJysQks1k4fCI148QkykJ5Ebm8ueJ/cw68dZnT4v0sbGBgcHB7S1tfH09MTV1ZWQkBD69++vYms1j4dCtMTDWkV6GuGvhRP0adub73WGb775Bm1tbZ577jmV5ampaNzwsLS0FG9vb4yMjEhMTFS3OSIiPYIlS5Y8EoIFGihaBgYG/PLLL4SGhqrbFBGRLsMxyFHdJvRYNE60tLW1sbKyUrcZjwyiZ7Z6cAoSt6PpLBonWu2htraWsrKyOx4PysPeeFvWr+XzR8Ez+2H/bB81eqRorV69GlNT0+aHnZ1dp/J5lBpvy/rdq64PU+O+12d7L/EW6Tn0SNEKCwujtLS0+ZGdnd3utEpBSXF1MZklmVw+cJni6mJqG2rvuOZR+DJXVaURs+8tkvaeJmbfW1RVpT1Uwn2vurRHvEU0G40M4wkJCSEuLo6UlBSWLFnCM888c8f7urq6rY4t+iMq6yrJq8hDXiEnuyybvPI8rhRcwfymOXGRcZjrm2NQYkBKQQq2xrZkhGf02DmH9PD0ZtubnisFJQ3KBkprSsm5mEPS/mjqXJ8nZ6cP5ZmDubg2iUpTNyor1wCqW4bXBJSCknpFPfKyG1TU5FFSXUROWQ7aUm3qlfXN17W8byKajUaK1qFDhzqcxhFHiquLm/206hX1ZJVmEbsvlrJhZRRWFpJwO4Gk/CSS8pOoV9YTlBdE+NlwbHJtmHtwLnv77cXNzw33Enfk5XKsjazJONqzBKxJcCvrKondF8sFmwucOnqKXtt78YnBJ7gc74ftNQdydo6mX2gE5anWjX9T+pB+5BrXzl/AYcxoQPMbclsCDZAZmUDS/mhKPT5HUXuVrD0hCINeBODGodnUDn6donotSnbOJMZ5DwHBiyg9WNZmXiKah0aKVmdwwom9V/fiWeOJma4ZyfnJyCvkVB6p5LLBZY5dP0ZpbWnz9W6FbrhmuJJ2Ow3PSE/MCsww2GvAf2T/4bHUxzh08BALhy5EekDa5hdYk77YTbYIgkBVXRUbL21kY9xGzOLNOLDnALN3zcY2zxbzg+bI5hxHX25Kv9BIjF3lmHunY+wq58oHs6nOtuDQe9/z7H4PDHUMNb7H2dK+9CPp9PLOITHtbZLftqL8mgeFW/WB4dTkWJCz0xeA6mwLKveNRheQ5JrSZ282NwZM4krG41w6fp7nvF4h/YjmfLYirXloRAtAT6ZHzM0YGqIaMB5jzG+5v1GSU8IvV38BwELfghE2IxjaZyg2n9tQUVDB0syl6L6hizxMTtnjZTjmO2KfYs/uU7s5nHGYp9OfxjjdmDH2YzDS+f24c01q0BnhGej56vF1zNdkJGWwff92AIIUQWhLtbkVfAvnQmcsnrZg7HBTqhIbhaol/UIjKU+1of+svfzw6xZ62/4VHcXI5vc1SaTvpqDwEDfzfkLv8hYA+oXaUHjVhmT/KMoldthdcaH0CQ/szJwpzchCZ7EbdTU3kKWUYj0nEq0cGyyS7LC7+hPv3fwKi+svMbh0MHamdy7waPI9eJR4qETLTN8MWyNbUmJTWKu7ltL4UmanzsZztCeDxgxiRM4IBvsOxtXCFa1/aLH/2n5m/XMW1iOs2X9hP4teXcSx5xtP651wcQInOYlOrA5/+fwvBIcEM7NsJuPmjkMmvddJwd1HUwMqrCokJjeG5//3PMoUJbMTZ9PXtS+DrAbhJ/cjzDsMD38PYupimLp8KteuhXGCuFb5NfW6DF3kuEmgPPq/xO0uRntSIVMeW6xWkW5LLHJjc0n9JQmp19do992P0BBErQKO3oIIQcIkbzOee+EgLhYunCs+R9ALjXN14ZHhBD3V+PzIlSMMX/j/2L1wI9XZUkr2+vLk33aTKEvir7s8GOv2DlOKAvGY7gFo1g/Vo0ynVw8VCgXV1dVAY+iNKnnzzTfx9/dn8eLF1NfX3z8BjcPDg78cZP3F9ZzKPMXVgquMvTAWy0JL5iTNYZ7HPAakDSDIMYiBvQZiN8oOp2AnbL1skUqk6Gvr427lzpx35mDuYs7T7z3Ns2nPYlloyZDTQ/gi+gv++81/2XhpI0nnklSytUhHuHtFM/VwKruSdvH4u4+TdzwPZYqSMRfGYFloyVt5b/F8+vOQDYU/FGJhYNEcSG5t/Rzmo35fNTMf1TJfCUXGYUSWWJG1wxfFTTMyvtjPd8e9qa57cF+4znK3y4JCUcPRf66jKK2U9E1alFy14cZ5J947MgRL+w85+FQ8/v39GW4zHEMdw3vmK5FI6GPiwIx3X8LC1RL3paO4ddWZqotOPFNviOLWcr78cjVnMs+gFJTdUVWRdtBp0Vq6dCmrVq3i6NGjfP/99yoz6PLly9y8eZOzZ8/i5ubGzp0725XOAguqtldxKP0QDcoGBlsNxu3Pbmj312bc8nEEOwVjpGN0z10gmsIqbL1scQ1xJTAkkOc+fA4DBwOk86TIJDKKq4v58y9/Zv2b6ylIKSDi0wiV1ft+3NFwC9M5nH6YebvmYX3YGotCC+YkzuFPH/4JSxdLZq6aif8KfyxcLPBZ5nNH/QwMnPFd/BaNH70Mc+/rgAyQYm4+hdCRH/KXkGtoPz0CiW0RdnMj6ZORy9X9pzmw40PUHV+fvP8c5y+4YxHyLfp2hZQERrJvgz+yPEtezl5J2NgwLAws7khzv5AZWy9bnINdCJi5Eq2zH1KdbUn2Tl9G9oIRxoWsPzGR5eHLWrnGiKiHTouWg4MDn3zyCdevX+fcuXMqMygiIoLAwEAApk6dyvnz59uVrpBCFDMVPKn7JN43vVmstxgXPxfcp7kzLnAc+XH5rXpHLb/MbXX7HX0cGfb4ML589Uu+CvmKXvq9UAgKdnnsosiyiCsBV7hdebvL/Lruzre6vpp1MesYu3EsaYVpKAQFBdMK0B+oz/JPlxMUEoRziDO2XrbYetk29yTvrp+NzTN4e6dgb/86vXs/gb3963h7pzB09uMAGOkY8eLivzF4ZgAX7YaS9rMvtTkW5Pw3lm0nhxC39/w9bewqFIoqUlJf5fbtbSjrMqkfKCfFMZ3zfYfz7OqlWLpaEvB6QPP19/psW77elpj5vuaHpasVU95+jXqdocgk8KKDgn5JOzi74yS79u4Se11qptOiNWLECKAxunzSpEkqM6i4+He3BVNTU4qKilpd01YYTzrpLJi1gJHnRyLLkaG9S5sgx6Dm4UFbG6/da37i7i+ztZE104RpjMoexbJey6geWE2KYwpv577Nol2LiNgVQYOyoUP1bE9jb+pd3bx4k6QDSSz+eDFLDy1FmibFJcOFv/X5G3tW7WH4jOEMGD2gQ+UbGDjh4LCaQYO24uCwGgMDp1b3w0jHiLen/oL9K7NR2hRjGxqJjTSJyB/fJ/LqByiVyi7zrm9KnxubS8rBeI5sGkvqyR0URTtxNMKGN5P7MLRfMLvn7WFc4Dicg53v2JvqXp9ty9fbuqZJ7J39xzBI5xBll8ZQnGKHwRFfjG6ZcPajLbx44AUSzyay+8ndD1RHkc7RYdH66quvmD59Onv37mXDhg3ExMSwZMkSlRlkZmbWHEtYWlqKubl5q2vuFcZTVltG9cxqdAboEPxWMFaGvwde+yzzuWO49Ee09WWO/jyaimsV+ET5sGveLmyNGxvIsevH2HN1D/+J+k+Hel3t9cYuqy3j+79/T1lGGcJ2AQGBGfEzsCiwYPi54Zgb3Hl/7teT6CgyqYzQ2YvwnD2BU9bDuFYJ2hKB2ry/s+/sYGrripuvfdCwqLvTNzSUEf7uFxSnV5C1bSAZP/tSlW1B5eG5HH82GjdLN3RkOg9Wwf+jrXt14T8XqLguUH/qfWwXFqJvV8ikZyMZrPyOz5d/RNLPSSopW6RjdFi01q1bx+bNm1m6dCnGxsbs2bNHpQb5+flx7NgxAMLDwxkzZkyra9oK40knndKaUnwn++LxmAfuY9yBO+eqWg6X2ktT+ibRG7t8LBMHTmSayzTCxoZhpGNEdX01b/z6BvN2zuP0jtPUNNQ0p+9srFt2TDbx++N54oMnWNt/LQUWBSSOT+TjSR+z4rMVWLpa4rvMt1W6+/UkOkLLhmygbcCaxw4gKd/AjQgXiq/aoHW1lCv7z3Fo2ys0KKpUIlTQuBNtReUVzkQMwHLaZvTtCrkREMnWEalo99dh+b+XY29qr1KBbmsI2fyZr5jM48+coP8UG/RdChhuBkHPnmDwvMEPVKZI52i3y0NcXBxubm5MnjwZXV1d3N3dcXd3Z/78+So1yNPTkz59+uDv74+9vT0rVqxodU1bYTwZZBDiHMIgu0GEy8KbX3/QhtuU/m7RG/bYMKaNn4ZXuRdRX0QRnxvPKU5hfNWY6MPRvDzqZdws3e5YJm963nSwwZBFQ1qJaL2inpSCFDa/vhn9a/oY7jUkb04edV51/PzWzwyzHoZEIrnDlq7am+nue6cl1cL0oDlacgsit8yilzSPuhwLbnybSHhva8rLViEIk5FI2u8ScrcbQVHxaZLSVlBcZIGFsphSO0h2yEA5aDK7Jn1IkjIJe2/7Vvap0hXhXp+5mclI/EZf4WL8XCzdLxHw0myVlSnSftotWps3byYpKYn8/HyGDx/OvHnzGDFiBJ6engwcOFClRq1Zs6ZT6fqZ9gO6Z4O1pi+2ZKcEo1tGLLm2hC8cv6BOUcd/f/svu67sInRQKKPLRnOz7Ca9DXs3p205vzb7x9nUKeoorCokqzSL9RfXs/vKbgxdDZmdNJuMyRl8M/0bbIps8LTx/ENbupKWvQ95rJxnPnmG8LQjZKYmYzgjEkNpOfLS4xw69SH6Ra+Sesi8TVFuy+dKEJQkndhF/O4L1Lk2OogWXhhN1KB44vv25rmBM3hhxgvIpDKSUN+QzDHIEQMDR/xHXyTr5jdqs+NRp92i9emnnzY/z8/PJyEhgYSEBA4dOsSGDRu6xLjO0t4J9o7SVvqmRvzYB4/hV+jHic9PkHY7jWu9r7H/wH6092nzndV3DLYcjM0uG5KHJWM8zRjZeRkZkzJ4++TbJOcnk3ArgWsl1+if2p8ypzKkDlL0x+iz7+199DfrT/hP4fe1pSu5u/fR37s/L3ovYXfkbk471BB1+hD2F5zoMzmelJ0JFKV6sOutfxDwtROK+FEMnTEWqVSbjPAMHAMHUlubS+rZ8yTsiafKdSl5O4dRleVB1g5f5DWglWOJw4VX+M/+/0de/zy1OfS2tQopkUjp3+8ltdgj0kmPeCsrKyZOnMjEiRNVbU+XoqqhYkuaGrG9tz0Xn7yIXp4eb+S+wYkJJ5DslWBeaE7vX3qTTz59svtw7t/n2D1nN0HWQYRnhUPWnfl5ZHowZf4U/uL9F7Lysuhv1ni6yt0ipSme2YY6hnwWvI5v//ctN3NucmrTWAYvOId+qg0DZh+kMFtO+pYgSnpNpFbQJTM7kBOnQpBJlFz512wqb3iQt3MY5rMikSfZsGtkBuMHzsIs34y5H8zFSMfoni4L3YGm3GeR33mownjUTXOv6++P8eSQJ7locpEzL55BZ54OacVp9MvrR/y4eHRlumhJteil1wt9LX2sDK3wtPZk8sDJaN/SJjQ4FJlURrbk933CNLnxSCVSpv1tGruv7Gb6h29wAFcK3dNJ1b/NiDLQ/r/rdCW1yKhDJlGiEED3sUgUyTb84hVLpb4DwQH2bF91DEtDS8JTwttcNNHk+yDSPWiUaJWWljJlyhSSk5OJiorCw8ND3Sa1i3utUI4PHE/N7Bpee/018qvyiaqMYsk7S1AoFSTkJPDpnz5FW6qNoY4hFvoW3Lp0iz0n93Dr0i1svWw19vCDtlbtGr3KnRk5YSQjGcnBvQdxDH6bI+lHuFKdwZ6sMnTSFLicc2OvaylVTjpY2vRh5Hh7vnjjBLYmtpxOOI2loaW6qiXSQ9ConUt76kk89/v1N9QxZIDZAHrp92KQ1SCG9BmC9yxvXCxcGNhrIGXnypBJZa0cYDW1V9GeVTu3EDfcrdyZpzMP1ww31rluZUH6axje7sXijL9yZPGv/BzauBrqYO6AnpbeHek1VbBF1I9GiVZ7T+LpioMtVEV7G1vLxt7kn9QRB1hNpK1J66jPoihKLSL2y1j8V/hj6WKJ/wp/dGQ6reJA7xdWJSICGiZa7UVVB1t0BXc3to44QHbWAVZTaEtoWgpxW/UThUqko6hlTisvL69Np9Rt27ZhbW193/RhYWG89tprzf+XlZVplHC15H5DqT9yNH0YuJ8Qi0Il0lHUIlrW1tacOnWq0+k7erCFJnO3o+nDPpfzsNdPpOvRuOFhSEgIR48e5YUXXmDTpk3qNqfLuXse62HseYhDQBFVIhHUvaubCigrK8PU1JTS0tLmbW16EuGvhRP06cN1dJeISFehUX5anaVJdzVpFbEj1NXV9VjbH3WMjY3vuRuuSNfwUPS0rl27hqOjOFci0v3cvn27XW46IqrjoehpNW0UmJWVhampabeW3bRymZ2drZahqTrLf1TLblm+jo5qNiEUaT8PhWhJpY3rCaampmqb0zIxMVHrfJo6y39UywbEoaEa0LjVQxEREZE/QhQtERGRHsVDIVq6urq88847anE4VWfZ6i7/US1bE8p/lHkoVg9FREQeHR6KnpaIiMijgyhaIiIiPYqHQrTefPNN/P39Wbx4MfX19V1eXmlpKd7e3hgZGZGYmAjAjh078PPzY9KkSeTk5HRZ2dHR0fj6+jJu3DgWLFhAfX19t5V969Yt/Pz8GD9+PBMnTkQul3Pu3Dn8/PwYO3YsCQkJXVZ2E1u3bm125uyuegNkZmZiZWVFQEAAAQEB5Ofnd2v5Ii0QejhxcXHCokWLBEEQhPfff1/YsmVLl5dZV1cn3L59W3j66aeFhIQEob6+XvDx8RFqa2uFc+fOCS+++GKXlZ2bmytUVVUJgiAIK1euFHbs2NFtZTc0NAgKhUIQBEHYuHGj8N577wnjxo0TioqKhBs3bgjBwcFdVnZT+bNmzRKGDx/erfdcEATh+vXrwpw5c5r/7+7yRX6nx/e0IiIiCAwMBGDq1KmcP3++y8u8e4fVtLQ03N3d0dHRYcyYMcTHx3dZ2TY2Nujr6wOgo6NDSkpKt5Utk8maHXnLy8txdHREJpPRq1cv7O3tKSoq6rKyobGXNXfuXKRSabfe8ybOnz+Pv78/b731llrKF2mkx4tWcXFxs0e0qalplzec+9kAoFAourzMGzducPToUcaOHdutZcfFxTF69GjWrl2Ln5/fHWVraWlRV1fXJeUqFAq2b9/OvHnzgO6/5zY2NqSnp3PmzBlu377N7t27u/0zF2mkx4fxmJmZNe+QUFpa2hyHqC4boLFH0pWUlZWxePFiNm3ahEKh6NayPT09uXDhAtu3b+eDDz64o+yGhoYui8X78ccfeeKJJ5p7et19z1tuPDl79mw2bdqEkZFRt5Uv8js9vqfl5+fHsWPHAAgPD2fMmDHdboOzszNXrlyhrq6OiIgIhg4d2mVlNTQ0MH/+fN555x1cXV27teyWvShTU1OMjIxoaGigpKSE7OzsLv3BSE5OZvPmzUydOpW0tDS+/PLLbqs3NA6Hmzh79izTpk3r1vJFWqDuSTVVsGLFCmHs2LHCwoULhdra2m4pMzg4WLCxsRF8fHyEjRs3Ctu2bRN8fX2FCRMmCFlZWV1W7ubNmwVzc3Nh/Pjxwvjx44Vt27Z1W9kXLlwQ/P39hYCAAGHq1KlCbm6ucPr0acHX11fw8/MT4uLiuqzslnh5eQmCIHRbvQVBEA4dOiSMGDFCGDt2rLB48WKhvr6+W8sX+R3RI15ERKRH0eOHhyIiIo8WomiJiIj0KETREhER6VGIoiUiItKjEEVLRESkRyGKloiISI9CFC0REZEehShaDynV1dXN26gYGxsTEBDAgAEDuiWgXESkKxGdSx8BRo4cycWLF9VthoiIShB7Wo8Q//jHPzh48CCZmZn4+fkxb948Bg8ezM8//8z06dMZNmwYaWlpAGzatAl/f3/8/Pw4ceKEmi0XEfmdHr/Lg0jnKC4u5uzZsxw/fpywsDBiYmI4cOAAP/zwA6+++irbtm3jzJkzVFVVMW3aNCZOnKhuk0VEAFG0HlkGDRqETCbD1tYWDw8PpFIpffv25dixY2RkZJCUlMSECRMAyM/PV7O1IiK/I4rWI0rL49xbPhcEAQcHB4YOHcrBgweRSCTdsu++iEh7EUVLpBWWlpbMnz+f8ePHI5PJGDJkCF988YW6zRIRAcTVQxERkR6GuHooIiLSoxBFS0REpEchipaIiEiPQhQtERGRHoUoWiIiIj0KUbRERER6FKJoiYiI9ChE0RIREelRiKIlIiLSoxBFS0REpEchipaIiEiP4v8D74zGbyyzbesAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 300x140 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "data_i, labels_i, latents_i, decodings_i = model_examples[\"square_kalmanSSL\"]\n",
    "data, labels, latents, decodings, model, cfg = model_dict[\"square_kalmanSSL\"]\n",
    "shift = 1 # model.prediction_shift\n",
    "\n",
    "def plot_latent_trajectory(ax, dim, state, state_cov, inferences, inference_cov):\n",
    "    ax.plot(state[:, dim], 'g', label=r'$h_t$')\n",
    "    ax.fill_between(range(state.shape[0]-shift), \n",
    "                    (state[:-shift, dim] - 2 * np.sqrt(state_cov[:-shift, dim, dim])), \n",
    "                    (state[:-shift, dim] + 2 * np.sqrt(state_cov[:-shift, dim, dim])), \n",
    "                    color='g', alpha=0.3)\n",
    "    ax.errorbar(range(inferences.shape[0]-shift),\n",
    "                inferences[:-shift, dim], \n",
    "                yerr=np.sqrt(inference_cov[:-shift, dim, dim]) * 1.8,\n",
    "                fmt='o', color='purple', alpha=1, label=r'$z_t$',\n",
    "                markersize=0.7, linewidth=0.4, zorder=10)\n",
    "    ax.set_xlim(0, 50)\n",
    "    # ax.set_title(f'Dimension {dim}')\n",
    "\n",
    "\n",
    "# Get final estimates\n",
    "A = model.encoder.A.data.cpu().numpy()\n",
    "# print(A)\n",
    "print(f\"Eigenvalues of A: {np.linalg.eigvals(A)}\")\n",
    "print(f\"Spectral radius of A: {np.max(np.abs(np.linalg.eigvals(A)))}\")\n",
    "D = model.encoder.D.data.cpu().numpy()\n",
    "b_z = model.encoder.b_z.data.cpu().numpy()\n",
    "estimates = latents_i[\"estimates\"].cpu().numpy() - b_z\n",
    "estimates_latent = np.einsum('ij,tj->ti', D, estimates)\n",
    "estimates_cov = latents_i[\"estimates_covariances\"].cpu().numpy()\n",
    "estimates_cov_latent = np.einsum('jk,ikl->ijl', D, np.einsum('ijk,kl->ijl', estimates_cov, D.T))\n",
    "inferences = latents_i[\"inferences\"].cpu().numpy() - np.dot(D, b_z)\n",
    "inferences_cov = latents_i[\"inferences_covariances\"].cpu().numpy()\n",
    "predictions = latents_i[\"predictions\"].cpu().numpy() - np.dot(D, b_z)\n",
    "prediction_cov = latents_i[\"prediction_covariances\"].cpu().numpy()\n",
    "\n",
    "sim_start = 25\n",
    "n_steps = estimates.shape[0]\n",
    "simulated_states_steps = np.arange(sim_start, n_steps)\n",
    "simulated_states = np.zeros((n_steps-sim_start, estimates.shape[1]))\n",
    "simulated_states[0] = estimates[sim_start]\n",
    "for t in range(n_steps-1-sim_start):\n",
    "    simulated_states[t+1] = A @ simulated_states[t]  \n",
    "simulated_states = np.einsum('ij,tj->ti', D, simulated_states)\n",
    "\n",
    "    \n",
    "# Plot results\n",
    "n = 2\n",
    "fig, ax = plt.subplots(n, 1, figsize=(P_width, P_height*n))\n",
    "\n",
    "for i in range(n):\n",
    "    plot_latent_trajectory(ax[i], i, estimates_latent, estimates_cov_latent, inferences, inferences_cov)\n",
    "    ax[i].plot(simulated_states_steps[:-shift], simulated_states[:-shift, i], color='y', label='Rollout')\n",
    "    ax[i].plot(simulated_states_steps[:-shift][0], simulated_states[:-shift, i][0], color='y', marker='o', markersize=5,label='Rollout start')\n",
    "\n",
    "    ax[i].set_ylabel(r'$h_{}$'.format(i+1))\n",
    "    ax[i].spines['top'].set_visible(False)\n",
    "    ax[i].spines['right'].set_visible(False)\n",
    "    if i < n-1:\n",
    "        ax[i].spines['bottom'].set_visible(False)\n",
    "        ax[i].set_xticks([])\n",
    "    else:\n",
    "        ax[i].set_xlabel('Time')\n",
    "ax[0].legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "\n",
    "ax[0].set_ylim(-1.5,1.5)\n",
    "ax[1].set_ylim(-1.5,1.5)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(plot_folder + \"latent_trajectory_square_kalmanSSL.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "82df1197",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 330x140 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "data_i, labels_i, latents_i, decodings_i = model_examples[\"square_kalmanSSL\"]\n",
    "pos = labels_i[\"sin_pos\"].cpu().numpy()\n",
    "fig, ax = plt.subplots(2, 1, figsize=(P_width*1.1, P_height*2))\n",
    "ax[0].plot(pos[:, 0], 'orange', label='Ground Truth Pos.')\n",
    "ax[1].plot(pos[:, 1], 'orange', label='Ground Truth Pos.')\n",
    "\n",
    "for key in [\"square_kalmanSSL_procrustes\", \"square_kalmanSSL\"]:\n",
    "    data_i, labels_i, latents_i, decodings_i = model_examples[key]\n",
    "    data, labels, latents, decodings, model, cfg = model_dict[key]\n",
    "    b_z = model.encoder.b_z.data.cpu().numpy()\n",
    "    estimates = latents_i[\"estimates\"].cpu().numpy() - b_z\n",
    "    estimates_cov = latents_i[\"estimates_covariances\"].cpu().numpy()\n",
    "    pos_inferred = labels_i[\"sin_pos\"].cpu().numpy()\n",
    "    # plot decodings of labels\n",
    "    pos_decoding_matrix = model.decoder.decoders[\"sin_pos\"].linear.weight.data.cpu().numpy()\n",
    "    bias = model.decoder.decoders[\"sin_pos\"].linear.bias.data.cpu().numpy()\n",
    "    pos_decoding = (pos_decoding_matrix @ estimates.T).T\n",
    "    pos_decoding_cov = np.einsum('jk,ikl->ijl', pos_decoding_matrix, np.einsum('ijk,kl->ijl', estimates_cov, pos_decoding_matrix.T))\n",
    "    color = 'lightgreen' if key == \"square_kalmanSSL_procrustes\" else 'darkgreen'\n",
    "    suffix = 'Stopgrad' if key == \"square_kalmanSSL\" else 'KNN Entropy'\n",
    "    ax[0].plot(pos_decoding[:, 0], color, label='Est. Pos. ' + suffix)\n",
    "    ax[0].fill_between(range(pos_decoding.shape[0]),\n",
    "                        (pos_decoding[:, 0] - 2 * np.sqrt(pos_decoding_cov[:, 0, 0])), \n",
    "                        (pos_decoding[:, 0] + 2 * np.sqrt(pos_decoding_cov[:, 0, 0])), \n",
    "                        color=color, alpha=0.3)\n",
    "    ax[1].plot(pos_decoding[:, 1], color, label='Est. Pos.' + suffix)\n",
    "    ax[1].fill_between(range(pos_decoding.shape[0]),\n",
    "                        (pos_decoding[:, 1] - 2 * np.sqrt(pos_decoding_cov[:, 1, 1])), \n",
    "                        (pos_decoding[:, 1] + 2 * np.sqrt(pos_decoding_cov[:, 1, 1])), \n",
    "                        color=color, alpha=0.3)\n",
    "    \n",
    "ax[0].legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "ax[0].set_ylim(-2,2)\n",
    "ax[1].set_ylim(-2,2)\n",
    "ax[0].set_xlim(0, 50)\n",
    "ax[1].set_xlim(0, 50)\n",
    "ax[1].set_xlabel('Time')\n",
    "ax[0].set_ylabel(r'x Position')\n",
    "ax[1].set_ylabel(r'y Position')\n",
    "ax[0].set_xticks([])\n",
    "ax[0].spines['top'].set_visible(False)\n",
    "ax[0].spines['right'].set_visible(False)\n",
    "ax[0].spines['bottom'].set_visible(False)\n",
    "ax[1].spines['top'].set_visible(False)\n",
    "ax[1].spines['right'].set_visible(False)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(plot_folder + \"position_estimates_square_kalmanSSL.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "97d9b1c0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 130x70 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot learning curves from wandb\n",
    "import wandb\n",
    "import pandas as pd\n",
    "\n",
    "# get wandb api key\n",
    "wandb.login()\n",
    "api = wandb.Api()\n",
    "\n",
    "def get_cos_sim(model_name):\n",
    "    # get the project name\n",
    "    project_name = \"probSSL\"\n",
    "    folder_name = \"../trained_models/\" + model_name + \"/\"\n",
    "    # get name of the most recent model by folder creation time\n",
    "    name = sorted(os.listdir(folder_name), key=lambda x: os.path.getctime(os.path.join(folder_name, x)))[-1]\n",
    "    run = api.run(f\"{project_name}/{name}\")\n",
    "    metrics = run.history()\n",
    "    cos_sim = metrics[\"grad_cos_sim\"]\n",
    "    # remove nan values\n",
    "    cos_sim = [x for x in cos_sim if not pd.isna(x)]\n",
    "    return cos_sim\n",
    "\n",
    "cos_sim_proc = get_cos_sim(\"square_kalmanSSL_procrustes\")\n",
    "cos_sim_stop = get_cos_sim(\"square_kalmanSSL\")\n",
    "\n",
    "plt.figure(figsize=(1.3,0.7))\n",
    "plt.plot(cos_sim_proc, label='Trained with KNN entropy', color='lightgreen')\n",
    "plt.plot(cos_sim_stop, label='Trained with Stopgrad', color='darkgreen')\n",
    "# remove spines\n",
    "plt.gca().spines['top'].set_visible(False)\n",
    "plt.gca().spines['right'].set_visible(False)\n",
    "plt.xlabel(\"Epoch\")\n",
    "plt.ylabel(\"Grad. Cos. Sim.\")\n",
    "plt.yticks(ticks=[0, 0.5, 1.0], labels=[0, 0.5, 1.0])\n",
    "# legend outside\n",
    "plt.legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "plt.savefig(plot_folder + \"grad_cos_sim_square.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "bb59c08e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Eigenvalues of A: [0.925777+0.3011849j 0.925777-0.3011849j]\n",
      "Spectral radius of A: 0.9735376238822937\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 240x140 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "data_i, labels_i, latents_i, decodings_i = model_examples[\"kalmanSSL_inferred_sigmas\"]\n",
    "data, labels, latents, decodings, model, cfg = model_dict[\"kalmanSSL_inferred_sigmas\"]\n",
    "shift = 1 # model.prediction_shift\n",
    "\n",
    "def plot_latent_trajectory(ax, dim, state, state_cov, inferences, inference_cov):\n",
    "    ax.plot(state[:, dim], 'g', label=r'$h_t$')\n",
    "    ax.fill_between(range(state.shape[0]-shift), \n",
    "                    (state[:-shift, dim] - 2 * np.sqrt(state_cov[:-shift, dim, dim])), \n",
    "                    (state[:-shift, dim] + 2 * np.sqrt(state_cov[:-shift, dim, dim])), \n",
    "                    color='g', alpha=0.3)\n",
    "    ax.errorbar(range(inferences.shape[0]-shift),\n",
    "                inferences[:-shift, dim], \n",
    "                yerr=np.sqrt(inference_cov[:-shift, dim, dim]) * 1.8,\n",
    "                fmt='o', color='purple', alpha=1, label=r'$z_t$',\n",
    "                markersize=0.7, linewidth=0.4, zorder=10)\n",
    "    ax.set_xlim(0, 50)\n",
    "    # ax.set_title(f'Dimension {dim}')\n",
    "\n",
    "\n",
    "# Get final estimates\n",
    "A = model.encoder.A.data.cpu().numpy()\n",
    "# print(A)\n",
    "print(f\"Eigenvalues of A: {np.linalg.eigvals(A)}\")\n",
    "print(f\"Spectral radius of A: {np.max(np.abs(np.linalg.eigvals(A)))}\")\n",
    "D = model.encoder.D.data.cpu().numpy()\n",
    "b_z = model.encoder.b_z.data.cpu().numpy()\n",
    "estimates = latents_i[\"estimates\"].cpu().numpy() - b_z\n",
    "estimates_latent = np.einsum('ij,tj->ti', D, estimates)\n",
    "estimates_cov = latents_i[\"estimates_covariances\"].cpu().numpy()\n",
    "estimates_cov_latent = np.einsum('jk,ikl->ijl', D, np.einsum('ijk,kl->ijl', estimates_cov, D.T))\n",
    "inferences = latents_i[\"inferences\"].cpu().numpy() - np.dot(D, b_z)\n",
    "inferences_cov = latents_i[\"inferences_covariances\"].cpu().numpy()\n",
    "predictions = latents_i[\"predictions\"].cpu().numpy() - np.dot(D, b_z)\n",
    "prediction_cov = latents_i[\"prediction_covariances\"].cpu().numpy()\n",
    "\n",
    "sim_start = 25\n",
    "n_steps = estimates.shape[0]\n",
    "simulated_states_steps = np.arange(sim_start, n_steps)\n",
    "simulated_states = np.zeros((n_steps-sim_start, estimates.shape[1]))\n",
    "simulated_states[0] = estimates[sim_start]\n",
    "for t in range(n_steps-1-sim_start):\n",
    "    simulated_states[t+1] = A @ simulated_states[t]  \n",
    "simulated_states = np.einsum('ij,tj->ti', D, simulated_states)\n",
    "\n",
    "    \n",
    "# Plot results\n",
    "n = 2\n",
    "fig, ax = plt.subplots(n, 1, figsize=(P_width*0.8, P_height*n))\n",
    "\n",
    "for i in range(n):\n",
    "    plot_latent_trajectory(ax[i], i, estimates_latent, estimates_cov_latent, inferences, inferences_cov)\n",
    "    # ax[i].plot(simulated_states_steps[:-shift], simulated_states[:-shift, i], color='y', label='Rollout')\n",
    "    # ax[i].plot(simulated_states_steps[:-shift][0], simulated_states[:-shift, i][0], color='y', marker='o', markersize=5,label='Rollout start')\n",
    "\n",
    "    ax[i].set_ylabel(r'$h_{}$'.format(i+1))\n",
    "    ax[i].spines['top'].set_visible(False)\n",
    "    ax[i].spines['right'].set_visible(False)\n",
    "    if i < n-1:\n",
    "        ax[i].spines['bottom'].set_visible(False)\n",
    "        ax[i].set_xticks([])\n",
    "    else:\n",
    "        ax[i].set_xlabel('Time')\n",
    "ax[0].legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "\n",
    "ax[0].set_ylim(-1.5,1.5)\n",
    "ax[1].set_ylim(-1.5,1.5)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(plot_folder + \"latent_states_inferred.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "3eede712",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x140 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "fig, ax = plt.subplots(2, 1, figsize=(P_width, P_height*2))\n",
    "\n",
    "data_i, labels_i, latents_i, decodings_i = model_examples[\"kalmanSSL_inferred_sigmas\"]\n",
    "data, labels, latents, decodings, model, cfg = model_dict[\"kalmanSSL_inferred_sigmas\"]\n",
    "\n",
    "pos = labels_i[\"pos\"].cpu().numpy()\n",
    "ax[0].plot(pos[:, 0], 'orange', label='Ground Truth Pos.')\n",
    "ax[1].plot(pos[:, 1], 'orange', label='Ground Truth Pos.')\n",
    "\n",
    "b_z = model.encoder.b_z.data.cpu().numpy()\n",
    "estimates = latents_i[\"estimates\"].cpu().numpy() - b_z\n",
    "pos_inferred = labels_i[\"pos\"].cpu().numpy()\n",
    "# plot decodings of labels\n",
    "pos_decoding_matrix = model.decoder.decoders[\"pos\"].linear.weight.data.cpu().numpy()\n",
    "bias = model.decoder.decoders[\"pos\"].linear.bias.data.cpu().numpy()\n",
    "pos_decoding = (pos_decoding_matrix @ estimates.T).T\n",
    "pos_decoding_cov = np.einsum('jk,ikl->ijl', pos_decoding_matrix, np.einsum('ijk,kl->ijl', estimates_cov, pos_decoding_matrix.T))\n",
    "color = 'darkgreen'\n",
    "suffix = 'Stopgrad' \n",
    "ax[0].plot(pos_decoding[:, 0], color, label='Est. Pos. ' + suffix)\n",
    "ax[0].fill_between(range(pos_decoding.shape[0]),\n",
    "                    (pos_decoding[:, 0] - 2 * np.sqrt(pos_decoding_cov[:, 0, 0])), \n",
    "                    (pos_decoding[:, 0] + 2 * np.sqrt(pos_decoding_cov[:, 0, 0])), \n",
    "                    color=color, alpha=0.3)\n",
    "ax[1].plot(pos_decoding[:, 1], color, label='Est. Pos.' + suffix)\n",
    "ax[1].fill_between(range(pos_decoding.shape[0]),\n",
    "                    (pos_decoding[:, 1] - 2 * np.sqrt(pos_decoding_cov[:, 1, 1])), \n",
    "                    (pos_decoding[:, 1] + 2 * np.sqrt(pos_decoding_cov[:, 1, 1])), \n",
    "                    color=color, alpha=0.3)\n",
    "    \n",
    "ax[0].legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "ax[0].set_ylim(-2,2)\n",
    "ax[1].set_ylim(-2,2)\n",
    "ax[0].set_xlim(0, 50)\n",
    "ax[1].set_xlim(0, 50)\n",
    "ax[1].set_xlabel('Time')\n",
    "ax[0].set_ylabel(r'x Position')\n",
    "ax[1].set_ylabel(r'y Position')\n",
    "ax[0].set_xticks([])\n",
    "ax[0].spines['top'].set_visible(False)\n",
    "ax[0].spines['right'].set_visible(False)\n",
    "ax[0].spines['bottom'].set_visible(False)\n",
    "ax[1].spines['top'].set_visible(False)\n",
    "ax[1].spines['right'].set_visible(False)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(plot_folder + \"position_decoding_inferred.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "aab5466e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x140 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "data_i, labels_i, latents_i, decodings_i = model_examples[\"kalmanSSL_figure8\"]\n",
    "pos = labels_i[\"pos\"].cpu().numpy()\n",
    "fig, ax = plt.subplots(2, 1, figsize=(P_width, P_height*2))\n",
    "ax[0].plot(pos[:, 0], 'orange', label='Ground Truth Pos.')\n",
    "ax[1].plot(pos[:, 1], 'orange', label='Ground Truth Pos.')\n",
    "\n",
    "for key in [\"kalmanSSL_figure8_procrustes\", \"kalmanSSL_figure8\"]:\n",
    "    data_i, labels_i, latents_i, decodings_i = model_examples[key]\n",
    "    data, labels, latents, decodings, model, cfg = model_dict[key]\n",
    "    b_z = model.encoder.b_z.data.cpu().numpy()\n",
    "    estimates = latents_i[\"estimates\"].cpu().numpy() - b_z\n",
    "    estimates_cov = latents_i[\"estimates_covariances\"].cpu().numpy()\n",
    "    pos_inferred = labels_i[\"pos\"].cpu().numpy()\n",
    "    # plot decodings of labels\n",
    "    pos_decoding_matrix = model.decoder.decoders[\"pos\"].linear.weight.data.cpu().numpy()\n",
    "    bias = model.decoder.decoders[\"pos\"].linear.bias.data.cpu().numpy()\n",
    "    pos_decoding = (pos_decoding_matrix @ estimates.T).T\n",
    "    pos_decoding_cov = np.einsum('jk,ikl->ijl', pos_decoding_matrix, np.einsum('ijk,kl->ijl', estimates_cov, pos_decoding_matrix.T))\n",
    "    color = 'lightgreen' if key == \"kalmanSSL_figure8_procrustes\" else 'darkgreen'\n",
    "    suffix = 'Stopgrad' if key == \"kalmanSSL_figure8\" else 'KNN Entropy'\n",
    "    ax[0].plot(pos_decoding[:, 0], color, label='Est. Pos. ' + suffix)\n",
    "    ax[0].fill_between(range(pos_decoding.shape[0]),\n",
    "                        (pos_decoding[:, 0] - 2 * np.sqrt(pos_decoding_cov[:, 0, 0])), \n",
    "                        (pos_decoding[:, 0] + 2 * np.sqrt(pos_decoding_cov[:, 0, 0])), \n",
    "                        color=color, alpha=0.3)\n",
    "    ax[1].plot(pos_decoding[:, 1], color, label='Est. Pos.' + suffix)\n",
    "    ax[1].fill_between(range(pos_decoding.shape[0]),\n",
    "                        (pos_decoding[:, 1] - 2 * np.sqrt(pos_decoding_cov[:, 1, 1])), \n",
    "                        (pos_decoding[:, 1] + 2 * np.sqrt(pos_decoding_cov[:, 1, 1])), \n",
    "                        color=color, alpha=0.3)\n",
    "    \n",
    "ax[0].legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "ax[0].set_ylim(-2,2)\n",
    "ax[1].set_ylim(-2,2)\n",
    "ax[0].set_xlim(0, 50)\n",
    "ax[1].set_xlim(0, 50)\n",
    "ax[1].set_xlabel('Time')\n",
    "ax[0].set_ylabel(r'x Position')\n",
    "ax[1].set_ylabel(r'y Position')\n",
    "ax[0].set_xticks([])\n",
    "ax[0].spines['top'].set_visible(False)\n",
    "ax[0].spines['right'].set_visible(False)\n",
    "ax[0].spines['bottom'].set_visible(False)\n",
    "ax[1].spines['top'].set_visible(False)\n",
    "ax[1].spines['right'].set_visible(False)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig(plot_folder + \"position_decoding_figure8.pdf\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "d3510ac6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model: kalmanSSL_inferred_sigmas, R2 : 0.959\n",
      "Model: kalmanSSL_figure8_procrustes, R2 : 0.991\n",
      "Model: kalmanSSL_figure8, R2 : 0.990\n",
      "Model: square_kalmanSSL_procrustes, Error: 'MLPDecoder' object has no attribute 'linear'\n",
      "Model: square_kalmanSSL, R2 : 0.967\n"
     ]
    }
   ],
   "source": [
    "# r2 score of position decoding from latents of all models\n",
    "from sklearn.metrics import r2_score\n",
    "\n",
    "for model_name, tpl in model_examples.items():\n",
    "    try:\n",
    "        data_i, labels_i, latents_i, decodings_i = tpl\n",
    "        data, labels, latents, decodings, model, cfg = model_dict[model_name]\n",
    "        b_z = model.encoder.b_z.data.cpu().numpy()\n",
    "        estimates = latents_i[\"estimates\"].cpu().numpy() - b_z\n",
    "        pos_inferred = labels_i[\"pos\"].cpu().numpy()\n",
    "        # plot decodings of labels\n",
    "        pos_decoding_matrix = model.decoder.decoders[\"pos\"].linear.weight.data.cpu().numpy()\n",
    "        bias = model.decoder.decoders[\"pos\"].linear.bias.data.cpu().numpy()\n",
    "        pos_decoding = (pos_decoding_matrix @ estimates.T).T\n",
    "        r2 = r2_score(pos_inferred, pos_decoding)\n",
    "        print(f\"Model: {model_name}, R2 : {r2:.3f}\")\n",
    "    except Exception as e:\n",
    "        print(f\"Model: {model_name}, Error: {e}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "79de53a6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 130x70 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot learning curves from wandb\n",
    "import wandb\n",
    "import pandas as pd\n",
    "\n",
    "# get wandb api key\n",
    "wandb.login()\n",
    "api = wandb.Api()\n",
    "\n",
    "def get_cos_sim(model_name):\n",
    "    # get the project name\n",
    "    project_name = \"probSSL\"\n",
    "    folder_name = \"../trained_models/\" + model_name + \"/\"\n",
    "    # get name of the most recent model by folder creation time\n",
    "    name = sorted(os.listdir(folder_name), key=lambda x: os.path.getctime(os.path.join(folder_name, x)))[-1]\n",
    "    run = api.run(f\"{project_name}/{name}\")\n",
    "    metrics = run.history()\n",
    "    cos_sim = metrics[\"grad_cos_sim\"]\n",
    "    # remove nan values\n",
    "    cos_sim = [x for x in cos_sim if not pd.isna(x)]\n",
    "    return cos_sim\n",
    "\n",
    "cos_sim_proc = get_cos_sim(\"kalmanSSL_figure8_procrustes\")\n",
    "cos_sim_stop = get_cos_sim(\"kalmanSSL_figure8\")\n",
    "\n",
    "plt.figure(figsize=(1.3,0.7))\n",
    "plt.plot(cos_sim_proc, label='Trained with KNN entropy', color='lightgreen')\n",
    "plt.plot(cos_sim_stop, label='Trained with Stopgrad', color='darkgreen')\n",
    "# remove spines\n",
    "plt.gca().spines['top'].set_visible(False)\n",
    "plt.gca().spines['right'].set_visible(False)\n",
    "plt.xlabel(\"Epoch\")\n",
    "plt.ylabel(\"Grad. Cos. Sim.\")\n",
    "plt.yticks(ticks=[0, 0.5, 1.0], labels=[0, 0.5, 1.0])\n",
    "# legend outside\n",
    "plt.legend(loc='upper left', bbox_to_anchor=(1, 1), frameon=False)\n",
    "plt.savefig(plot_folder + \"grad_cos_sim_figure8.pdf\", bbox_inches='tight')\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "psl",
   "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
