{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 416,
   "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": 417,
   "metadata": {},
   "outputs": [],
   "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",
    "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": 418,
   "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": 419,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.19295613 0.19295613 0.19295613] 0.2009713795376621 [0.22407002 0.16805252 0.13444201]\n"
     ]
    }
   ],
   "source": [
    "\n",
    "\n",
    "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": 420,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.317 0.238 0.19 ]\n",
      "z_decode: 29.88747186796699\n"
     ]
    }
   ],
   "source": [
    "neural_noise_sigma = 0.2\n",
    "z_truth = 30\n",
    "phi_truth = np.mod(z_truth / Lambda, 1) * 2 * np.pi \n",
    "sigma_phi_noise = np.array([31.7,23.8,19])*neural_noise_sigma/20\n",
    "print(sigma_phi_noise)\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_noise * np.random.randn(M)\n",
    "\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 * np.random.randn(num_g)\n",
    "dis_x = x - z_e\n",
    "Ip = np.exp(-dis_x ** 2 / (4 * a_p ** 2)) + neural_noise_sigma * np.random.randn(num_p) \n",
    "z_decode, phi_decode = Simplified_PSC_MAP_decoder(activation_gs=Ig, n_pos=4000, n_phi=100, M=3)\n",
    "print('z_decode:', z_decode)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 421,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Progress: 0.00%\n",
      "Progress: 1.00%\n",
      "Progress: 2.00%\n",
      "Progress: 3.00%\n",
      "Progress: 4.00%\n",
      "Progress: 5.00%\n",
      "Progress: 6.00%\n",
      "Progress: 7.00%\n",
      "Progress: 8.00%\n",
      "Progress: 9.00%\n",
      "Progress: 10.00%\n",
      "Progress: 11.00%\n",
      "Progress: 12.00%\n",
      "Progress: 13.00%\n",
      "Progress: 14.00%\n",
      "Progress: 15.00%\n",
      "Progress: 16.00%\n",
      "Progress: 17.00%\n",
      "Progress: 18.00%\n",
      "Progress: 19.00%\n",
      "Progress: 20.00%\n",
      "Progress: 21.00%\n",
      "Progress: 22.00%\n",
      "Progress: 23.00%\n",
      "Progress: 24.00%\n",
      "Progress: 25.00%\n",
      "Progress: 26.00%\n",
      "Progress: 27.00%\n",
      "Progress: 28.00%\n",
      "Progress: 29.00%\n",
      "Progress: 30.00%\n",
      "Progress: 31.00%\n",
      "Progress: 32.00%\n",
      "Progress: 33.00%\n",
      "Progress: 34.00%\n",
      "Progress: 35.00%\n",
      "Progress: 36.00%\n",
      "Progress: 37.00%\n",
      "Progress: 38.00%\n",
      "Progress: 39.00%\n",
      "Progress: 40.00%\n",
      "Progress: 41.00%\n",
      "Progress: 42.00%\n",
      "Progress: 43.00%\n",
      "Progress: 44.00%\n",
      "Progress: 45.00%\n",
      "Progress: 46.00%\n",
      "Progress: 47.00%\n",
      "Progress: 48.00%\n",
      "Progress: 49.00%\n",
      "Progress: 50.00%\n",
      "Progress: 51.00%\n",
      "Progress: 52.00%\n",
      "Progress: 53.00%\n",
      "Progress: 54.00%\n",
      "Progress: 55.00%\n",
      "Progress: 56.00%\n",
      "Progress: 57.00%\n",
      "Progress: 58.00%\n",
      "Progress: 59.00%\n",
      "Progress: 60.00%\n",
      "Progress: 61.00%\n",
      "Progress: 62.00%\n",
      "Progress: 63.00%\n",
      "Progress: 64.00%\n",
      "Progress: 65.00%\n",
      "Progress: 66.00%\n",
      "Progress: 67.00%\n",
      "Progress: 68.00%\n",
      "Progress: 69.00%\n",
      "Progress: 70.00%\n",
      "Progress: 71.00%\n",
      "Progress: 72.00%\n",
      "Progress: 73.00%\n",
      "Progress: 74.00%\n",
      "Progress: 75.00%\n",
      "Progress: 76.00%\n",
      "Progress: 77.00%\n",
      "Progress: 78.00%\n",
      "Progress: 79.00%\n",
      "Progress: 80.00%\n",
      "Progress: 81.00%\n",
      "Progress: 82.00%\n",
      "Progress: 83.00%\n",
      "Progress: 84.00%\n",
      "Progress: 85.00%\n",
      "Progress: 86.00%\n",
      "Progress: 87.00%\n",
      "Progress: 88.00%\n",
      "Progress: 89.00%\n",
      "Progress: 90.00%\n",
      "Progress: 91.00%\n",
      "Progress: 92.00%\n",
      "Progress: 93.00%\n",
      "Progress: 94.00%\n",
      "Progress: 95.00%\n",
      "Progress: 96.00%\n",
      "Progress: 97.00%\n",
      "Progress: 98.00%\n",
      "Progress: 99.00%\n",
      "Total runtime: 1086.26 seconds\n"
     ]
    }
   ],
   "source": [
    "\n",
    "# 主程序\n",
    "neural_noise_sigma = 0.4\n",
    "trial_num = 500\n",
    "z_truth = 30\n",
    "phi_truth = np.mod(z_truth / Lambda, 1) * 2 * np.pi \n",
    "sigma_phi = np.array([31.7,23.8,19])*neural_noise_sigma/20\n",
    "\n",
    "z_decode_MAP = np.zeros(trial_num)\n",
    "z_decode_net = np.zeros(trial_num)\n",
    "\n",
    "start_time = time.time()  # 记录开始时间\n",
    "\n",
    "for i in range(trial_num):\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 * 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 * 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",
    "    if i % 5 == 0:\n",
    "        print(f\"Progress: {i/trial_num*100:.2f}%\")\n",
    "\n",
    "end_time = time.time()  # 记录结束时间\n",
    "print(f\"Total runtime: {end_time - start_time:.2f} seconds\")\n",
    "\n",
    "\n",
    "# 保存数据为npz文件\n",
    "np.savez('results/cth/MAP_vs_Net_distribution.npz', z_decode_net=z_decode_net, z_decode_MAP=z_decode_MAP)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 443,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Net decoding error: 0.20105807710786114\n",
      "MAP decoding error: 13.013699945379102\n"
     ]
    },
    {
     "data": {
      "image/png": "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Rx8VCoVCNG7cWLzzzjtOqC3dyYgRIwQAm49z584Z9+U9lSZbYUQI3lOp4f2qPqyFESGE+OOPP0TXrl2Fl5eXqFOnjnj55ZeFRqNxTiXJphUrVtz2tXPVqlVCCN5PqSkuLhbz5s0ToaGhQqlUikaNGonY2FjjB7NCOO6eyoRwwAhQIiIiIiKiu8QxI0RERERE5BQMI0RERERE5BQMI0RERERE5BQMI0RERERE5BQMI0RERERE5BQMI0RERERE5BQMI0RERERE5BQMI0RERERE5BQMI0RE5BSjRo2CTCaz+QgICHB2FYmI6D5zd3YFiIjowRUYGIiNGzda3ebh4VHFtSEioqrGMEJERE6jVCrRoUMHZ1eDiIichN20iIjIpUVFRWH48OEYMmQIVCoV+vbti/Pnz0MmkyEpKQmPPPIIatasidWrVwMADh8+jF69eqFWrVpQqVTo378/Tpw4YTzf3r17IZPJsGzZMgQHB6Nu3br47rvvnHR1REQPNraMEBGRU5WXl1stl8vlkMlkAIB169bhmWeewaZNm8z2j4mJwQcffAC1Wo2wsDCkpqaiZ8+eiIyMxMqVK1FSUoL58+ejY8eOOHToEJo3b248dsaMGVi2bBmKi4sRHh5+fy+SiIisYhghIiKn+euvv2yODZk7dy5iY2MB6IPJJ598Ah8fHwDA+fPnAQBPP/00Ro8ebTzmmWeeQUhICHbs2AG5XA4A6NGjB5o2bYpZs2Zh3bp1xn3HjRuHIUOG3I/LIiKiSmIYISIip3nooYewZcsWq9vq169v/L5x48bGIGLq0UcfNX5fUFCAX3/9FXFxccYgAgBqtRr9+vXDtm3bbB5LRETOwTBCREROo1AoEBYWdsf96tate8fynJwcCCEQGBhosV9gYCBycnIqdU4iIqo6HMBORETVglqthkwmw5UrVyy2Xb58meuWEBG5IIYRIiKqFnx8fBAWFoZ169ZBq9Uay3Nzc7F161Y8+eSTTqwdERFZw25aRETkNCUlJTh48KDN7Xc7riMhIQE9e/ZEr1698Oqrr6K0tBQJCQkoKSnBrFmz7rW6RETkYAwjRETkNFeuXLnttLq//vrrXZ2va9eu2L17N+Li4jBs2DAolUpERETgs88+Q8uWLe+1ukRE5GAyIYRwdiWIiIiIiOjBwzEjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFAwjRERERETkFP8PyPkt92SUZ/8AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 800x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "z_decode_all = np.concatenate([z_decode_net, z_decode_MAP])\n",
    "min_z = np.min(z_decode_all)\n",
    "max_z = np.max(z_decode_all)\n",
    "\n",
    "# 设置字体\n",
    "plt.rcParams.update({'font.size': 12, 'font.family': 'Arial'})\n",
    "error_Net = z_decode_net-z_truth\n",
    "error_MAP = z_decode_MAP-z_truth\n",
    "print('Net decoding error:', np.std(error_Net))\n",
    "print('MAP decoding error:', np.std(error_MAP))\n",
    "\n",
    "# Compute the histogram for the PSC and LSC errors\n",
    "counts_psc, bin_edges_psc = np.histogram(error_MAP, bins=np.linspace(-30,30,500))\n",
    "counts_lsc, bin_edges_lsc = np.histogram(error_Net, bins=np.linspace(-30,30,500))\n",
    "\n",
    "# Calculate the bin centers\n",
    "bin_centers_psc = (bin_edges_psc[:-1] + bin_edges_psc[1:]) / 2\n",
    "bin_centers_lsc = (bin_edges_lsc[:-1] + bin_edges_lsc[1:]) / 2\n",
    "\n",
    "# Remove zero points\n",
    "# nonzero_indices_lsc = counts_lsc > 0\n",
    "# bin_centers_lsc = bin_centers_lsc[nonzero_indices_lsc]\n",
    "# counts_lsc = counts_lsc[nonzero_indices_lsc]\n",
    "\n",
    "# Plot the histogram with non-zero points only\n",
    "plt.figure(figsize=(8, 4))\n",
    "plt.plot(bin_centers_lsc, counts_lsc/np.sum(counts_lsc), label='Net decoding Error', color='red')\n",
    "plt.plot(bin_centers_psc, counts_psc/np.sum(counts_psc), label='MAP decoding Error', color='blue')\n",
    "\n",
    "plt.fill_between(bin_centers_psc, counts_psc/np.sum(counts_psc), color='blue', alpha=0.3)\n",
    "plt.fill_between(bin_centers_lsc, counts_lsc/np.sum(counts_lsc), color='red', alpha=0.3)\n",
    "\n",
    "plt.xlabel('Error')\n",
    "plt.ylabel('Frequency')\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.ylim(-0.05,0.25)\n",
    "plt.xlim(-30,30)\n",
    "plt.savefig('figures_cth/MAP_vs_NET.pdf')\n",
    "plt.show()\n",
    "\n",
    "\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.8.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
