{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/usr/local/lib64/python3.6/site-packages/h5py/__init__.py:36: FutureWarning: Conversion of the second argument of issubdtype from `float` to `np.floating` is deprecated. In future, it will be treated as `np.float64 == np.dtype(float).type`.\n",
      "  from ._conv import register_converters as _register_converters\n",
      "Using TensorFlow backend.\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "from utils import *\n",
    "\n",
    "import keras.utils\n",
    "from keras import backend as K\n",
    "from keras.datasets import mnist\n",
    "from keras.models import save_model,load_model,Model\n",
    "from keras.optimizers import Adam, RMSprop\n",
    "from keras.layers import Conv2D, UpSampling2D, AveragePooling2D, MaxPooling2D, Dense,Input, Dropout\n",
    "from keras.layers import LeakyReLU,Reshape,BatchNormalization, Flatten\n",
    "from keras.models import save_model\n",
    "from tempfile import TemporaryFile\n",
    "import tensorflow as tf\n",
    "import pickle\n",
    "import scipy.io as sio\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython import display\n",
    "import time\n",
    "from sklearn.mixture import GaussianMixture\n",
    "import umap\n",
    "from copy import deepcopy\n",
    "\n",
    "import numpy as np\n",
    "from utils import *\n",
    "from keras.datasets import fashion_mnist\n",
    "\n",
    "\n",
    "import keras.utils\n",
    "from keras import backend as K\n",
    "from keras.datasets import mnist\n",
    "from keras.models import save_model,load_model,Model\n",
    "from keras.optimizers import Adam, RMSprop\n",
    "from keras.layers import Conv2D, UpSampling2D, AveragePooling2D, MaxPooling2D, Dense,Input, Dropout\n",
    "from keras.layers import LeakyReLU,Reshape,BatchNormalization, Flatten\n",
    "from keras.models import save_model\n",
    "from tempfile import TemporaryFile\n",
    "import tensorflow as tf\n",
    "import pickle\n",
    "import scipy.io as sio\n",
    "import matplotlib.pyplot as plt\n",
    "from IPython import display\n",
    "import time\n",
    "from sklearn.mixture import GaussianMixture\n",
    "import umap\n",
    "import scipy.io as sio\n",
    "from copy import deepcopy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# import os\n",
    "# os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"0\"\n",
    "# from keras import backend as K\n",
    "# config = tf.ConfigProto()\n",
    "# config.gpu_options.allow_growth = True\n",
    "# K.tensorflow_backend.set_session(tf.Session(config=config))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ! pip install umap-learn\n",
    "# ! pip install keras\n",
    "# ! pip install matplotlib\n",
    "# ! pip install np_utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "zdim=64 # This is the dimension of intermediate latent variable \n",
    "epochs = 100   * 100\n",
    "epochstep = epochs/100\n",
    "nofclasses=10\n",
    "batchsize=1000 \n",
    "nofprojections = 100"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Data Generation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Read MNIST\n",
    "\n",
    "(x_train,y_train),(x_test,y_test)=fashion_mnist.load_data()\n",
    "dataX_train=np.expand_dims(x_train.astype('float32')/255.,3)\n",
    "dataX_test=np.expand_dims(x_test.astype('float32')/255.,3)\n",
    "\n",
    "labelX_train = keras.utils.to_categorical(y_train, nofclasses)\n",
    "labelX_test = keras.utils.to_categorical(y_test, nofclasses)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "numdataTrain = labelX_train.shape[0]\n",
    "numdataTest = labelX_test.shape[0]\n",
    "\n",
    "dataX_train1 = np.reshape(dataX_train,[numdataTrain,28*28]) \n",
    "dataX_test1 = np.reshape(dataX_test,[numdataTest,28*28]) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.image.AxesImage at 0x7fb332cb1d30>"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "iii = 55110\n",
    "trainim = np.reshape(dataX_train1[iii,:],(28,28))\n",
    "plt.imshow(trainim)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "##  X is input and Z the embedding space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    " \n",
    "imgX = Input(shape=(28, 28, 1), name=\"input_img\")  # adapt this if using `channels_first` image data format\n",
    "labelX=K.placeholder(shape=(None,nofclasses),dtype='float32') #labels of input images oneHot\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Encoder, Decoder and Classifier NN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def encoderXNN(imgX):\n",
    "\n",
    "\n",
    "\n",
    "    bn_model = 0\n",
    "    p_activation = \"relu\"\n",
    "\n",
    "    dimS = 2\n",
    "    x = Conv2D(16, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(imgX)\n",
    "    x = Conv2D(16, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = Conv2D(16, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = MaxPooling2D((dimS, dimS), padding='same')(x)\n",
    "    x = Dropout(0.3)(x)\n",
    "    x = Conv2D(32, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = Conv2D(32, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = MaxPooling2D((dimS, dimS), padding='same')(x)\n",
    "    x = Dropout(0.3)(x)\n",
    "    x = Conv2D(16, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "\n",
    "    x = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = MaxPooling2D((dimS, dimS), padding='same')(x)\n",
    "    x = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = Conv2D(64, (3, 3), activation='sigmoid', padding='same',kernel_initializer='random_uniform')(x)\n",
    "    x = MaxPooling2D((dimS, dimS), padding='same')(x)\n",
    "    x = Dropout(0.3)(x)\n",
    "    embed = MaxPooling2D((dimS, dimS), padding='same')(x)\n",
    "\n",
    "    #embed = Conv2D(128, (3, 3), activation='relu', padding='same',kernel_initializer='random_uniform')(x)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    encoderX = Model(imgX, embed)\n",
    "\n",
    "    return encoderX\n",
    "    #encoderX.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "def decoderXNN(yin):\n",
    "\n",
    "\n",
    "    xx = UpSampling2D((4, 4))(yin)\n",
    "\n",
    "    xx = Conv2D(128, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    #encoded = UpSampling2D((2, 2))(yin)\n",
    "\n",
    "    xx = Dropout(0.3)(xx)\n",
    "    #x = UpSampling2D((2, 2))(x)\n",
    "    xx = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Conv2D(64, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Dropout(0.3)(xx)\n",
    "    xx = UpSampling2D((2, 2))(xx)\n",
    "    xx = Conv2D(32, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Conv2D(32, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Dropout(0.3)(xx)\n",
    "    xx = UpSampling2D((2, 2))(xx)\n",
    "    xx = Conv2D(16, (3, 3), activation='relu', padding='same',kernel_initializer='random_normal')(xx)\n",
    "    xx = Conv2D(16, (3, 3), activation='sigmoid')(xx)\n",
    "    xx = UpSampling2D((2, 2))(xx)\n",
    "    decoded = Conv2D(1, (2, 2), activation='sigmoid', padding='same',name=\"decoded\")(xx)\n",
    "\n",
    "    decoderX = Model(yin, decoded)\n",
    "    \n",
    "    return decoderX\n",
    "\n",
    "    #decoderX.summary()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# def classifierNN(nofclasses,z):\n",
    "#     probX=Dense(nofclasses,activation='softmax')(z)\n",
    "#     classifier=Model(inputs=[z],outputs=[probX])\n",
    "#     return classifier\n",
    "\n",
    "\n",
    "def classifierNN(nofclasses,yin):\n",
    "    interdimX = 32\n",
    "    y=Flatten()(yin)\n",
    "    y=LeakyReLU(alpha=0.2)(y)\n",
    "    y=Dense(interdimX)(y)\n",
    "    #y=LeakyReLU(alpha=0.2)(y)\n",
    "    #y=BatchNormalization(momentum=0.8)(y)\n",
    "   # y=Dense(zdim)(y)\n",
    "    y=LeakyReLU(alpha=0.2)(y)\n",
    "\n",
    "    probX=Dense(nofclasses,activation='softmax')(y)\n",
    "    classifier=Model(inputs=[yin],outputs=[probX])\n",
    "    return classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING: Logging before flag parsing goes to stderr.\n",
      "W0206 06:17:09.246687 140408956434240 deprecation.py:506] From /usr/local/lib64/python3.6/site-packages/tensorflow_core/python/ops/resource_variable_ops.py:1630: calling BaseResourceVariable.__init__ (from tensorflow.python.ops.resource_variable_ops) with constraint is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "If using Keras pass *_constraint arguments to layers.\n",
      "W0206 06:17:09.383743 140408956434240 module_wrapper.py:139] From /nas/home/mrostami/.local/lib/python3.6/site-packages/keras/backend/tensorflow_backend.py:4070: The name tf.nn.max_pool is deprecated. Please use tf.nn.max_pool2d instead.\n",
      "\n"
     ]
    }
   ],
   "source": [
    "encoderX = encoderXNN(imgX)\n",
    "yin =  Input(shape=(1,1, 64) )\n",
    "\n",
    "decoderX = decoderXNN(yin)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "def step_decay(epoch):\n",
    "   initial_lrate = 1e-4\n",
    "   drop = 0.5\n",
    "   epochs_drop = 10.0\n",
    "   lrate = initial_lrate * math.pow(drop,  \n",
    "           math.floor((1+epoch)/epochs_drop))\n",
    "   return lrate"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training on Task 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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jOOBs81sNfFmupOBOej6AsFIrBYBgxKjFbKId5yb53FMkrx5QM2or0uB5+od/0Vi5Ni3gfJEc9JC4HQz65oYXXY6sDBxbqOZfTvRxv+Les8lLSDx//5X4FPn3Kx6p3SNfcLMeppZl8R2O+A04n3p/gWcmyi2/TWuufbnAq8kai3nARckRpUNAAYRH5ig93UPOefGOOf2cc71J/5//4pz7mJ0feO87j3fEl/ERxUtJ97GP87rePS10+v1OFAAVBNvW/8f4u3xLwOfA/8EuiXKGnBP4tp+DAyt9zy/It4ZcgXwy3rHhwJLEo+5m8REufpqsev7HeLNfB8BHyS+fqBr7J8v4FDg/cQ1XgJMSBzfj/iNcwXwNJCdOJ6T2F+ROL9fvecal7iOy6g3ulPv663jCziRulGBur7OaeZ1ERERkVQJQlOgiIiISItQYiUiIiKSIkqsRERERFJEiZWIiIhIiiixEhEREUkRJVYiIiIiKaLESkRERCRFlFiJiIiIpIgSKxEREZEUUWIlIiIikiJKrERERERSRImViIiISIoosRIRERFJESVWIiIiIimixEpEREQkRZRYiYiIiKSIEisRERGRFFFiJSIiIpIiSqxEREREUkSJlYiIiEiKKLESERERSRElViIiIiIposRKREREJEWUWImIiIikiBIrERERkRRRYiUiIiKSIkqsRERERFJEiZWIiIhIiiixEhEREUkRJVYiIiIiKaLESkRERCRFlFiJiIiIpIgSKxEREZEUUWIlIiIikiIZXgcA0KNHD9e/f3+vwxARERHZqXfffXedc65nU+daRWLVv39/Fi9e7HUYIiIiIjtlZl82d05NgSIiIiIposRKREREJEWUWImIiIikiBIrERERkRTZaWJlZvebWbGZLal3rJuZvWRmnye+d00cNzO708xWmNlHZnZkOoMXERERaU12pcbqQeCM7Y7lAy875wYBLyf2Ac4EBiW+LgfuTU2YIiIiIq3fThMr59xrQOl2h4cDDyW2HwLOqXf8YRf3FtDFzPJSFayIiIhIa7a7fax6O+cKEtuFQO/E9t7Amnrl1iaOScBFYw7nXMqez0VT91wiIiKpsscThDrnnJklfZczs8uJNxfSr1+/PQ2jzXORCJbR8HJcdPYwBh32XaoqK5k8424yQ9bgfFVlhJuu+TkTbvsr2e3bA3DWjwdz2BHnQcFKpt07lx/9eDBHDDoBgLW9B7M+I5eBmwv45r/zyTjlEj61PgyihAFFH5CZ2Yny9t2IhDKoyMghJ1rF6uweOIzObivZLsKm1//IP579hHEjLuSLgd9lBT2IxMLslVFOSawD+1g5YWJ8Hu1J+3AVnUOVDCv+mLd7HsKqLT3o3m4TJ379H/6193foywaGFC+nOjubVZ32YWksj2isLtcPh2J0CFeRY9WURtsTiYbZL2s9vavLeS0yiHAoRsgcWeEIh8fWcv+1V6f5KomIiOyY7Uotgpn1B+Y7576V2F8GnOicK0g09b3qnDvQzP6U2H58+3I7ev6hQ4e6tjjz+rgRFxLtvR8ZWzZQ2q0f7UuXMnP6QzjnGH/TGN7pehB9q0vJ21RMJCOLFbl7sY4O9IuWkrvsZT4dci7d3GYyXYwPq/emV3YFUUL0dhtZHunFxvXZmANncGifYuDZfQAAIABJREFUQrKIsJFsesU2URDqxMr1PaDK0bFHFWdu+4AlHffls7LeuEogA8I5jui2EFQ7nEFN+usMwu0gtgVcCCwGLgzE6so0Z7+89WyM5bBufQeIABkWr/escrXPBUCWQdRBFMgGtkGoA8Q2131vJNsIZ8Zqd2NRI7Yt8ZzZYCFwW+Pxh7IhnOlwDiJbjJzcKMvGnL1nF1RERGQXmNm7zrmhTZ7bzcTqNmC9c266meUD3Zxz15vZWcBI4AfAMcCdzrlhO3v+1phYjbluBBs3rcH6H8eGrA4UhXIpqOqMAwzolrWFtRu7NEgQwh0d7bKrccCWzVnxBKcZlkPD81kWT04SCZALQV6vCrqEtvLl1m5sLWtcudi+WzWdsrZSUNSpNiHK7hJl/3brKIrmsi2aQU5GNZ1DlVQRZlB1Md2ryvm/7EPYWpXFETlr6Fv0NqU9DmdJxl5kWZR9Y6VkxKLkRirZEs6iz6ZiMiLb2NquEy9kHcbm9Vk4g249t3IYa3GvP8mm0BYGHvojMoq/YMWgE4hYiI7/foiHnn2DK26bycLyg8juGOWnXz7JgoE/obwkh+69tnAwBZSEO5JJlD7V5WR+8V/uuffvDV7jhecdw6ABhzHulrv43/NPYdl3RrBua0cuqHyNyTfdBsDgGc8Ri4VYPuZHu3OpRUREkrJHiZWZPQ6cCPQAioCJwD+Ap4B+wJfAz5xzpWZmwN3ERxFuAX7pnNtpxuRlYrV1UwXtOuYyavpUlrbbm8LqTmzalI2rpDaJqhFqD6GQwzkjshks2ziwWxHbyKQTW/lo/d5YqO7/88z2S6gMZbE+syMZLsrelaV03LCKT/KG8WHxXvTttYEQjs2xLE4vegmAL5d+SI/jhtOxciNTJ80AYGz+b3i1z0n0dRs4cN0KNnbsTk71Vm7Jn4BlZfG76dNYk9ODfbaVcuf1NxC/DM1zzuG2RQnlJNcSPGHcaBb2OIZhVau464b8nT8gYeLYq6kuK2bavXOZeOM1LOpxMEcWvM+U6Xcl9fMhHvu2LZvJ6dCx9thBM54jEg3x+VglViIikn57XGOVbl4lVpfePofXNg0kFHZEN8WTEWsHuR230SNjE9WE2Su2kbzKUjpsq0t0IJ5kRKIVTJv+59pjp/5oAGBgRiwW45X5q5r92SOuPJu77nmOUMiIxCJkhFrFetht0kG3PkckosRKRERaxo4Sq8DezX976wxeLfsW4faQkRFjQF4Z3163hCkTb9ulx0+eOqvRsX8+33witb177p1Xu62kas8Y8VpEERERrwX2jv56aBCWaVxS+U8m39g4SZK2w8zF221FREQ8Fsi1AsdcN4KNpdn07FrRZM2TtC2G8ioREWkdAplYrc0bjMXgsOjXXociqaDMSkREWolANgV+Qh/+f3t3Hh9XXe9//PU5s2Rtky5pmi7QFkoBRZRduXBV1Kug4vZDXLmI1w21rFJKgba0pQgWUBFQ8YLKdQG5131FQXABy6rsUOhC0zRt2iTNPnO+vz9myaQLlGamk3zP+/lQMplMMp/k23PO53y+G3Fj0nMPlDsUKQLbYf6mlINzbocZqRctvADXso5l19+af27+Zz9C3/Q5JDs2cvnyr0MYQjB4j3fRmR/i+f2PJSRgxpMPg0HPjBnUd21m0WVf2Wu/j4jInohkYtWTSpKoTrNsya0v/2IZ8cxgBExuHbGcc8y/8AuZJCY9ALHEDl93LmThxecTJpIsWbg8/7WLFl6AYXStXUPVPvvQWVFHOgjYkKxnrasnZo63tP6drk1rWHnwB5iZ3sSUbRtZPWYys9rX8v3+4wkmw8prb2ZCuI1VNLBh3IexjsxabT+45FfEEo5ZY1vpdJX0hzHaxn8MWjIN+vCsqaTTAQNbA2Y2bb9lqYjIyBPJxCqdNoJAV2JfWAT6AcPuAYLqwYTocx9/N+lDjmfC5qcIYjX0VdeTWPcMy66/lfOXLmRz5Vg2x8cwtbeN9mQN9wbvYONXVvBIMA2Aw9Jr6IxV0mNJHu9pIjUQkA5PgD749YrbqI4N0LxtLKnezHZINAE92f/nxMGl4X+q3kpyVore5jhrgjqC5H64FljbNB5rzqz99nRzA9CAM5jQ0M3MWBvr3VgGXIwtXdU809wAMSBmjKnr5fBgDeti43h2cwPxisxq/L0ukqcrERllInmmCsOAZCJV7jCkiEZjxSoMQ740/3R6t2zl6zf+lAvnfZ5/NB3Ja7pWs2L+JfnXXfKlT/E/de9iTFUfx/U/zbZ4FfcfcjrbNidJjJ3DQG8MOh1MPIrfX/0jWjuPhM7M9z5sU7CEYaHjzi1zIJUZjvY7Dsr//KAGxtT0UlfXS3eYoLW1BkKoqAuZ3bSJkICOsJLJQQf16W4SYZoJfe30PXM/va9+M3/sO5Dejjj7NG1lzeZxhP2Zxli7oY6gEt675jYqpzXRnahhTM9WFp591ZAuw3M++W6qpx/MwPpNXHHjt4f8jbq6uqipqWHGRb9UZ6+IjArRTKzSRrwyfPkXyqgQvPI9wEtiwaJ5OAdLLr2ccy+/jHur53Bc19NsqB7HhL4Oaja/SHPTbFpiY6l1fTzSM40+TiFogvYVX+Uv1e8gbIZnKybyr6/ezICLcfCTP+fZg04ktT6gbVsVP3OHApn9Ehsau2jdWENQ6XhN03oe62yitaWWuoZe3hCuonqgh/9NH07Y5TKvbakhWRfy3vQ/aK0cR31/J2D0PnEP3/je7/K/x9zT34sl4apldxAPXj6dcS7krE9+nOWLv8ully8kVZngl8Gh9G2NMbGui6sW3vKS37/i2z/b5ddqamoG3+dlI5FyuXT+2ewzdj/OmPf5Hb+Y6od4coennXOsuPwyHq+o5Zl4I42ugznrH2HJl2/YCxGLlE4kEysXQtzS5Q5DimkvXnUXzDuTsH0ry66/lTNOOZ6KI07iqYomnu09DnNw68Jf4wYOxzrgJxyWrx5RcShsLPhBFbBP01bWto7jno37QQW8pqmZR1qaeHp9AwDr9/0ofRvjVI0b4J2ph9hQNYHadC8NW5tZdNZXuGDppSR7Olgy9xrOm3canRMOIfbQb/jG9/8AQNWieaxtaKLmwZ+z+ogPcUj7KpZfsvQlf79r//t/X/Lr2zMLuPam7wPw5QWLAXj22m/x6NYpHOg2vKKftes3UWK1M93d3VRWVhIEAdfMX0ZrsoOlBWPknHNcsPRSVtc2MXvDv1iy/DpSqRTx+I6n/tO+ci0vxus5dtOj9FTUEN+8mq7GA6jq72H5JUsIneOS889k/eTZVLgU4zc8hgUV/HXasTzn3kKVpfjGituIW8hxPU+wqbKeqc3/4ldNb2dbbwXJRAoDYkHIW3of5/7aWaztPQI6Mi27hnqqG/v31p9OpGQiuaXNjIt/xaQJndx/1gf32ntK6Ry24ja2tFfz/KKTivpzv7h8Gb3xJLX93QBsS1bzQLAvm1qriVXCW6ufoC1Rwz827ANA7YR+9kluYX16LAD/3v8UD1XNYFq4hbXBOLalKzjMraGpcyPtVfXUblnH5cu/zlmXL+UfVTN50+aHWHLZlcxdvpSQgOeqGnl882RiyZCTgweHdA+OdBed+SGe3f94Jj/8f1x7y2+H/fNmLPglkyd28vezTi1CdCOLc45LL/g0i7/8TRac/Vlap+zH2P5OrrxoEfO+eBob93ktUzY9y5Ll1w35vvmLL+ZHwdFUVgzw2viLPDQwje62BFMnt/PG1b8jJOCP+76Nlg1jgMzG75adfDlhzDaO6VvFY9VTmZzu4LGwifbWykw8ZFcwCcDCTHW0fmIvnd2VhP0GA9lrRtIIEo6wC6rGp+hpzyRrLiS/KXysNrNdWGJsSBhmqp/pXsOlMu9RO7GfV8U3MKmvnZ9vOYQ3ND7P/5y9k6qXyAijvQK3s+/8XzJtUjt/OevDe+09pXQOX3Ebm9ureWGYidXnPn0yExr3Y3PLc2w58K38rWXmDq9xATQ0dLG5owYXQlVNit7uOB9I3E/bv/7CTT/4wy5//s6WI3g5vdsGqKxNvPwLPTfj4l/ROKGD+0ZRYnXJlz7F6sZXEXMhjS3PkkhW8NzEWUx77j7+OftNrE/V0RDvZGtYxcaNtew7eSsvdtST7sps+H4Kf+X22OtJdRpUGP9R+zg3nn8eFy2cxx/qX0vL5rGZwYUOSGf+bdbUD9DdlsAZEIClYfLkTg4IN/K33pkk4mliQUjH5goKe9BdADMbt3BQfzP/TE5lPF1sdGOYYF280D+Brq0Jaur7ScbSHGQbMODh1FT6+hMcXf0Cb+43ntvyCAC9Y6fxZM00OqhkXXMd1eMHmBdv4ePnfBrIJIT/W3EE+1du4udzzwDgnGWLuaPjcF7f+Dw/UGIlo4D2Cixw6klHYYdcSkJdgd7IbGnzyhKWBfO+QNv4aXztvC9xwfJFPFQzi1VNn8L1AuNOwLVkBm8fl3yGniAzPiTpUkzb+BxLzv4aH1/xNf68cRY9fXGmTO7gyrMW7Uacr3z4tZKqQW4EDV8PXSYaM2P+5z5Cv4vz+IH/zgAB41wPz7vxtCZPhtbM6xPj9ieZSNHVksQmzsQ1AxWwtS9TJYrVwJrmeqwKGhq30dpSy4+q3oDrhoObWniyvZHfbTuIj179Ne5NH4/b6Kip7+ftqUep7u+mfWw9dR0bmLh1Eg9PqubZ2CTSzpjpNvM/Z2USlbA/jSUCzIxzll3GEzVTObT9OdqqxzF5WwuL5+58n1TnQq5buJjPn79wx79DTw9BVdUu/06fuuoqJm1Zx8eXXpN/btkll7F4YIB4YvDftuWP35HTxiJ7KnKJ1f6HHM3fgaQSK28Yu7dX4IJ5X+CJplezT1cr9zYeR2tLDd3XXMufOo/EOqCiPs3M8Zvpc3FCjOM3P8hlF+58y6MxT/0ByyZir+tbXeTfSHamnMX1L332w2za70g645XMaX2Kv096Hd0uQV3Qy+N1HyaoJJMs5WINYNzEHg53a9gQr+Ox5skMkKSiPk1ve4w5U1r53Rf/k09feRUd8SqmP7OStlmHMGXjk6S39HHr2A/iehxzmjbyq7mf4MwrlvPLLYdwb8ssYjUh74ut5MovLQTes9u/Q5CM5R+vmH/xbn+fWcDnFy3c+c98iaQK4JvnnbfT5wuTqsx7ZM7HoRIr8UDkEqugohL6IOmUWPlidwpBFy28kFuT74BmxwPJ6bjOTMXhT5tnA/D/xtzPlfO2rzp9dJc/77obf8rJ136LJ3sa6X3g18C8PY5fdoPl/1NyCy4+n0cnHsDsrmb6gwSY8ef9358fg/SPin2hOZPlraeO5NjMDOOD6zdwyNbn6ElUU9GxlmXnfAuAZefNp3nKWDp7K/lA8+3gHEu/+D0Abjx/54nHo9fexDO9Ezni+buA07nugnk8de0tbHMVvOnFe7j8yut2+n2jVb5l96CqKzLSRC6x6gsqAKhwA2WORIplVxWreYsXkIrHaXn4DzxwxHm4rY4jJq9l5cbpQHbAbVuc+kk9XHnOy3flbe8nn/0EQTwgCN493F9BdkOxC1aXfOEsNk1r4onKKRzV+SxXLFjEx1d8lT+7N0IzPMqUIa+f1bSZaemt/LltP2on9HNgooUX3HhObP4DC6+4AeccsSC2w/vMv2oZnx1IccW5n2HpV7+7W7Hd9vnTSQZGYCfnn/v9Fz6GBQEwesaZ7bZsObL8I35Fhi9yiVUqnkmsVLHyR2bl9aGn5EsWnMMPYydgDt589Fi6WxMcNGUjP/niZ/jkVSvoiFexT3cLd9QcyQk9T+zR+8aTO15EpUSKsFZZeutWgpoaLrr4c7ROPJC/TH8j3W2ZLqmtkw6h9aqruXvzASRqQ96WeJxVlZMYE/YSuMw8uR/N/SwAF55/JpXrElx67TXZCQmnZGPc9XuPS8RZ/tVv7/oF26mMBTs8Z8GOz/kitxGGEivxQeQSq/5Y5ldOhlp53RcGQ87IHz3lWB4+6jzYlHn6nuT+EINXr78LOJ1vn3dO/rVX9XdBsrjLNEhpuFc4QSFn/rz/4oEpx/JU6yQs5giD92CbgBjMadpIc6qOLW1V3FVxALFq+EjvXSy6aNebPRd2w+3JhATZmcwBrDFW4gN/b4F2oT+7Aa26Av1huCEDmzuP/U+2bUoyZXIHAP3tMerG93LV8p2sAJ6s2fE5GXFsDxcI/eRVK/hB5Xt4an0DVbUDjB3TS9OkTv694Vk+mv4Nv517Oq+1dZCCsAteN2Ytiy7bdVIlpWFhdicMJarigchVrPpyFau0Kla+KNx15dNXXsXDWw6ielyKOfdex8bXzSPVaRwUtJQvQCmKV7rcwvuvvZGVm+cQq4a3THySb5137k5fN23jPyG2HwTGnA0PFCNUeYWCbNuOgGUVRYYtchWrPsslVto6wRe5wesXLLmU32w5iFgVvL/nb/z37fcys3YzxGH/lsfKHabsRfMWL2BlyzQq69J8uOtXu0yqAJYt/xaHT1rH8eOeY9nyb+3FKCUnJFOxUl4lPohcxao/yHQFJlNKrHyRq2OsqWnAtsG7Eo+w5OLMfmmzH7+D10/cj8XL/ZqeHjmvoCvwojPP5P/2eRvWB+/v++sOW8HszE/mfnp48cmwxMJcYqWuwNHm7jv/wC//fi/vP/p1HP2Wkwl7ewkqs9sjpUKIGZd+eTHr4mNZdMp7mT59RtHe2w2k+cNT/8KlU7zt0MPpSKVJZk8UlYnyTS6KXmJF5o8d1+B1b+QqVmG2AJsMB5Pmb9z403KFJUX2coPX58/7L1Y3HsrWAw6jtznOIU0bWDZ3+Ut+j4wMLjvr02mM1bCl02mcc8TjcZxz3HPX35hQX8V3fv1zJqa6ufCSwWNi5X13c+VfH6cy7OeDs6dwwOxXc/OPv8uWyjr+ktif7r4KwrSRTKY4aeBh/jlmXzamatnWV0EynmZcsoe1rXWQOpKfrkxT9ejt9A4kOCqxhoA0f+2bxaTqbazrPgL64O6bH+MrRz/Ge942vAlDqf40d9z3N+bfs4VUR4ALYMLdv6atxUHacdGJB/Ffx88a7p9yj0Uvscp2BdLbWd5ApGgse5+bPylroIZ3Xu56e96nT+E3+3+IbS3Z7YfqQuY8/lvgjNIHJ8NmWsfqJV208EKer5vK1J5NXDl/IYQhBAEfX/FVJva2kw4C/la5PwEhLe1jcSlj0oRO6qyXp1smYsEWSB0Ocbhp6S8wc4yv7qKzv5Lutswm8ndvNeyhVbief4MeIA6VY1Ik4o6urQluSx8JnUCFUVGZYltHBdsGkoyZ2EdDYhvPbZhAr1VhCfhzXzapSRjr2usgDuMm9bBlYxVP/OvRPUqsvv79m7mjLcaL3XX0dsSwEEjEOHT/Dl7orKZtI+wzYwz/se8kjpw5vlh/+j0SvcTKZSpWW9rXljkSKZbcNTedfRQosYqcR191ItvWJ9m3aQshxrHtT7D8xh+XOyzZTYPLVvhbsXLOccNXL2dtew+JgW4WLr4KM2PRgrmsr5vOfx52GGs3PM/3Njm6SPLarhcAeLBmJs/3/RvWAsRm8vS13+L5gQlUx/vZsHE/iAGB4doz572acQPEg5CNG2rZSC1V41LUJvuYErSzLl1PbypBOgxoaR2Dc3DglFYmpTp5jMn0pRPsW9/GmLCX13WvZd4FlwNw9rIl3F05hzcMPMfXL8jsMvHdW27k6bUvsuS8xQCce/liqsMBjnnN4fz0qWfpjFXyroZK7mluoybVxzOxyWyhao9ufH/+619z1TMNuD5IjHE0TepkbDzFZ+ZM4n1vPRXnHL2966iqmj7sdiqGyCVWA8RwAVx3/c/KHYoUiWXvc1OWSZpjoRIr79hLj79Z1TmRxNiQu+fuehsiGcHyC4SOrsTqD//3C/708L3MmjyJu7rjbIrXsk9fGw0dLdw54TDaemoYU9nL8T1P0Vw1jr+2HpqptAC/uuY2XuXWc3fibbjNcO+D7fQPTGKgIzOkYRUTMi/sgLqJfRzDKn7XcTCPNE/BBdARVlA1PkVfT5wwBafW3MeJxxzN8W96NxYLeNu1N7M5XcMX6zdx2ic+u0PsqYEBnn3iSea8+kQseOm/+9XzF2QXwx183cdPGzou8SsXXpJ/fNK7Bp//cPbje6/5JgDp3Tw/r31xHV/58e2kzfhN9yzcQMAnZq7n0k//1w6vNbMRk1RBFBMrF7zsPyIZXYL84oKZE5Jp/JyXdnWjO3/xxaS6j+GAKZv2bkBSNEF2J4wRe0vkXL4/urujn7/+4bf86smH+W3i1XT1HodbQz5heoJGSB6Ma3Ekx4Rs2lTDHRwGXZCoCXlV7QbWh3W0ttRwF7MJqmDy5A7WbxhLUAWHTV7HAR1rWTV2KgbM2LaB5ecuxMyYu3wZj0yYzls2P8zG6gnMTqTp7Oqk10IWL7hqSMi//eJppBwkdnG9iycSHPiaQ3b7TzD8xXBz5+mXd/41V/KTjoMJuzN7uRKHf2tczaWf/twwY9g7opdYhTEsNmIPX9kDgxWr3IGv9vXNS53SH6qbCd3w2o5Vey0eKa7c+qC7c9EthYsWzuOecYewdaCK2ngfXekk3X1JnDPClIHB/vWtHN/8IDfXvI3UtgA4DBfAPk3tdKcTzAo2cWDbM/x63DFs2lLLieMf5/rzz2f+onn8gONwA3BK+n6Wzb0MgPdfcwNpAo7rXsO5Zy3hskvO5i3/9i5ef8KuZ6heO29+9tHHXvZ3MjMSI6iGkL8B3o2uwF/0H0A4YBw5eS3VpHj/nBm8+x2jI6mCCCZWaRdg2uLNK7l8KlexiqF9IP3jdtlNtKp7IrEax5cXLN7LMUmxBPm9W4ufCfT395NMZiY1nL/0UhKpPpZdupwF887kucaDOKDteX4QfzPpZgiqoL2vgqACKipTBOaIV4b0DcR5en0Dq+veQqrdaGjcxlRr59VbV7Fku5mnlznHXT/8AW/80PkALLt0OYmLzyUVT7L0ksvzr/vJWZ8Z8n0XL7666L/7iJI9Ub9cT+B1t9xId9s0ZjS1cdsoXQYlcolVKgwIVLHySm5GUTo7Hd+FSqy8Y+y0EDl/0Tz6eo5j1pTNez0kKR4LSzMr8P3X3MAjXVN5dU0zoRmPdB2FGfx2xW1sTpwELY6/V87E9cCbJj3DzeecRX9/P4lEYkjX1+9/8jPOWtVP1+Yk8TGOZftW89b3fXAXv4zxxg99eMhT2iZp8DwdvkQrn/GVa3iA6RjwjuToPaYjt/J6OgwIAiVWPsmd/9K5MVbKq7yzi7yKB8fNAeDIjuf2ajxSXC47QKmY61h94JobWLlhOqmegIc3TOXR5ilUjAlJ1oZs2lRN7dg+pk5ux/VC5bgUp0zODH5OJpM7jCd66/vfzXu6/k5FfZq3Jp/gre97187eUl6C5bsCd2zjrVvaOOaaH3Bn62y2tlZSOS7FvM/O3dshFk3kKlbpdEAsVq6efCmFIH8nNLjwgnjG8v8Z4rltE4mPcVxx0aK9HpIUT1jEvQIvOf9MHp72Wh7ZMJ2qcSlO2fQ7tk6YSspiTGxbx+JlK7h66XzOPv9KnHN89JrrmNG+lhM/fMVL/tylC6/gsp4egqp3Dz/ICMrXM2zHRj7vO7ewofUAGhu38bG6Nt54zLF7N7gii1xiFYZGRUKJlU9su1mBoTbY9tL2F90LllzCwLajmTOltTwBSfGkBzIf97BilQ7TfO+Gb/PMxlXcWnUSNDsq69OcsuX3LL5yxy2Nzr7kyuzbGbee/fndfp+gqmqP4hPI1ZzTO6lYPWDTIQ5fe/10jjpmF12so0j0Equ0EQuUWPkk15+dO2DTKlh5aMe73JVj9odtcHjbU2WIR4preBWrd3ztezzdNo2KqiZch+NNk55hzpZ1XLj860WMUYZjcAnYoYnVb377S9o2V9EwsYujjhneVjcjReQSKxcacVNi5ZPBilXmgB2IbStnOFICtt0Coc45XugYT7IuZNmFl7/Ed8qokHYQe+WD18PQ8cmrr+ap1jlg0NcXY2bTFm6ee1ZJwpQ9Z9nFNNLbtfJ3H38eS8/kyPSacoRVEpFLrAghUGLlFduuxJxsV1eglwrOx/OWLiTddRRzmlrKF48UjYtljtlXsvL6gkXz+N/qo+naPIdYNZwa3stj42ZxSPMDgFbgH2lsJ48AVtt4nMEXTvyPvR1SyUQvsQKCnQyek9ErP3jdZToFN7SuLmc4UgKZitWgp2unwjZ4VZf2/PRB8ArXr7pk4fl8P3wTbotjdtMmjt30aMGSBjtueSLll2vhcLu2buurIVHjOPjg3V8FfqSLXmLlXL7CIX7YvivwO7ffW85wZC94IRwPCSOxVhUrH5gDbMeL7q78YuxRuFbHe8c+zDVzLyptcFIUufP09mPXe7oSjK/vKkNEpRO5daxcOLi0vvihcH2UnUw4ES84XEHjbu2sYkxdL8uuv76MMUmxxCx7j78bx+8XrlhO28Zqpk9u55oLlVSNFrnlFgonKFx34zeg39EY6yxPUCUSucQKIKaLr1eCl1h4TvxQOAt//qJ5uB6YltxavoCkqOrGNwIMSZ535qsrlvOb9Ksgabyx+S97IzQpEpevWA228UPb+gDYZ2BLWWIqleglVg51BXpm8E7ISrHVmIwwq8buA8AB3RvKHIkUS2P1VODlZwXeHdQz0BFw+IS1LFm+4/pUMnLlxsIW3iWtiY8D4PWTxpcjpJIZVmJlZmeb2WNm9i8z+4GZVZrZTDO7z8yeNbMfmVmyWMEO1/HvnIk5dQX6p6BipcTKX9nDdq3V4wxqVj9W3nikaD70+U8ALz0rcMG8M3mgbR8SY0P+PeVXhSMKzHLn6cHnWlJjsEr4z4+fXqaoSmOPEyszmwp8ETjCOfdqIAacClwBXO2c2x/YApxRjECL4ZDpmVkHWm7BL0F2fZTQ2Z4u3CwjnJnLX3Q399WQqA1Zdv2tZY5Kiunlbnf/3PQG6Hf8R+wxvnjuvL1XEUg2AAAd20lEQVQSkxRfWHCS7txWSW1tXxmjKY3hdgXGgSoziwPVQDPwZuD27NdvAd4zzPcomqq6TNkxKMaGVDJi5A7TlxufIaPf/M9+ht7OBPXVPeUORYrNdl2x+vEN32VNWz01E/r5+gVKqkajWK5Ulf3wvZu/TdgDjQm/Bq7DMBIr59yLwFXAGjIJVTvwALDVOZdboXEdMHW4QRZLIl0LqCvQN7n2zIyxUnLlIyM7m6ipAVKOKUF7uUOSEtjVmfmvW9ZCH8xJbNyr8UgRZU/NYfbjPZs6AJiRaitTQKUznK7AccDJwExgClADvP0VfP+nzGylma1sbd07m6jGkglAC4T6Zsg0XuVVfsq26+oxmdlj+/ZsKmMwUjo7P4AfrZ4OwMFbn9+bwUgR2XaD159PZAasHzu+tlwhlcxwugLfAjzvnGt1zg0AdwDHAvXZrkGAacCLO/tm59w3nXNHOOeOaGhoGEYYuy/I/rqaFegXc5kxVpoV6K/cMftiUI8DxmxdVd6ApPhs6MDmQmu7xpEYG7Jk4fK9G5MUzeDK69kdMgbGYlVw+un+rZQ/nMRqDXCMmVWbmQEnAI8DfwI+kH3NacBPhxdi8bhY5mNMY6y8kpttksmv1La+cs7oczEsBsuWf6vc4UgJ7GyY5CULzqG/I2BqjdYtG83yK69nP+/pS1JVNVC+gEpoOGOs7iMzSP1B4J/Zn/VN4ALgHDN7FpgA3FSEOIvCxTIrP+RmkYkfgnBwjJWGWPkp164OVJX0lbHTzOrpCftjwKt71+/1kKR4cl2BuQVC0ykjGUuXM6SSGdZegc65S4FLt3t6FXDUcH5uqeRmjWlWoGdyFSuNsfKYK8iq1Mg+MnZeb37aNUDcmNqtcXWj2eBRmz1fp4xkLLWrl49qkVp53QWZPDKmipVX8l27TpdcX+UuuhpH5zHbMbH60Tduoa29mrH1vVy46MqyhCXFkV9vkIC2zZtxKUeFKbEa9VJB5oys87Jf8n33oS663spN1S5vFFJyQw/g+7auhj7YL65q1ahX0BV43/33Yg4q0RirUc8FmV83pgHOXsmvnqGuQG8ZLjv+Rsmzv3acr/1YbWZfyIPaNQt0tAsKztOPPf00AFX4WbEa1hir0SbMTgsMnJ8D5qIqv+Br6LSUhuc0eN1jOxm8vrpnHLEaWHaxllkY9bJNmw6Mzb39AFQ5VaxGvdweRZH6pSPAwtw6Vuii66khY6yUPHursKv34vPPpKc9TuOYjrLFI8VTOGlsW3apy6rQv30CIWI5RhjkZgVqpIZP8v+InXa08ZWZy08MVBtHw9qm2VgIB6abyx2KFEN+9rbRnV36qCbtZ8UqWl2BlukK1KxAv+S6/zLHraoZPtvVJr3ige02Yf6XTYEYNLU+VcagpFjyW4+Z0W2Z7eVq1BU4+qWzB22wq30TZFSyggqkqhn+yvckqI29t3TBuWzaUsvY8X0sXX5jucORIrCCA7jHMhWrsXE/D+ZIJVaDswJVsfJJ4YB1JVZ+soJuBPGUuXz7rqtrhH7HQcGGMgclxVK4e0JPtmI1o7GxfAGVUKQSq3Ru8LpWXvfK0Aqk2tZHBhpjFSGPJafiDGZverrcoUiR5JbFCc3oJYED3vKmt5U1plKJWGKV+XW1V6BfDHUFiox2VrDyeneYwBLGkqXXlDUmKZ78XoFAn4tjcZg8ZVp5gyqRSCVWuTJzTGOsvFKYS5mpbX2UX11fW9p4LTd4fSCMEcR0LPsk350P9IWZxMpXkUqs8hUrLRDqFVUg/TekK7DMsUjppcJAiZVncsetI2Ag7XfiHKnEKswmVlqd2zNOg9ejwJGtaKgq6aeCrsBUOkYQ0w2TTwYTKxhIx4jF/T2OI5VY5Qavx0JVrHxihYmVkmYvZRYINc1N8FxuuEYYGjGPKxpRFMvfEBnpdEA85u91OFKJVYgGr/vI0oWD13Uy9lHh3a6qkn4qbNZ0OiAR+HvhjaRs0uwM0ikjocTKD7m9Ak3LLXilMJnSNddXTjsw+66gaV0K4kqsvBJkz9MhhksZySBV5ohKJ1KJVW7l9VhaB6xPYqEqVr7LLy7o1MY+y80KdGlIKrHyiuXa1gyXclR63L6RSqzCIR0K4g9VrHyXy6W0V6DHLJM4//Ab/41LQ8L8vfBGUSx/+XWY83sHlGglVtlZgRq87pcYg+2paoafMoPXM92BGmPlp1yzPrbmPsxBBf52FUVRrn1zY519nmgUrcQq27Q6L3um4PhU2/rN31OxgMNhWLwagCS6AfZJLHtHlB/r7PFNcKQSq9wYKzTGyisFQ6y8PlijzPKD10HplacMcNCfrAKgwg2UNx4pqlyFKl+x8vgwjlRilRufEYT9ZY5Eiiko6Nr1ubwsmXWO1BXoLwcMxCoBqHTqCvRJLBYDyNchfT5XRyqxyo2xCtP+DpqLIgsGr7S65vrJcGiVFL/ljt2+eAJQxco3AbmuQI2x8krochWrvjJHIsWUHujNP1ZXoJ8KJhSpjX1lmYpkXyybWKWUWPkklits5MY6e3wXHK3EKtugqc6uMkcixZSmsGKli66PBk/CHp+NBYC+IAlApYZseCUWy6QbuW2LfF6oO5KJ1ao1z5Q5EimmRMEJOFL/oCMkN3jdOaVWvjLLzArsDeIAJNNKrHwSCzJn57QqVn4Js/WMH/7y/nKHIkWU7urMP1Y3UQR4fEIW6LVMYlWR0pANn8QTmcHrLr/skb/n6sglVjop+2dLqjv/2OeDNcoMCipWamNfOQd9ZBIrejpf+sUyquwwxkpdgX4InRIrH/Vs3px/rIqVnwaTKS234Ktcu/ZnE6vK7LIL4odkIjMpIZdY+Zx8+Py77SDUSdlL/cHg8hmR+gcdIcZg1UoLhPqtjxgugAVLv1LuUKSIgiDTFZibne9z5TlS1yGnrkAv3frjvxbUM/w9WKMs164OHcK+yqxVZgyEMSyuVvZNVWWmAhlqjJVfMmOsdMB6KdusgccHq+RWXlcbeyl7DA+EcSymNvZNZUVmq6LBwev+ilRi5TTGyns+3wVFWb5d1bzeS4UBgRIr78ST242x0uB1P2QGr/vbmJFmQz6IZ4auvF7WUKSEnMskVjElVt4ZU5WtWGmMlV80eN1j6gr0WmHFyucTcpRluniNMAwIAu3n6ptYMjvGKpdYeXwtjlRi5UAlDU/lmlWJlZ8KK1biN62u76exY8cAg735WsfKE5mBr+WOQkoi1xWorl4vDVasNHjdV8Zg4qw29k8yOXRWoM83wZFKrLRAqMfM/9V8oyx/Q6Rqhr+yDZsbgyN+GT9+IlA4xspfkUqsHLrb9Z3Pd0GRlkuYndMYK08NGUen87R3JjQ0AAWJlcc3wdFKrDSjyF/5WYH+HqxRpopVNDhnWgTWY47CBUL9naAwrMTKzOrN7HYze9LMnjCz15vZeDP7vZk9k/04rljBDlcmU9aF12excgcgJZG7u3WqZnhrMHlWz4K3bLD47HNVZ7i/27XAb5xzBwKHAk8A84A7nXOzgTuzn48IGrzuseyJWBUrP+UPWzWvx1zuf6pYeUzrWL0EM6sDjgduAnDO9TvntgInA7dkX3YL8J7hBlksDt3t+srUFeg5rWPlu3x/gqqSXsttaRN43MTDqVjNBFqB/zazh8zs22ZWAzQ655qzr9kANA43yGJRxcp/gfO33z7KcicqQ+MkvefxBTfyCroCfW7o4SRWceAw4Hrn3OuALrbr9nPOOXbx1zOzT5nZSjNb2draOowwdp82cPVYbuV1XXS9VFilUsXKT2YFXYE6T3sr1xWovQJ3bh2wzjl3X/bz28kkWi1m1gSQ/bhxZ9/snPumc+4I59wRDdlpmKWmWYH+yjWrz1N4o8x28Vh8Y5r56bkoLAK7x4mVc24DsNbM5mSfOgF4HPgZcFr2udOAnw4rwiJyGp/hr+yZOKbEykuqWPmvcM622thPZgUVq9DfNo4P8/u/ANxqZklgFXA6mWTtx2Z2BrAaOGWY71E86gr0lwavR4aqGZ7KZlbaK9BjBQ3r87V4WImVc+5h4IidfOmE4fzcUnEOAo8bM8oGN2HW4HUfDRmPoWPYS9oPMhoGx1j5e672eY2uHThMd0K+yq9jJX4avNBG6qQVNdnpTqo8+8oGx1iVN5CSitY5SuujeE9jrPwUaIyV94aMsfL5qhtlBuQrVuUNpZQilVipK9Bf+QVCPS4vS4auuZ7KnZudNlP32WDFyt9zdaQSK5WY/aeKpJ+GLregNvaRweDS62pjb6kr0DOqWPkrl1D5PIU3yoZ2BYrXNCvQXwX9vT6vORipxAqnweu+UxeCnwrvh1Sx8lOuXZ3GwvqrILHSyuu+UN+9t3JjrLTcgq8KKla66HrJGOwmitaFKUrc4HILHlc5ovXv1znMdOH1Ue5u19QV6KUhY6zUxF6ygsHrqkr6aXAcnboCveFcxH7hKMlvwuzvwRplhTOIdNH1mNNegV4rHGPl8XEcrTxDg9e9F4SqSPpIswL9ZwUf1cY+y0408riJo5dY6YD1Un5WYJnjkNIYklipnOEpNziwWedpb+WWGgw8zqwidR0y9d17K79XoBYI9VLhDCKfx2ZEmRWMXtcEBU8VdgV6PB42MonVqScdBehOyFv5Koba10eqWPnPCtYF9XkqvmT4nHz4/LsNsd+M/QCIaTq+l3KVyJjGWHlKewX6zsxFYlXuKDMbnNXr83jnyCRWsZoxgE7KvhrcK7C8cUhpmLoCI8BFYsaYZMQ8Lj1HJrEKgwSgrkBf5U7EgaXLHImUQuGYm8ictCLGGLwxUmLlqYJcyueb4Mico2JBEvC7/BhpuQNW1Qzv6aLrJ42j819hswYeL70emcQqHYsDGhTpq3zFSu3rpcLNtZVY+aqgKqnj2Hsxf/OqaCRWFy28gMcn7A9ATCdlL+WO0ZjHU3ijrLAr0OPzcaRpEdgIKDiOlViNcmvGTuPBDdMAqA17yxyNlELuwmthqsyRSCkU9uCrmuGnwu6/SFyYIi4WxModQsnEyx3A3jBrywtMHNtOjBT333kLnP35cockJaOLro8C7RXoPRuypIaWTfFRYZEqbv6mz5FIrBZd9pWCzxaWKwwpoVzFKoZmBfpOA5v9pyb2VEHDJjzuC/Q3ZZRIyU8KDJVY+aiwK1DrWPlpyIwxbU3lvXjC37qOEivxQq4bQctYecoVrmOlxMpHhV2BamM/FU5CSZi/Y6yUWIkXBu92lVn5KNCsQO+pXaMlkUiUO4SSUWIlXsjdCYVpzQr0kRWuY6WuQE8VTMVXxcp7lZXJcodQMkqsxAu5u920ClZe0owx/9kuHos/CieeJBMV5QukxJRYiRdyFauB2LYyRyKloIuu/4bsB6nB696rrKgqdwglo8RKvJCraCTb1RXooyEzxtRN5KUhMz/Vxp4abNfKqsoyxlFaSqzEC7kL74bW1WWNQ0rDCioYga65Xhra3Ss+KuwKrB0zpnyBlJgSK/FC7qT8ndvvLXMkUgpDL7TKrHxU2MYxTVDw3tgxdeUOoWSUWIkXDIfTba7HNCvQf24Xj8VHdfXjyh1CySixEi8op/JbULjcgumi66OhmzBr8LqPCo/dZEV1GSMpLX/XlJdIOTC9gf5J/q7kG3VDVuUOlVj5aMgEBbWx1xwwfuLEcodRMkqsxAs3nXtOuUOQklLFyndDZgKqjb1kOzzwk7oCRWTEs4IBdIHGWHlPg9c95XlClaPESkRGPCvYXVtrHPlp6HILamMf5dvV8wRLiZWIjHhDVl7XNddLQxYI1RgrGcWUWInIyFfYNaRuIi8NmaCg7NlL+ZmfqliJiJRX4dpVuuj6aUhVUhUrT2Xa1fO8aviJlZnFzOwhM/tF9vOZZnafmT1rZj8ys+TwwxSRKNNF139DKlYaY+UlzQrcfXOBJwo+vwK42jm3P7AFOKMI7yEiUZYu2CuwjGFI6QwdR6fEykv5rkC/M6thnaPMbBpwEvDt7OcGvBm4PfuSW4D3DOc9RERiBSdizRjz09CKlVZe91FUjt3h3vxdA3wJ8kfBBGCrcy6V/XwdMHWY7yEiURem8g/N6aLrOw2j85zfBas9T6zM7J3ARufcA3v4/Z8ys5VmtrK1tXVPwxCRCBjSTRSRu96oKVz4NVDy7CXP86m84VSsjgXebWYvAD8k0wV4LVBvZrmtcqYBL+7sm51z33TOHeGcO6KhoWEYYYiI/woWj1Re5Sm1se/y21F53sB7nFg55y50zk1zzs0ATgX+6Jz7CPAn4APZl50G/HTYUYpIpBWOdVU1w0/BkP0g0y/xShntfK9clWKCzQXAOWb2LJkxVzeV4D1EJEICV3ih9ftuN7IKJyhoVqDfPM+s4i//kpfnnLsLuCv7eBVwVDF+rogIMCSX8vycHFmFyZQSKz+Z512AOVoSRkRGPFdQsdJUfD8VdgUGodrYR1ogVERkhLDC66yqGV4aWsxQG/vM87xKiZWIjAKFXYFKrDw12K4xVay8lF8qxfPMSomViIwCBV2Buuh6aei1Vm3sI3UFioiMEBYMzrMJfD8rR5QVJFMxJVZeyg1e9/0IVmIlIiNeeqB38JOC7W3EH0Muturt9ZzfDazESkRGvHThZTciU7ajZkhildYCoT5SV6CIyEiR1hpHvivcA9JCtbGf1BUoIjIiWKw//1jjb/wUFORSaVUlvZRfXN/zzEqJlYiMeP0MJlaeD8+ILFfYsOoK9NJgvuz3QazESkRGvGT74IB1rbbgp6BwS5u0GtlH+VmBqliJiJRXXzhYsQpCVTO8F1Mby+ilxEpERr4hSyzoouujwg16NcTKT7kJCqpYiYiUWTI2Kf849H3ka0QFBTMBJ0ycWsZIpFQGj1y/M2clViIy4l1+w42Dn2hgs58KylSN1dPKGIiUjipWIiIjRu6ym9bAZi/lrrUOOPXM08sZipSI5/lUnhIrERkdsmflwjWtxB/5WYFRufpGUK5SZZ4PolNiJSKjinV2lTsEKQHzfNyNRKeNlViJyKjSmxoodwhSAvkihipW3vO9iZVYicjokOtGGLL0gnjH96tuhGkTZhGRESgZbyx3CFICQUQ26I2y/DpWnncJKrESkdHBMjPGll1/Q7kjkZLQ4HXf5fdg9ryNlViJyKhg+f+Ijyy3QKjvV90IG6xUqWIlIlJ+ut56zffuIRnk+6GsxEpERg/fz8gRFpWBzVE2uFeg30m0EisRGSV0xfWZEiv/RaVplViJyOhgROfMHEHmMlsVqUvQX6a9AkVERhbPz8eRpoqV//KzAj1PnpVYicjooIqV57Tcgu98T6hylFiJyCiiq66vPB/PLGgTZhGRkUUVK68FuTFWamN/uWisrq/ESkRGCb/vckVdgb7LV6w8P5aVWInIqKCV1/1mLtO4amJ/WUQqVvFyByAislt8PxtHnFk6+8DvakaUDe4V6Hcbq2IlIqOI3yfkKFPeHAXROH5VsRKR0cF08fWai8bikVGWq+RojJWIiEiJBflrrd8X3SjL5cy+Jx6+/34i4gkNXvedKla+0ybMIiIjiS64XrNQ61j5brALUImViMiIoIuuvwJlzpHhe0srsRKR0cH3s3HEhWQrVp5XM6IsyK9j5Xcb73FiZWbTzexPZva4mT1mZnOzz483s9+b2TPZj+OKF66IRJXGWPktlu0KVBv7LBoLhA6nYpUCznXOHQwcA5xpZgcD84A7nXOzgTuzn4uIDJPz/oQcZc6iMbA5yoJsYhV4fiDvcWLlnGt2zj2YfdwJPAFMBU4Gbsm+7BbgPcMNUkQkswmzLrq+ym13Ih7LD5L0u62LMsbKzGYArwPuAxqdc83ZL20AGovxHiIi4i/LXnQ1QcFfUdkrcNiJlZnVAj8BznLOdRR+zTnn2EVqamafMrOVZraytbV1uGGIiOfMdNH1WvZK4fvA5igL0OD1l2VmCTJJ1a3OuTuyT7eYWVP2603Axp19r3Pum865I5xzRzQ0NAwnDBERGeUCl9mEWcmzvywiyfNwZgUacBPwhHNuRcGXfgacln18GvDTPQ9PRCTHaWCzx3KTAn0ffxNt0egKHM4mzMcCHwP+aWYPZ5+bDywHfmxmZwCrgVOGF6KIiCoZvlPFyn+W/+h38rzHiZVz7l52nXiesKc/V0RkV3TR9ZeF0Rh/E2WWXQQ28LyNtfK6iIwKSqr85iy38rr4yrb76CslViIyaqia4a+Q3HILamNfRaUrUImViIwSzv9b3ShLD5Q7AimxXELle/V5OIPXRUT2mopEqtwhSEllrraBKlbeCiKy3IISKxEZFU7edGf20fvKGoeUSNpBrNxBSGllK1aeb1+kxEpERoUly68rdwhSQi6WqUj6Xs2IMm1pIyIispcE6gr0Xm5igu/JsxIrEREpO+VT0eF74uH77yciIqNAzDIjU3yvZkRZLCKLwCqxEhGRsqsb3wjoouQ1y31QYiUiIlJSjdVTAS0Q6rP84HXPZwUqsRIRkbL70Oc/AfhfzYiygNy2RX63sRIrEREZERz+T8WPtuzMTyVWIiIie4H5X82IMt+7AHOUWImIyIgRqGTlrVzT+l6x0srrIiIyIuwzeSuH9K0rdxhSIrkxVoHfeZUSKxERGRnumfuRcocgpZTvCvQ7s1JXoIiIiJRcrlLle1egEisREREpPS0QKiIiIlIcQW6B0DLHUWpKrERERKTkcjM+A8+XXVBiJSIiIiVXHQ4AUBX2lzmS0lJiJSIiIiV3wdx5vLf+Ec593wfLHUpJabkFERERKbm6qgRXz5tf7jBKThUrERERkSJRYiUiIiJSJEqsRERERIpEiZWIiIhIkSixEhERESkSJVYiIiIiRaLESkRERKRIlFiJiIiIFIkSKxEREZEiMTcCNkM0s1ZgdYnfZiKwqcTvIeWlNvaf2th/amP/+dDG+zrnGnb2hRGRWO0NZrbSOXdEueOQ0lEb+09t7D+1sf98b2N1BYqIiIgUiRIrERERkSKJUmL1zXIHICWnNvaf2th/amP/ed3GkRljJSIiIlJqUapYiYiIiJRUJBIrM3u7mT1lZs+a2bxyxyN7xsy+Y2YbzexfBc+NN7Pfm9kz2Y/jss+bmX012+aPmtlh5YtcdoeZTTezP5nZ42b2mJnNzT6vNvaEmVWa2f1m9ki2jRdln59pZvdl2/JHZpbMPl+R/fzZ7NdnlDN+2X1mFjOzh8zsF9nPI9PG3idWZhYDrgPeARwMfMjMDi5vVLKHbgbevt1z84A7nXOzgTuzn0OmvWdn//8p4Pq9FKPsuRRwrnPuYOAY4Mzssao29kcf8Gbn3KHAa4G3m9kxwBXA1c65/YEtwBnZ158BbMk+f3X2dTI6zAWeKPg8Mm3sfWIFHAU865xb5ZzrB34InFzmmGQPOOf+DLRt9/TJwC3Zx7cA7yl4/rsu4+9AvZk17Z1IZU8455qdcw9mH3eSOSlPRW3sjWxbbct+msj+3wFvBm7PPr99G+fa/nbgBDOzvRSu7CEzmwacBHw7+7kRoTaOQmI1FVhb8Pm67HPih0bnXHP28QagMftY7T6KZbsDXgfch9rYK9kuooeBjcDvgeeArc65VPYlhe2Yb+Ps19uBCXs3YtkD1wBfAsLs5xOIUBtHIbGSiHCZKa6a5jrKmVkt8BPgLOdcR+HX1Majn3Mu7Zx7LTCNTI/CgWUOSYrIzN4JbHTOPVDuWMolConVi8D0gs+nZZ8TP7Tkun+yHzdmn1e7j0JmliCTVN3qnLsj+7Ta2EPOua3An4DXk+nGjWe/VNiO+TbOfr0O2LyXQ5VX5ljg3Wb2ApmhN28GriVCbRyFxOofwOzsjIQkcCrwszLHJMXzM+C07OPTgJ8WPP/x7MyxY4D2gu4kGYGy4ypuAp5wzq0o+JLa2BNm1mBm9dnHVcBbyYyl+xPwgezLtm/jXNt/APij0+KLI5pz7kLn3DTn3Awy19s/Ouc+QoTaOBILhJrZiWT6fGPAd5xzS8sckuwBM/sB8EYyO6O3AJcC/wf8GNgHWA2c4pxry16kv05mFmE3cLpzbmU54pbdY2b/BtwD/JPBsRnzyYyzUht7wMxeQ2agcozMjf2PnXOLzWwWmerGeOAh4KPOuT4zqwS+R2a8XRtwqnNuVXmil1fKzN4InOece2eU2jgSiZWIiIjI3hCFrkARERGRvUKJlYiIiEiRKLESERERKRIlViIiIiJFosRKREREpEiUWImIiIgUiRIrERERkSJRYiUiIiJSJP8fYH/UKNsgvQkAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 720x720 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "lamda2=1e-3\n",
    "\n",
    "thres = .95\n",
    "lamda=4e2\n",
    "Classifier = []\n",
    "Encoder = []\n",
    "Decoder = []\n",
    "GMMs = []\n",
    "\n",
    "fig=plt.figure(figsize=(10,10))\n",
    "# reinitLayers(encoderX)\n",
    "# reinitLayers(encoderY)\n",
    "# reinitLayers(classifier)\n",
    "\n",
    "#seed=np.random.randint(0,high=100)\n",
    "loss = []\n",
    "testXperf = []\n",
    "for cl in range(5):\n",
    "    # extract the dataset\n",
    "    #for jj in range(cl+1):\n",
    "    classifier = classifierNN(2*cl+2,yin)\n",
    "    if cl == 0:\n",
    "        indCL = np.concatenate([np.where(np.argmax(labelX_train,axis=1)==0)[0],np.where(np.argmax(labelX_train,axis=1)==1)[0]])\n",
    "        labelX_trainCL =  labelX_train[indCL,:2*cl+2] \n",
    "        dataX_train1CL =  dataX_train[indCL,:,:,:] \n",
    "    else:\n",
    "        indCL = np.concatenate([indCL,np.where(np.argmax(labelX_train,axis=1)==2*cl)[0],np.where(np.argmax(labelX_train,axis=1)==2*cl+1)[0]])\n",
    "        xreptrainlabelonehotExtended = labelX_train[indCL,:2*cl+2] \n",
    "        labelX_trainCL =  labelX_train[indCL,:2*cl+2] \n",
    "        dataX_train1CL =  dataX_train[indCL,:,:,:] \n",
    "        \n",
    "\n",
    "    labelX=K.placeholder(shape=(None,2*cl+2),dtype='float32') #labels of input images oneHot\n",
    "   \n",
    "     \n",
    "    #discriminationLoss=K.mean(K.categorical_crossentropy(labelX,classifier(encoderX(imgX))))\n",
    "    discriminationLoss=K.mean(K.binary_crossentropy(labelX,classifier(encoderX(imgX))))\n",
    "    \n",
    "    myLoss=  discriminationLoss\n",
    "     \n",
    "  \n",
    "    params=encoderX.weights   + classifier.weights\n",
    "\n",
    "    \n",
    "    \n",
    "    #loss=[]\n",
    "    \n",
    "    \n",
    "    \n",
    "    opt = Adam(lr=1e-4)\n",
    "    updates = opt.get_updates(myLoss,params)\n",
    "    train = K.function(inputs=[imgX,labelX],outputs=[discriminationLoss],updates=updates)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "    for itr in range(epochs):\n",
    "        \n",
    "        indTrainDataX,trainLabelX=batchGenerator(labelX_trainCL,batchsize,nofclasses=2*cl+2)\n",
    "        trainDataX=dataX_train1CL[indTrainDataX,...]\n",
    "        \n",
    "        \n",
    "  \n",
    "        loss.append(train(inputs=[trainDataX,trainLabelX]))\n",
    "  \n",
    "        if itr%epochstep==0: \n",
    "\n",
    "            for jj in range(2*cl+2):\n",
    "                if jj == 0:\n",
    "                    indCLtest = np.where(np.argmax(labelX_test,axis=1)==0)[0]\n",
    "                else:\n",
    "                    indCLtest = np.concatenate([indCLtest,np.where(np.argmax(labelX_test,axis=1)==jj)[0]])\n",
    "            labelX_testCL = labelX_test[indCLtest,:]\n",
    "            dataX_test1CL = dataX_test[indCLtest,:,:,:]   \n",
    "\n",
    "            perd_label_X = classifier.predict(encoderX.predict(dataX_test1CL))\n",
    "\n",
    "\n",
    "\n",
    "            \n",
    "            testXperf.append(100*float(sum(1*(np.argmax(perd_label_X,axis=1)==np.argmax(np.squeeze(labelX_testCL),axis=1))))/labelX_testCL.shape[0])\n",
    "            \n",
    "            plt.subplot(2, 1, 1)\n",
    "            plt.plot(np.asarray(loss))\n",
    "            \n",
    "            plt.subplot(2, 1, 2)\n",
    "            plt.plot(np.asarray(testXperf))\n",
    "            \n",
    "            display.clear_output(wait=True)\n",
    "            display.display(plt.gcf()) \n",
    "            time.sleep(1e-3) \n",
    "    Classifier.append(classifier)  \n",
    "    Encoder.append(encoderX)  \n",
    "    Decoder.append(decoderX)  \n",
    "    \n",
    " "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.plot(testXperf)\n",
    "plt.ylim([70,100])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.plot(testXperf)\n",
    "plt.ylim([70,100])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.imshow(xreptrainData[3010,:,:,0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "import json\n",
    " \n",
    "testpermutedperf = testXperf\n",
    "with open('FR_Incremental_MNSIT.json', 'w') as fp:\n",
    "    json.dump(testpermutedperf, fp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('FR_Incremental_MNSIT.json', 'r') as fp:\n",
    "    data = json.load(fp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "task = 1\n",
    "numtasks = 9\n",
    "epochs1 = epochs/epochstep\n",
    "fig, ax=plt.subplots(1,1,figsize=(10,10))\n",
    "\n",
    "plt.plot(np.arange(epochs1*(task),epochs1*(numtasks+1)),testpermutedperf,linewidth=4)\n",
    "plt.grid()\n",
    "plt.tick_params(labelsize=20)\n",
    "\n",
    "ax.set_xlabel('Epochs',fontsize=24)\n",
    "ax.set_ylabel('Testing Accuracy',fontsize=24)\n",
    "\n",
    "#ax.set_title('Training loss for the AE tasks',fontsize=14)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "xencoded = encoderX.predict(dataX_train1CL)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "umapperT=umap.UMAP()\n",
    "#xumapT=umapperT.fit_transform(xreptrainDataencoded,xreptrainlabel) \n",
    "xumapT=umapperT.fit_transform(np.concatenate([xencoded[:,0,0,:]],axis=0) )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "vislabel = np.concatenate([np.argmax(labelX_trainCL,axis=1),np.argmax(labelX_test,axis=1)])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "labelX_trainCL.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure(figsize=(10,10))\n",
    "\n",
    "plt.scatter(xumapT[:,0],xumapT[:,1],c=vislabel[:],cmap='tab10',linewidth=1,s=1)\n",
    "\n",
    "\n",
    "plt.grid()\n",
    "plt.tick_params(labelsize=20)\n",
    "\n",
    "\n",
    "plt.show()\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
