{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from Network_bayesian import Place_net, Grid_net, Coupled_Net\n",
    "import brainpy as bp\n",
    "import brainpy.math as bm\n",
    "from matplotlib.animation import FuncAnimation\n",
    "import time\n",
    "from scipy.stats import ttest_ind, norm\n",
    "\n",
    "# 圆周距离函数\n",
    "def circ_dis(phi_1, phi_2):\n",
    "    dis = phi_1 - phi_2\n",
    "    dis = np.where(dis > np.pi, dis - 2 * np.pi, dis)\n",
    "    dis = np.where(dis < -np.pi, dis + 2 * np.pi, dis)\n",
    "    return dis\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "20\n"
     ]
    }
   ],
   "source": [
    "\n",
    "# 默认参数\n",
    "# grid spacing\n",
    "lambda_1 = 3\n",
    "lambda_2 = 4\n",
    "lambda_3 = 5\n",
    "Lambda = np.array([lambda_1, lambda_2, lambda_3])\n",
    "L = lambda_1 * lambda_2 * lambda_3\n",
    "# cell number\n",
    "num_p = int(200)\n",
    "rho_p = num_p / L\n",
    "rho_g = rho_p\n",
    "num_g = int(rho_g * 2 * np.pi)  # 为了让两个网络的rho相等\n",
    "print(num_g)\n",
    "M = len(Lambda)\n",
    "# feature space\n",
    "x = np.linspace(0, L, num_p, endpoint=False)\n",
    "theta = np.linspace(0, 2 * np.pi, num_g, endpoint=False)\n",
    "# connection range\n",
    "a_p = 0.3\n",
    "a_g = a_p / Lambda * 2 * np.pi\n",
    "# connection strength\n",
    "J_p = 20\n",
    "J_g = J_p\n",
    "J_pg = J_p / 25\n",
    "\n",
    "\n",
    "# divisive normalization\n",
    "k_p = 20.\n",
    "k_g = Lambda / 2 / np.pi * k_p\n",
    "# time constants\n",
    "tau_p = 1\n",
    "tau_g = 2 * np.pi * tau_p / Lambda\n",
    "# input_strength\n",
    "alpha_p = 0.05\n",
    "alpha_g = 0.05\n",
    "\n",
    "noise_ratio = 0.007\n",
    "\n",
    "P_CANN = Place_net(z_min=0, z_max=L, num=num_p, a_p=a_p, k=k_p, tau=tau_p, J0=J_p)\n",
    "# Grid cells\n",
    "G_CANNs = bm.NodeList()\n",
    "for i in range(M):\n",
    "    G_CANNs.append(Grid_net(z_min=0, z_max=L, num=num_g, num_hpc=num_p, L=Lambda[i], a_g=a_g[i], k_mec=k_g[i], tau=tau_g[i], J0=J_g, W0=J_pg))\n",
    "\n",
    "Coupled_model = Coupled_Net(HPC_model=P_CANN, MEC_model_list=G_CANNs, num_module=M)\n",
    "\n",
    "def Net_decoding(z_truth, phi_truth, Ip, Ig, alpha_p=0.05, alpha_g=0.05, Coupled_model=Coupled_model):\n",
    "    Coupled_model.reset_state()\n",
    "    def initial_net(Ip, Ig): \n",
    "        Coupled_model.initial(alpha_p=1, alpha_g=1, Ip=Ip, Ig=Ig)\n",
    "\n",
    "    def run_net(i, Ip, Ig): \n",
    "        Coupled_model.step_run(i, alpha_p=1, alpha_g=1, Ip=Ip, Ig=Ig)\n",
    "        phi_decode = Coupled_model.phase\n",
    "        z_decode = Coupled_model.HPC_model.center\n",
    "        rp = Coupled_model.HPC_model.r\n",
    "        up = Coupled_model.HPC_model.u\n",
    "        rg = bm.zeros([M, num_g])\n",
    "        ug = bm.zeros([M, num_g])\n",
    "        for mi in range(M):\n",
    "            rg[mi, :] = Coupled_model.MEC_model_list[mi].r\n",
    "            ug[mi, :] = Coupled_model.MEC_model_list[mi].u\n",
    "        return z_decode, phi_decode, rp, up, rg, ug\n",
    "\n",
    "    T_init = 500\n",
    "    z0 = z_truth\n",
    "    phi_0 = phi_truth\n",
    "    fg = np.zeros((M, num_g))\n",
    "    for i in range(M):\n",
    "        dis_theta = circ_dis(theta, phi_0[i])\n",
    "        fg[i, :] = np.exp(-dis_theta ** 2 / (4 * a_g[i] ** 2))\n",
    "    dis_x = x - z0\n",
    "    fp = np.exp(-dis_x ** 2 / (4 * a_p ** 2))\n",
    "    I_place = 1 * np.repeat(fp[np.newaxis, :], T_init, axis=0)\n",
    "    I_grid = 1 * np.repeat(fg[np.newaxis, :, :], T_init, axis=0)\n",
    "    I_place[int(T_init / 3):, :] = 0\n",
    "    I_grid[int(T_init / 3):, :, :] = 0\n",
    "\n",
    "    bm.for_loop(initial_net, (I_place, I_grid), progress_bar=False)\n",
    "    T = 5000\n",
    "    indices = np.arange(T)\n",
    "    I_place = alpha_p * np.repeat(Ip[np.newaxis, :], T, axis=0)\n",
    "    I_grid = alpha_g * np.repeat(Ig[np.newaxis, :, :], T, axis=0)\n",
    "    z_record, phi_record, rp, up, rg, ug = bm.for_loop(run_net, (indices, I_place, I_grid), progress_bar=False)\n",
    "\n",
    "    return z_record, phi_record, up, rp, ug, rg\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "Ap = 1.0084058\n",
    "Rp = 0.0128615275\n",
    "Ag = 0.9814125\n",
    "Rg = 0.013212965\n",
    "\n",
    "def Grid_tuning_generation(phi_candidate, a_g, num_g):\n",
    "    # phi_candidate shape: [M, n_candidate]\n",
    "    n_candidate = phi_candidate.shape[-1]\n",
    "    theta = np.linspace(0, 2 * np.pi, num_g, endpoint=False)  # Assuming theta is linspace\n",
    "    \n",
    "    # Broadcasting phi_candidate to shape [M, 1, n_candidate]\n",
    "    phi_candidate_expanded = phi_candidate[:, np.newaxis, :]\n",
    "    \n",
    "    # Calculate dis_theta using broadcasting\n",
    "    dis_theta = circ_dis(theta[:, np.newaxis], phi_candidate_expanded)\n",
    "    \n",
    "    # Calculate fg_prime using broadcasting\n",
    "    a_g_expanded = a_g[:, np.newaxis, np.newaxis]\n",
    "    fg_prime = np.exp(-dis_theta ** 2 / (4 * a_g_expanded ** 2))\n",
    "    \n",
    "    return fg_prime\n",
    "\n",
    "\n",
    "\n",
    "def loglikelihood_Ig(Ig, fg, sigma_g):\n",
    "    n_phi = fg.shape[-1]\n",
    "    # 使用 expand_dims 将矩阵扩展为 (n, m, 1)\n",
    "    Ig_expand = np.expand_dims(Ig, axis=-1)\n",
    "    # 使用 tile 将矩阵沿最后一个轴重复 K 次\n",
    "    Ig_expand = np.tile(Ig_expand, (1, 1, n_phi))\n",
    "    log_prob = -0.5 * (Ig_expand - fg)**2 / sigma_g[:,None,None]**2 # shape [M, n_g, n_phi]\n",
    "    log_prob = np.sum(log_prob, axis=1) \n",
    "    return log_prob # shape [M, n_phi]\n",
    "\n",
    "def mapping_func(x):\n",
    "    lambda_gs = Lambda\n",
    "    phi = x[:,None] % Lambda *np.pi*2/Lambda\n",
    "    return phi # [n_pos, M]\n",
    "\n",
    "def prior_function(phi, z_candidates, sigma_phi):\n",
    "    '''\n",
    "    P(phi | x), phi shape [M], phi_x is a scalar\n",
    "    '''\n",
    "    phi_x = mapping_func(z_candidates) # shape [n_pos, M]\n",
    "    kappa_phi = 1 / (sigma_phi)**2\n",
    "    log_prob = kappa_phi*np.cos(phi-phi_x) # shape [n_pos, M]\n",
    "    log_prob_z = np.sum(log_prob, axis=1)\n",
    "    return log_prob_z # shape [M]\n",
    "\n",
    "def Simplified_PSC_MAP_decoder(activation_gs, n_pos=4000, n_phi=100, M=3, \n",
    "                               alpha_p_infer=0.05, alpha_g_infer=0.05, Ap=Ap, Rp=Rp, Ag=Ag):  \n",
    "    '''\n",
    "    MAP: Maximum A Posteriori\n",
    "    activation_gs shape [M, n_g]\n",
    "    '''\n",
    "    sigma_g = np.sqrt(np.sqrt(np.pi) * Ag ** 3 * rho_g * tau_g / (a_g * alpha_g_infer))\n",
    "    sigma_phi = np.sqrt(8 * np.pi * Ag * tau_g / (Lambda * J_pg * rho_p * Rp))\n",
    "    sigma_g_infer = sigma_g * noise_ratio\n",
    "    sigma_phi_infer = sigma_phi * noise_ratio\n",
    "    \n",
    "    L_env = 60\n",
    "    ## parameter space\n",
    "    z_candidates = np.linspace(0, L_env, n_pos)\n",
    "    \n",
    "    phi_candidates = np.linspace(0, 2*np.pi, n_phi, endpoint=False)\n",
    "    phi_candidates_modules = np.tile(phi_candidates[:, np.newaxis], (1, M))\n",
    "    phi_candidates_modules = phi_candidates_modules.T\n",
    "    fg_modules = Grid_tuning_generation(phi_candidates_modules,a_g,num_g) # shape [M, n_g, n_phi]\n",
    "\n",
    "    log_likelihood_fr = loglikelihood_Ig(activation_gs, fg_modules, sigma_g=sigma_g_infer) # shape [M, n_phi]\n",
    "    phi_decode_index = np.argmax(log_likelihood_fr, axis=1)\n",
    "\n",
    "    phi_decode = phi_candidates[phi_decode_index]\n",
    "    # Second step: decode z\n",
    "    prior = prior_function(phi_decode, z_candidates, sigma_phi=sigma_phi_infer)\n",
    "    # plt.plot(prior)\n",
    "    z_est_index = np.argmax(prior)\n",
    "    z_decode = z_candidates[z_est_index]\n",
    "    return z_decode, phi_decode"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.19295613 0.19295613 0.19295613] 0.2009713795376621 [0.22407002 0.16805252 0.13444201]\n"
     ]
    }
   ],
   "source": [
    "sigma_g = np.sqrt(np.sqrt(np.pi)*Ag**3*rho_g*tau_g/(a_g*alpha_g)) * 0.01\n",
    "sigma_phi = np.sqrt(8*np.pi*Ag*tau_g/(Lambda*J_pg*rho_p*Rp)) * 0.01\n",
    "sigma_p = np.sqrt(np.sqrt(np.pi)*Ap**3*rho_p*tau_p/(a_p*alpha_p)) * 0.01\n",
    "print(sigma_g, sigma_p, sigma_phi)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Trial Progress: 100%|██████████| 200/200 [06:26<00:00,  1.93s/it]\n",
      "Trial Progress: 100%|██████████| 200/200 [06:06<00:00,  1.83s/it]\n",
      "Trial Progress: 100%|██████████| 200/200 [06:13<00:00,  1.87s/it]\n",
      "Trial Progress: 100%|██████████| 200/200 [06:25<00:00,  1.93s/it]\n",
      "Trial Progress: 100%|██████████| 200/200 [07:05<00:00,  2.13s/it]\n",
      "Trial Progress: 100%|██████████| 200/200 [07:16<00:00,  2.18s/it]\n",
      "Trial Progress: 100%|██████████| 200/200 [07:25<00:00,  2.23s/it]\n",
      "Sigma Progress: 100%|██████████| 7/7 [46:58<00:00, 402.64s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total runtime: 2818.46 seconds\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import time\n",
    "from tqdm import tqdm\n",
    "sigma_number = 7\n",
    "# 主程序\n",
    "neural_noise_sigma = np.linspace(0.05, 0.2, sigma_number)\n",
    "trial_num = 200\n",
    "z_truth = 30\n",
    "phi_truth = np.mod(z_truth / Lambda, 1) * 2 * np.pi \n",
    "error_MAP = np.zeros(sigma_number)\n",
    "error_net = np.zeros(sigma_number)\n",
    "std_error_MAP = np.zeros(sigma_number)\n",
    "std_error_net = np.zeros(sigma_number)\n",
    "\n",
    "start_time = time.time()  # 记录开始时间\n",
    "\n",
    "for n_sigma in tqdm(range(sigma_number), desc=\"Sigma Progress\"):\n",
    "    z_decode_MAP = np.zeros(trial_num)\n",
    "    z_decode_net = np.zeros(trial_num)\n",
    "    sigma_phi = np.array([31.7, 23.8, 19]) * neural_noise_sigma[n_sigma] / 25\n",
    "\n",
    "    for i in tqdm(range(trial_num), desc=\"Trial Progress\", leave=True):\n",
    "        z_truth = 30\n",
    "        phi_truth = np.mod(z_truth / Lambda, 1) * 2 * np.pi \n",
    "        z_e = z_truth\n",
    "        psi = phi_truth + sigma_phi * np.random.randn(M)\n",
    "        Ig = np.zeros((M, num_g))\n",
    "        for j in range(M):\n",
    "            dis_theta = circ_dis(theta, psi[j])\n",
    "            Ig[j, :] = np.exp(-dis_theta ** 2 / (4 * a_g[j] ** 2)) + neural_noise_sigma[n_sigma] * np.random.randn(num_g)\n",
    "        dis_x = x - z_e\n",
    "        Ip = 0 * (np.exp(-dis_x ** 2 / (4 * a_p ** 2)) + neural_noise_sigma[n_sigma] * np.random.randn(num_p))\n",
    "        z_decode_n, _, up, rp, ug, rg = Net_decoding(z_truth, phi_truth, Ip, Ig)\n",
    "\n",
    "        max_up = np.max(up, axis=1)\n",
    "        max_rp = np.max(rp, axis=1)\n",
    "        Ap = np.max(max_up[-1])\n",
    "        Rp = np.max(max_rp[-1])\n",
    "        Ag = np.zeros(M,)\n",
    "        Rg = np.zeros(M,)\n",
    "        for mi in range(M):\n",
    "            max_ug = np.max(ug[:, mi, :], axis=1)\n",
    "            max_rg = np.max(rg[:, mi, :], axis=1)\n",
    "            Ag[mi] = np.max(max_ug[-1])\n",
    "            Rg[mi] = np.max(max_rg[-1])\n",
    "\n",
    "        z_decode_m, _ = Simplified_PSC_MAP_decoder(activation_gs=Ig, Ap=Ap, Rp=Rp, Ag=Ag)\n",
    "\n",
    "        z_decode_MAP[i] = z_decode_m\n",
    "        z_decode_net[i] = z_decode_n[-1]\n",
    "\n",
    "    error_MAP[n_sigma] = np.var(z_decode_MAP - 30)\n",
    "    error_net[n_sigma] = np.var(z_decode_net - 30)\n",
    "    # std_error_MAP[n_sigma] = np.std(np.abs(z_decode_MAP - 30))\n",
    "    # std_error_net[n_sigma] = np.std(np.abs(z_decode_net - 30))\n",
    "\n",
    "end_time = time.time()  # 记录结束时间\n",
    "print(f\"Total runtime: {end_time - start_time:.2f} seconds\")\n",
    "\n",
    "# 保存数据为npz文件\n",
    "np.savez('results/cth/MAP_vs_Net_comparison.npz', error_MAP=error_MAP, error_net=error_net, \n",
    "         std_error_MAP=std_error_MAP, std_error_net=std_error_net)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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fsHziUQiLF26FK+++irc3d3h6OiI+Ph4FBQUCLZbtGgRAgICYG9vj549e2LHjh0QiUTIzs6uNS4AkEqlmD59Onx8fNCsWTNER0erjUXPGrH6jHTHmDGEh4dTeHg43b17l1q0aEGdO3emiooK5Xp/f38aP368cvmVV14hGxsbSkxMpL1791Jqaio5ODhQdHS0ckz18ePHk7+/PxFVD8Pr5eVFw4YNo71799LOnTupR48e5OLiQkVFRUREdODAARKLxRQbG0s7duygb7/9lvz8/Khjx461jnQpk8moffv21KdPH9q1axft3buXBg4cSDY2NnT+/Hm6efMmJSUlEQBKT0+nCxcuKF+PjY0NTZ06lfbu3Us//fQTlZWV0dNPP03Nmzen//znP7Rr1y566aWXyMbGhvbv3698TgDk4uJCEyZMoD179tDXX39NdnZ2NGLECOU2s2fPJisrK/roo49oz5499N5775G9vb1yhMm64rKysqL+/ftTRkYGrVy5klxdXalbt271/yMzs8AJgJkdRQIgItq+fTsBoMTEROX6mgng9OnTBIDmzZsnOMaaNWsIAO3evZuIhAng8OHDBIB++ukn5fYXLlygDz/8UDmhxnPPPUedOnWiqqoq5TZ//vknWVtb05dffqkx7hs3bhAAWrt2rbKsqKiI3nvvPfrf//5HRJqHVfb39yc/Pz+SyWTKsn//+98EgH755RdlmVwup969e1NoaKiyDAA9//zzgjgmTpxIjo6ORER0//59sre3p7fffluwzeTJkwVDDNcWV6tWraiyslJZlpiYSACUY9Ozxo2bgJhZi4uLw5gxY5CamooTJ06orT9w4AAAYPTo0YLyESNGwNraGllZWWr7dOrUCV5eXoiLi8OUKVOwY8cOPPHEE1iwYAFatWqF0tJS/PLLLxg4cCCICFVVVaiqqkJAQADat2+PzMxMjbE2b94cHTp0wGuvvYYJEyZgw4YNICIsXrwYnTp1qvN1dujQQTCh9/79++Hj44Nu3bopn18mkyEuLg7Hjh3D3bt3lduqzsvr6+uLBw8eAAAOHz6MsrIyvPzyy4JtRo4cWWc8Cs8++yzEYrFyOSAgAED1dJCs8eMEwMze559/Dk9PT429gu7cuQMA8PHxEZTb2NjA09NT44nK0dEROTk5GDhwINavX49BgwbBy8sLkydPRnl5Oe7evQu5XI7U1FSIxWLB4/fff8f169c1xikSiZCZmYkJEyYgIyMDI0aMgLe3N4YPH66MszbNmzcXLN++fRv5+flqz//hhx8CAG7cuKHc1sHBQbCvlZWVsj+/4pqFt7e3YBvV+qpNs2bN1I4NAHK5XKv9mXmzMXUAjD2Om5sbli1bhsGDB2PevHmCde7u7gCA/Px8wUVeqVSKwsLCWicmDwoKwpo1ayCTyXD06FGsWbMGX3/9NQICAvDmm29CJBLhvffe0/hJWfWEW1OLFi2wdOlSfPXVVzh58iQ2b96M+fPnw93dHV9//bXWr9nV1RVPPvkkvvvuO43r27Rpo9VxfH19AQA3b95EUFCQsvzmzZtax8KaLv4GwBqFF198EaNGjUJKSoqgJ054eDgAIC0tTbD9+vXrIZPJ8Pzzz6sda/PmzfDy8kJ+fj6sra0RFhaGpUuXwtXVFXl5eXByckLXrl1x9uxZhIaGKh8dO3bErFmzkJ2drTHGw4cPo3nz5vj1118hEokQEhKCefPmoXPnzsjLywMAWFtba/V6w8PDkZeXB29vb0EM+/btw4IFC2Bjo91nt5CQELi4uCA9PV1QvmXLFsGytnGxpoW/AbBG44svvsD+/fsFXRw7dOiA8ePHY9asWSgrK0NERARyc3Mxa9YsREZGIjY2Vu04PXv2hEwmw+DBg5GQkABnZ2ds2LABxcXFeOmllwAAn376KQYMGIDRo0dj9OjRkMlkWLRoEY4cOYKkpCSN8XXp0gUODg4YO3YsZs2aBR8fH+zbtw+5ubl49913AVR/sgeA9PR0DBgwAMHBwRqPNXHiRHz55Zfo168fPv74Y/j5+SEzMxOpqal4++23Be3ydXFycsL06dPxySefwMHBAREREThw4IDy24iiSUfbuFgTY+KL0IypqdkLSFV6ejoBEHQDraqqonnz5lFAQACJxWJq3bo1zZgxg8rKypTb1OwFRER09OhRio6OJnd3d7Kzs6PQ0FBKT08XPNe+ffuoV69eZG9vTy4uLtSnTx/KycmpM/Zz587R0KFDydvbm2xtbaljx470zTffKNffu3ePoqKiyNbWlgYMGEBE6t1aFQoKCmjSpEnk7e1NEomEgoKCaMGCBYLeQgBo5syZgv1mzpxJNf+15XI5zZ07l3x9fcnW1pZ69epFS5YsIQB0/PhxneKqbXJ41jjxpPCMNWFVVVX47rvvEBkZiVatWinLv/rqK7zzzju4ffu28tM/szycABhr4jp27AiJRIKkpCR4enri5MmTSE5OxpAhQ7By5UpTh8dMiBMAY03cpUuXMGPGDGRlZaGoqAh+fn4YO3YsZsyYofW1BNY0cQJgjDELxd1AGWPMQnECYIwxC8UJgDHGLBQnAMYYs1CcABhjzEJxAmCMMQvFCYAxxiwUJwDGGLNQnAAYY8xC/T8Kq9ij+mjX5gAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "# data = np.load('results/cth/MAP_vs_Net_comparison.npz')\n",
    "# sigma_number = 7\n",
    "# neural_noise_sigma = np.linspace(0.05, 0.4, sigma_number)\n",
    "# error_MAP = data['error_MAP']\n",
    "# error_net = data['error_MAP']\n",
    "plt.rcParams.update({'font.size': 12, 'font.family': 'Arial'})\n",
    "plt.figure(figsize=(4, 4))\n",
    "\n",
    "plt.plot(neural_noise_sigma, error_MAP, label='MAP decoding', color='blue', linewidth=3)\n",
    "# plt.plot(g_noise_stre, z_decode_psc_gop_mean, label='PSC GOP', color='green')\n",
    "# plt.fill_between(neural_noise_sigma, error_MAP - std_error_MAP/30, error_MAP + std_error_MAP/30, color='blue', alpha=0.3)\n",
    "\n",
    "plt.plot(neural_noise_sigma, error_net, label='Network decoding', color='red', linewidth=3)\n",
    "# plt.fill_between(neural_noise_sigma, error_net - std_error_net/30, error_net + std_error_net/30, color='red', alpha=0.3)\n",
    "\n",
    "plt.xlabel('Noise strength')\n",
    "plt.ylabel('Decoding Error')\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.savefig('figures_cth/net_decoding.pdf')\n",
    "plt.show()"
   ]
  }
 ],
 "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.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
