{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5d0c5b0f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import random\n",
    "np.set_printoptions(precision=3)\n",
    "\n",
    "from SQANN.model import SQANN, double_selective_activation\n",
    "from SQANN.utils import make_layer_setting, simple_evaluation, standard_evaluation"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cedddf66",
   "metadata": {},
   "source": [
    "# Data and some settings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "1afe9201",
   "metadata": {},
   "outputs": [],
   "source": [
    "a1 = 0.001\n",
    "a2 = 0.5\n",
    "t_admission = 0.1\n",
    "t_threshold = 0.9\n",
    "\n",
    "# training data\n",
    "X = np.array([[1.,1.2],[1.2,0.8],[-1.,-1.],[-1.2,-1.2],])\n",
    "Y = np.array([1.,1., 0.,0.])\n",
    "N = len(X)\n",
    "\n",
    "# test data\n",
    "X_test = np.array([[1.25,1.25],[-1.25,-1.],[-1,-1.4]])\n",
    "Y_test = np.array([1.,0.,0.])"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a0b0ba9a",
   "metadata": {},
   "source": [
    "# Some scratch Works"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "c674d4c6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "act1_0: 0.3344454438567918\n",
      "act2_0: 5.655468838366701e-05\n",
      "act3_0: 4.7165361758324685e-05\n",
      "act3_2: 0.5009573514714075\n",
      "next layer:\n",
      "act1_2: 6.187353050365053e-05\n",
      "act3_1: 0.007324948714461226\n",
      "\n",
      "test:\n",
      "test0_0: 0.5052960903090358\n",
      "test0_2: 4.9377839225755484e-05\n",
      "test_0_1: 0.5164688877956867\n",
      "test_0_3: 0.0009858581567478246\n",
      "test_1_0: 5.048720149442117e-05\n",
      "test_1_2: 0.5059247004830902\n",
      "test_1_3: 0.9879598030304475\n"
     ]
    }
   ],
   "source": [
    "act1_0 = double_selective_activation(np.sum((X[1]-X[0])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "act2_0 = double_selective_activation(np.sum((X[2]-X[0])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "act3_0 = double_selective_activation(np.sum((X[3]-X[0])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "act3_2 = double_selective_activation(np.sum((X[3]-X[2])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "\n",
    "print('act1_0:',act1_0)\n",
    "print('act2_0:',act2_0)\n",
    "print('act3_0:',act3_0)\n",
    "print('act3_2:',act3_2)\n",
    "\n",
    "# next layer\n",
    "\n",
    "act1_2 = double_selective_activation(np.sum((X[1]-X[2])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "v1 = np.array([act1_0,act1_2])\n",
    "v3 = np.array([act3_0,act3_2])\n",
    "act3_1 = double_selective_activation(np.sum((v3-v1)**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "print('next layer:')\n",
    "print('act1_2:',act1_2)\n",
    "print('act3_1:',act3_1)\n",
    "\n",
    "# test\n",
    "test0_0 = double_selective_activation(np.sum((X_test[0]-X[0])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "test0_2 = double_selective_activation(np.sum((X_test[0]-X[2])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "print('\\ntest:')\n",
    "print('test0_0:',test0_0)\n",
    "print('test0_2:',test0_2)\n",
    "\n",
    "vtest_0 = np.array([test0_0,test0_2])\n",
    "test_0_1 = double_selective_activation(np.sum((vtest_0-v1)**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "test_0_3 = double_selective_activation(np.sum((vtest_0-v3)**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "print('test_0_1:',test_0_1)\n",
    "print('test_0_3:',test_0_3)\n",
    "\n",
    "test_1_0 = double_selective_activation(np.sum((X_test[1]-X[0])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "test_1_2 = double_selective_activation(np.sum((X_test[1]-X[2])**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "print('test_1_0:',test_1_0)\n",
    "print('test_1_2:',test_1_2)\n",
    "\n",
    "vtest_1 = np.array([test_1_0,test_1_2])\n",
    "test_1_3 = double_selective_activation(np.sum((vtest_1-v3)**2)**0.5 ,r=0.5, a1=a1, a2=a2)\n",
    "print('test_1_3:',test_1_3)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6b50257c",
   "metadata": {},
   "source": [
    "# Automated Model Building"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "8999d07d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Exiting layer_k_sample_collection() because all data have been used.\n",
      "Final positions of indices in the layers:\n",
      "  [1] [0, 2]\n",
      "  [2] [1, 3]\n"
     ]
    }
   ],
   "source": [
    "MAX_LAYER = 3\n",
    "layer_settings = {\n",
    "    # make_layer_setting(a1, a2,  admission_threshold, activation_threshold, max_n)\n",
    "    i: make_layer_setting(a1, a2,t_admission, t_threshold, ) for i in range(1,1+MAX_LAYER) # BEST NOW\n",
    "}\n",
    "\n",
    "net = SQANN(layer_settings, N)\n",
    "net.fit_data(X,Y,verbose=20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "9f135f29",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "simple_evaluation()\n",
      "Show fitting on training data\n",
      "N_MISSED_ACTIVATIONS:0, N_large_error (>0.1):0\n",
      "avg error          :     0.0, avg_frac_error          :     0.5 \n",
      "avg exclusive error:     0.0, avg exclusive frac error:     0.5\n"
     ]
    }
   ],
   "source": [
    "simple_evaluation(X,Y,net, header_text='Show fitting on training data', verbose=20)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "da6c4572",
   "metadata": {},
   "source": [
    "# Testing Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "cdf5022f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " i     Layer     |y-y0|                           abs error\n",
      "[0]    L=[1, 2]    |(1.0  ) - (1.0  ))|                 0.0      INTERPOLATE\n",
      "[1]    L=2         |(0.0  ) - (0.0  ))|                 0.0      HIT\n",
      "[2]    L=[1, 2]    |(0.0  ) - (0.0  ))|                 0.0      INTERPOLATE\n",
      "N_INTERPOLATED:2, N_large_error (>0.1):0\n",
      "avg error          :     0.0, avg_frac_error          : 0.66667 \n",
      "avg exclusive error:     0.0, avg exclusive frac error: 0.66667\n"
     ]
    }
   ],
   "source": [
    "_, INTERP_INDICES_test = standard_evaluation(X_test, Y_test, net, get_interp_indices=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "1ed7108a",
   "metadata": {},
   "outputs": [],
   "source": [
    "X_INTERP, Y_INTERP = [], []\n",
    "INTERP_MEMBER_LIST = []\n",
    "for i in INTERP_INDICES_test:\n",
    "    X_INTERP.append(X_test[i,:])\n",
    "    y_interp,_,_, interp_info = net.SQANN_propagation(X_test[i,:],ALLOW_INTERPOLATION=True)\n",
    "    Y_INTERP.append(y_interp)\n",
    "    \n",
    "    interpolation_ends = []\n",
    "    for _, info in interp_info.items():\n",
    "        _, layer_k, this_index = info\n",
    "        interpolation_ends.append(net.used_indices[layer_k][this_index])\n",
    "    INTERP_MEMBER_LIST.append(interpolation_ends)\n",
    "    \n",
    "X_INTERP, Y_INTERP = np.array(X_INTERP), np.array(Y_INTERP )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "d4a87257",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXIAAAFlCAYAAAD292MqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjMuNCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8QVMy6AAAACXBIWXMAAAsTAAALEwEAmpwYAAAWNklEQVR4nO3de3BV9bmH8e+bkIgJERCQFqOEEcZyaYsSKYxjhYASkGJRYaxjO21taTv2NtOxlaH28odtz9CxtUerpQPtnFbxglpawRJog3haPRpS6iCgBIUSrRhAS+Qeec8fGwghwVz2Tlbe7OczkzF77Z213mzlcfHbKzvm7gIAxJWT9AAAgPQQcgAIjpADQHCEHACCI+QAEBwhB4DgeiVx0IEDB3pJSUkShwaAsNavX7/b3Qedvj2RkJeUlKiqqiqJQwNAWGa2o6XtLK0AQHCEHACCI+QAEFwia+QA0F5Hjx5VbW2tDh06lPQona53794qLi5WXl5emx5PyAGEUFtbq6KiIpWUlMjMkh6n07i79uzZo9raWg0bNqxNX8PSCoAQDh06pAEDBvToiEuSmWnAgAHt+psHIQcQRk+P+Ant/T4JOQC00TvvvKNf/vKX7f66GTNm6J133sn8QMcRcgA9T0ODtHChdMEFUkGBNHastHx52rs9U8gbGhre9+tWrlypfv36pX38MyHkAHoWd+naa6Uf/ECqrZUOHpT++U/pppuku+9Oa9e33367tm3bprFjx+qyyy7TFVdcoVmzZmnUqFGSpE9+8pMaN26cRo8erUWLFp38upKSEu3evVvbt2/XyJEj9cUvflGjR4/W1VdfrYMHD6Y1k6TUK6Rd/TFu3DgHgPbYtGlT2x749NPuhYXuqaQ3/Tj7bPf6+g7P8Nprr/no0aPd3b2ystILCgr81VdfPXn/nj173N39wIEDPnr0aN+9e7e7uw8dOtTr6ur8tdde89zcXP/HP/7h7u5z5szx3/3udy0eq6XvV1KVt9BUzsgB9CzLlkkHDrR8X16eVFmZsUONHz++ySWCv/jFL/TRj35UEyZM0M6dO7V169ZmXzNs2DCNHTtWkjRu3Dht37497Tm4jhxAz3LsWOr8+0wy+AvnCwsLT36+du1arVmzRs8++6wKCgo0adKkFi8hPOuss05+npubm5GlFc7IAfQs118vnRLYJo4elSZN6vCui4qKVF9f3+J9//nPf9S/f38VFBRoy5Yteu655zp8nPbijBxAzzJpkjRxovS3v6Ve6DyhoCD1Aug553R41wMGDNDll1+uMWPG6Oyzz9bgwYNP3ldeXq77779fI0eO1MUXX6wJEyZ0/HtoJ/MM/jWjrUpLS533IwfQHps3b9bIkSPb9uAjR6T/+i/p3nulPXukESNSEZ87t1NnzKSWvl8zW+/upac/ljNyAD1Pfr50xx2pjyzAGjkABEfIASA4Qg4AwRFyAAiOkANAcIQcANqoo29jK0k///nPdeBMbx2QJkIOoGd55RXpqqukXbuabv/736VZs6T9+zu86+4acq4jB9CzvPlmKtqTJ6feIGvw4NTtadOkD35Q2rfvzD/C34pT38b2qquu0nnnnadHHnlEhw8f1uzZs/XDH/5Q+/fv19y5c1VbW6v33ntPd9xxh3bt2qU33nhDkydP1sCBA1WZwTfukgg5gJ7m4x+XVqyQrrkmFfM775Q+85lUxNeuTf2zg37yk59o48aN2rBhgyoqKrRs2TI9//zzcnfNmjVL69atU11dnYYMGaIVK1ZISr0HS9++fXXXXXepsrJSAwcOzNA32oilFQA9z6RJqZhv3ixdd52Um5uK+JAhGTtERUWFKioqdMkll+jSSy/Vli1btHXrVn34wx/W6tWr9Z3vfEfPPPOM+vbtm7FjnknaZ+RmdoGk/5E0WJJLWuTu6f0aDgBIV35+09u5uRndvbtr/vz5+tKXvtTsvurqaq1cuVLf/e53NWXKFH3ve9/L6LFPl4kz8gZJ33L3UZImSLrVzEZlYL8A0DEn1sRHjJAeeCD19rWTJzd/AbSdTn0b22nTpmnJkiV69913JUmvv/663nrrLb3xxhsqKCjQzTffrNtuu03V1dXNvjbT0j4jd/d/S/r38c/rzWyzpPMlbUp33wDQbqe+sHliOWXIkMY18xMvgHbAqW9jO336dN10002aOHGiJKlPnz76/e9/r5qaGt12223KyclRXl6e7rvvPknSvHnzVF5eriFDhmT8xc6Mvo2tmZVIWidpjLvvO+2+eZLmSdKFF144bseOHRk7LoCer81vY7tpk/TlL0sPPdR0TXztWunHP079Kriiok6bM1MSeRtbM+sj6TFJ3zw94pLk7oskLZJS70eeqeMCQBOjRklPPy2ZNd0+aZJ05ZXNt/cAGblqxczylIr4A+7+eCb2CQAddqZY98CISxkIuZmZpMWSNrv7XemPBABoj0yckV8u6dOSysxsw/GPGRnYLwA0kcSvpkxCe7/PTFy18r+SeubfVwB0G71799aePXs0YMAAWQ9dIpFSEd+zZ4969+7d5q/hR/QBhFBcXKza2lrV1dUlPUqn6927t4qLi9v8eEIOIIS8vDwNGzYs6TG6Jd5rBQCCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIjpADQHCEHACCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDiBznn5auuee5tuXLpUef7zr58kShBxA5ixeLH3ta9LddzduW7pUuvlm6b77JPfkZuvBeiU9AIAeZPFiaf9+6ZvfTN0+77xUxK+4QvrDHySzJKfrsQg5gKbeekt6+OHUP0tLpWuukXq1MRV5edJDD0k33tgY8yuvlFaskAoLO23kbEfIATR68EHplltSZ84HD0pFRdLAgdIzz0jnn9+2feTlSdde27gmfvXVRLyTEXIAKTU10he+IB061Litvj4V9Ouvl557rm37WbpU+tznpMsvl/r0kRYsSIX8G9/onLlByAEc96tfSQ0Nzbc3NEgvviht3SqNGPH++zjxwuYVV6SWU/Lzmy6zEPNOwVUrAFJeeUU6erTl+/LzpZ07W9/H+vWNES8sbFwzv+466fnnuWqlk3BGDiDlkkukVaukw4eb33f4sDR8eOv7WLgw9djevRu3nYi5GVetdBLOyAGkzJvX8tUp+fnSpEnShRe2vg+zphE/IS+v7Ve+oN0IOYCUIUOkP/1J6ts3dbVKYaFUUCBddllq7RvdFv+LBNBo8mRp1y7pz3+W6uqkceNSSy7o1gg5gKbOOit1HTjCYGkFAIIj5AAQHCEHgOAIOQAEl5GQm9kSM3vLzDZmYn8AgLbL1Bn5byWVZ2hfAIB2yEjI3X2dpL2Z2BcAoH26bI3czOaZWZWZVdXV1XXVYQGgx+uykLv7IncvdffSQYMGddVhAaDH46oVAAiOkANAcJm6/HCppGclXWxmtWZ2Syb2CwBoXUbeNMvdP5WJ/QAA2o+lFQAIjpADQHCEHACC4xdLAEifu/TCC1JVlTRwoDRzZurXxKFLEHIA6Xn7bam8XHrpJenYsdQvWXaXHn00tR2djqUVAOm56SZpwwZp/37p4EGpvl56913p+uulf/0r6emyAiEH0HE7d0pr10pHjjS/r6FBuv/+Lh8pGxFyAB23bVvqlzW35MgR6cUXu3aeLEXIAXTc0KHS4cMt35eXJ33oQ107T5Yi5AA6btgw6bLLUi9wnq5XL+krX+n6mbIQIQeQnkcflS66SCoqknJyUpcdnn229JvfpLaj03H5IYD0DB4sbdokrV7deB35nDnSuecmPVnWIOQA0peTI02blvpAl2NpBQCCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIjpADQHCEHACCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4DIScjMrN7OXzazGzG7PxD4BAG2TdsjNLFfSvZKmSxol6VNmNird/QIA2iYTZ+TjJdW4+6vufkTSQ5KuzcB+AQBtkImQny9p5ym3a49vAwB0gS57sdPM5plZlZlV1dXVddVhAaDHy0TIX5d0wSm3i49va8LdF7l7qbuXDho0KAOHBQBImQn5C5JGmNkwM8uXdKOkP2ZgvwCANuiV7g7cvcHMvipplaRcSUvc/aW0JwMAtEnaIZckd18paWUm9gUAaB9+shMAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIjpADQHCEHACCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIjpADQHCEHACCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIjpAH5O7t2t7R/QGIgZAHs2/fPpWVlWnlypVNttfW1mrChAmqrq5u034OHz6sBQsW6Nxzz1Vubq5KSkq0ePFiog4ElFbIzWyOmb1kZsfMrDRTQ+HM3nvvPe3bt0+zZ88+GfPa2lpNnjxZmzdv1uHDh1vdh7tr+vTp+tnPfqa3335b7q4dO3bo61//ur7//e939rcAIMPSPSPfKOk6SesyMAvaoH///lqzZo3GjBmj2bNn69e//rUmT56sXbt2qaKiQhMnTmx1H2vXrtULL7yggwcPNtl+4MABLVy4UHv37u2s8QF0grRC7u6b3f3lTA2DtjkR8/79+2vevHmqqalRRUWFJkyY0KavX758ufbv39/iffn5+Vq7dm0GpwXQ2bpsjdzM5plZlZlV1dXVddVhe6z9+/eroaHh5O32nEX36tXrfe/Pzc3t8FwAul6rITezNWa2sYWPa9tzIHdf5O6l7l46aNCgjk+Mk2viR44c0cqVK3XppZc2WTNvzZw5c1RQUNDifUePHtWUKVMyOS6ATtZqyN19qruPaeFjeVcMiKbefPPNJmvi06dPb7JmvmrVqlb3MX78eF1zzTXNYl5QUKCFCxeqT58+nTU+gE7A5YfB9O/fX6WlpU3WxE+smZeXl+uiiy5qdR9mpgcffFA//elPNWLECJ1zzjn62Mc+pmXLlunWW2/t7G8BQIZZOtcNm9lsSf8taZCkdyRtcPdprX1daWmpV1VVdfi4AJCNzGy9uze71Pv9X/Vqhbs/IemJdPYBAEgPSysAEBwhB4DgCDkABEfIe4CamhrNmTNHffr0UZ8+fTR37lxt27Yt6bEAdJG0XuxE8mpqalRaWqr6+nodO3ZMkvTYY4+poqJCVVVVGj58eMITAuhsnJEHN3/+/CYRl6Rjx46pvr5eCxYsSHAyAF2FkAf31FNPNYn4CceOHdOKFSsSmAhAVyPkABAcIQ9uxowZyslp/q8xJydHM2fOTGAiAF2NkAf3ox/9SEVFRU1inpOTo6KiIt15550JTgagqxDy4IYPH66qqirdcMMNKiwsVGFhoebOnav169e36Q20AMTH5Yc9wPDhw/Xwww8nPQaAhHBGDgDBEXIACI6QA0BwhBwAgiPkWWj79u36xCc+obq6uibbq6urdcMNN+jAgQMJTQagIwh5FtqxY4f+8pe/qKys7GTMq6urNXXqVK1fv1579+5NeEIA7UHIs9CVV16pJ598Utu2bVNZWZlWrVqlqVOnqm/fvqqsrFRxcXHSIwJoB0KepcrKyvTkk09q48aNKi8v15EjR1RZWamSkpKkRwPQToQ8i/Xr1+/k5zk5OSosLExuGAAdRsiz1Ik18ZKSEi1ZskQNDQ1N1swBxMGP6GehExE/sSZeUlKioUOHaubMmSorK9Nf//pXDRo0KOkxAbQRZ+RZKD8/X8OHD2+yJn5izbx///7q1Yv/vwORmLt3+UFLS0u9qqqqy4+LRu4uM2vzdgDJM7P17l56+nbOyLPUmWJNxIF4CDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIjpADQHCEHACCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIJLK+RmttDMtpjZi2b2hJn1y9BcAIA2SveMfLWkMe7+EUmvSJqf/kgAgPZIK+TuXuHuDcdvPiepOP2RAADtkck18s9LeupMd5rZPDOrMrOqurq6DB4WALJbr9YeYGZrJH2ghbsWuPvy449ZIKlB0gNn2o+7L5K0SJJKS0u9Q9MCAJppNeTuPvX97jezz0qaKWmKuxNoAOhirYb8/ZhZuaRvS7rS3Q9kZiQAQHuku0Z+j6QiSavNbIOZ3Z+BmQAA7ZDWGbm7D8/UIACAjuEnOwEgOEIOAMERcgAIjpADQHCEHACCI+QAEBwhB4DgCDkABEfIASA4Qg4AwRFyAAiOkANAcIQcAIIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0BwhBwAgiPkABAcIQeA4Ag5AARHyAEgOEIOAMERcgAIzty96w9qVidpR5cfuKmBknYnPEN3wXPRiOeiEc9Fo+7yXAx190Gnb0wk5N2BmVW5e2nSc3QHPBeNeC4a8Vw06u7PBUsrABAcIQeA4LI55IuSHqAb4bloxHPRiOeiUbd+LrJ2jRwAeopsPiMHgB4hq0NuZgvNbIuZvWhmT5hZv6RnSoqZzTGzl8zsmJl121fnO5OZlZvZy2ZWY2a3Jz1PUsxsiZm9ZWYbk54laWZ2gZlVmtmm438+vpH0TC3J6pBLWi1pjLt/RNIrkuYnPE+SNkq6TtK6pAdJgpnlSrpX0nRJoyR9ysxGJTtVYn4rqTzpIbqJBknfcvdRkiZIurU7/neR1SF39wp3bzh+8zlJxUnOkyR33+zuLyc9R4LGS6px91fd/YikhyRdm/BMiXD3dZL2Jj1Hd+Du/3b36uOf10vaLOn8ZKdqLqtDfprPS3oq6SGQmPMl7Tzldq264R9YJMfMSiRdIun/Eh6lmV5JD9DZzGyNpA+0cNcCd19+/DELlPor1ANdOVtXa8tzAaA5M+sj6TFJ33T3fUnPc7oeH3J3n/p+95vZZyXNlDTFe/i1mK09F1nudUkXnHK7+Pg2ZDkzy1Mq4g+4++NJz9OSrF5aMbNySd+WNMvdDyQ9DxL1gqQRZjbMzPIl3SjpjwnPhISZmUlaLGmzu9+V9DxnktUhl3SPpCJJq81sg5ndn/RASTGz2WZWK2mipBVmtirpmbrS8Re9vypplVIvaD3i7i8lO1UyzGyppGclXWxmtWZ2S9IzJehySZ+WVHa8ERvMbEbSQ52On+wEgOCy/YwcAMIj5AAQHCEHgOAIOQAER8gBIDhCDgDBEXIACI6QA0Bw/w9K/ef7YKrB3AAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x432 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def plot(X,Y, X_test, Y_test, X_INTERP, Y_INTERP, INTERP_MEMBER_LIST, with_annot=True):\n",
    "    VMAX = 1.5\n",
    "    plt.figure(figsize=(6,6))\n",
    "    \n",
    "    c = np.array([Y,Y*0,Y*0]).T\n",
    "    plt.gca().scatter(X[:,0],X[:,1],s=48, c=c, vmin=0.,vmax=VMAX,label='train')\n",
    "\n",
    "    c1 = np.array([Y_test,Y_test*0,Y_test*0]).T\n",
    "    plt.gca().scatter(X_test[:,0],X_test[:,1],s=48, c=c1, marker='x', vmin=0.,vmax=VMAX,label='test')\n",
    "    \n",
    "    if with_annot:\n",
    "        c2 = np.array([Y_INTERP,Y_INTERP*0,Y_INTERP*0]).T\n",
    "        plt.gca().scatter(X_INTERP[:,0],X_INTERP[:,1],s=128, facecolors='none' , edgecolors=c2 ,\n",
    "            linewidth=1, vmin=0.,vmax=VMAX,cmap='Oranges')\n",
    "\n",
    "        alt_line_style = ['-','-.']\n",
    "        alt_line_width = [0.5,1., 1.5]\n",
    "        alt_colors = ['b','g','y','cyan']\n",
    "        for i, (j,k) in enumerate(INTERP_MEMBER_LIST):\n",
    "            plt.plot([X[j,0], X_INTERP[i,0] ,X[k,0]],[X[j,1], X_INTERP[i,1] ,X[k,1]], \n",
    "                linewidth= alt_line_width[i%len(alt_line_width)] , \n",
    "                c=alt_colors[i%len(alt_colors)], \n",
    "                linestyle=alt_line_style[i%len(alt_line_style)])\n",
    "    plt.gca().set_xlim([-2.5,2.5])\n",
    "    plt.gca().set_ylim([-2.5,2.5])\n",
    "\n",
    "plot(X,Y, X_test, Y_test, X_INTERP, Y_INTERP, INTERP_MEMBER_LIST, with_annot=False)    \n",
    "plt.legend()\n",
    "plot(X,Y, X_test, Y_test, X_INTERP, Y_INTERP, INTERP_MEMBER_LIST) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "16e369c2",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
