{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "import scipy.stats as stats\n",
    "import lstnn.nmar.nmar_transformer as nmar_transformer\n",
    "import lstnn.nmar.dataset as nmar_dataset\n",
    "plt.rcParams['font.sans-serif'] = \"Arial\"\n",
    "sns.set_style(\"ticks\")\n",
    "import numpy as np\n",
    "import os\n",
    "%matplotlib inline\n",
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "import h5py\n",
    "from torchmetrics import MeanSquaredError, R2Score\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\", category=UserWarning, module=\"torch\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 109,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1d-fixed\n",
      "2d-fixed\n",
      "relative\n",
      "rope\n",
      "learn-0.1\n",
      "learn-0.2\n",
      "learn-1.0\n",
      "learn-2.0\n",
      "random\n",
      "c-nope\n",
      "nope\n"
     ]
    }
   ],
   "source": [
    "model_label = 'nmar-Transformer'\n",
    "curriculum = 'All'\n",
    "seeds = [7785, 195, 6914, 29, 6312, 2235, 6068, 9742, 8880, 2197, 669, 6256, 3309, 2541, 8643]\n",
    "device = 'mps'\n",
    "mask_train = 0.50\n",
    "\n",
    "# model params\n",
    "nblocks = [4] #, 5]\n",
    "attnheads = [1]# 4, 8]\n",
    "# wdecays = [0.0, 0.01, 0.05, 0.1, 0.2] #wdecay = 0.0\n",
    "wdecays = [0.0] #, 0.1] #wdecay = 0.0\n",
    "dropout = 0.0\n",
    "hidden_size = 64 #160\n",
    "learning_rate = 0.0001\n",
    "last_iteration = 20000\n",
    "checkpoint_freq = 2000\n",
    "\n",
    "positional_encodings = {\n",
    "      'pe-1dpe_':'absolute',\n",
    "      'pe-2dpe_':'absolute2d',\n",
    "      'pe-shaw_':'relative',\n",
    "      'pe-rope2_':'rope2',\n",
    "      'pe-learn-0.1_':'learn',\n",
    "      'pe-learn-0.2_':'learn',\n",
    "      'pe-learn-1.0_':'learn',\n",
    "      'pe-learn-2.0_':'learn',\n",
    "      'pe-rndpe_':'rndpe',\n",
    "      'pe-cnope_':'cnope',\n",
    "      'pe-nope_':'nope'\n",
    "}\n",
    "pe_labels = {\n",
    "      'pe-1dpe_':'1d-fixed',\n",
    "      'pe-2dpe_':'2d-fixed',\n",
    "      'pe-shaw_':'relative',\n",
    "      'pe-rope2_':'rope',\n",
    "      'pe-learn-0.1_':'learn-0.1',\n",
    "      'pe-learn-0.2_':'learn-0.2',\n",
    "      'pe-learn-1.0_':'learn-1.0',\n",
    "      'pe-learn-2.0_':'learn-2.0',\n",
    "      'pe-rndpe_':'random',\n",
    "      'pe-cnope_':'c-nope',\n",
    "      'pe-nope_':'nope'\n",
    "}\n",
    "\n",
    "# pe_string = '' #'pe-rndpe_' #'pe-nope_' #'pe-2dpe_' #''\n",
    "# pe_param = 'absolute1d' #'rndpe' #'absolute2d'\n",
    "\n",
    "# load results and model\n",
    "\n",
    "df = pd.DataFrame()\n",
    "for pe in positional_encodings:\n",
    "      pestr = pe_labels[pe]\n",
    "      print(pestr)\n",
    "      for layer in nblocks:\n",
    "            for attnhead in attnheads:\n",
    "                  for wdecay in wdecays:\n",
    "                        resultdir = f\"../results/nmar/\"\n",
    "                        modelname = f\"model-{model_label}_\" \\\n",
    "                                    f\"mask-{mask_train}_\" \\\n",
    "                                    f\"{pe}\" \\\n",
    "                                    f\"nl-{layer}_\" \\\n",
    "                                    f\"do-{dropout}_\" \\\n",
    "                                    f\"wd-{wdecay}_\" \\\n",
    "                                    f\"at-{attnhead}_\" \\\n",
    "                                    f\"hs-{hidden_size}_\" \\\n",
    "                                    f\"lr-{learning_rate}\"\n",
    "                        for seed in seeds:\n",
    "                              # print(f'{resultdir}{modelname}/df_{seed}.csv')\n",
    "                              # _df = pd.read_csv(f'{resultdir}{modelname}/df_{seed}.csv')\n",
    "                              try:\n",
    "                                    _df = pd.read_csv(f'{resultdir}{modelname}/df_{seed}.csv')\n",
    "                                    _df.insert(1,'PE',np.repeat(pestr,len(_df)))\n",
    "                                    df = pd.concat([df, _df])\n",
    "                              except:\n",
    "                                    # _df = pd.read_csv(resultdir + modelname + checkpoint +'.csv')\n",
    "\n",
    "                                    continue\n",
    "\n",
    "df_ds = df.copy()\n",
    "# df_ds = pd.DataFrame()\n",
    "# # get max epoch\n",
    "# for i in df.epoch.unique():\n",
    "#     data = df.loc[df.epoch == i]\n",
    "#     df_ds = pd.concat([df_ds, data])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 190x190 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(1.9,1.9))\n",
    "tmpdf = df_ds.loc[(df_ds.Condition=='Train') & (df_ds.Iteration<=10000)]\n",
    "# tmpdf = tmpdf.loc[tmpdf.Iteration<=50000]\n",
    "ax = sns.lineplot(x=\"Iteration\",y=\"Loss\",hue='PE',data=tmpdf,legend=True)\n",
    "handles, labels = ax.get_legend_handles_labels()\n",
    "l = plt.legend(handles[:], labels[:], loc=1, borderaxespad=0., prop={'size': 4.5})\n",
    "plt.xlabel('Training step',fontsize=8)\n",
    "plt.ylabel(\"MSE\",fontsize=8)\n",
    "plt.xticks(fontsize=8)\n",
    "plt.yticks(fontsize=8)\n",
    "# plt.ylim(0.5,1.5)\n",
    "plt.title('Training loss',fontsize=10)\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "# plt.savefig(f'../figures/manuscript_figures_v2/appendix/fmri_supp/loss_trajectory_mask{mask_train}.pdf',transparent=True)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load actual models, and measure train and validation across masks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 110,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading in training data\n",
      "Loading in validation data\n"
     ]
    }
   ],
   "source": [
    "print('Loading in training data')\n",
    "train_dataset = nmar_dataset.DatasetSampler(\n",
    "    n_timepoints=5000,\n",
    "    noise_level=0.2 # training data cohort\n",
    "    )\n",
    "#### Load in validation data\n",
    "print('Loading in validation data')\n",
    "valid_dataset = nmar_dataset.DatasetSampler(\n",
    "    n_timepoints=5000,\n",
    "    noise_level=0.2 # training data cohort\n",
    "    )"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Instantiate dataloader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 111,
   "metadata": {},
   "outputs": [],
   "source": [
    "device='mps'\n",
    "\n",
    "train_batch_size=2048\n",
    "train_dataloader = torch.utils.data.DataLoader(train_dataset, batch_size=train_batch_size, shuffle=True)\n",
    "train_dataloader_iter = iter(train_dataloader)\n",
    "\n",
    "valid_batch_size=2048\n",
    "valid_dataloader = torch.utils.data.DataLoader(valid_dataset, batch_size=valid_batch_size, shuffle=True)\n",
    "valid_dataloader_iter = iter(valid_dataloader)\n",
    "\n",
    "\n",
    "train_batch = next(train_dataloader_iter).float().to(device)\n",
    "valid_batch = next(valid_dataloader_iter).float().to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1d-fixed\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_53422/387745478.py:73: FutureWarning: The behavior of DataFrame concatenation with empty or all-NA entries is deprecated. In a future version, this will no longer exclude empty or all-NA columns when determining the result dtypes. To retain the old behavior, exclude the relevant entries before the concat operation.\n",
      "  df_eval = pd.concat([df_eval,row],ignore_index=True)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2d-fixed\n",
      "relative\n",
      "rope\n",
      "learn-0.1\n",
      "learn-0.2\n",
      "learn-1.0\n",
      "learn-2.0\n",
      "random\n",
      "c-nope\n",
      "nope\n"
     ]
    }
   ],
   "source": [
    "mask_levels = [0.5,0.6,0.7,0.8,0.9]\n",
    "# mask_levels = [0.9]\n",
    "attnheads = 1\n",
    "wdecay = 0.0\n",
    "layers = 4\n",
    "model_pes = {}\n",
    "mask_train = mask_train\n",
    "\n",
    "#eval metrics\n",
    "mse_metric = MeanSquaredError().to(device)\n",
    "r2_metric = R2Score(num_outputs=15).to(device)\n",
    "#\n",
    "df_eval = pd.DataFrame(columns=['Iteration', 'MSE', 'Condition', 'Mask', 'R^2', 'PE', 'Seed'])\n",
    "# iteration = 100000\n",
    "iteration = 50000\n",
    "for pe in positional_encodings:\n",
    "    pestr = pe_labels[pe]\n",
    "    pe_param = positional_encodings[pe]\n",
    "    print(pestr)\n",
    "    if pe_param in ['learn','rndpe','absolute']:\n",
    "        model_pes[pestr] = []\n",
    "        \n",
    "    resultdir = f\"../results/nmar/\"\n",
    "    modelname = f\"model-{model_label}_\" \\\n",
    "                f\"mask-{mask_train}_\" \\\n",
    "                f\"{pe}\" \\\n",
    "                f\"nl-{layers}_\" \\\n",
    "                f\"do-{dropout}_\" \\\n",
    "                f\"wd-{wdecay}_\" \\\n",
    "                f\"at-{attnheads}_\" \\\n",
    "                f\"hs-{hidden_size}_\" \\\n",
    "                f\"lr-{learning_rate}\"\n",
    "    for seed in seeds:\n",
    "        try: \n",
    "            checkpoint = f\"s-{seed}_\" \\\n",
    "                        f\"i-{iteration}\" \n",
    "            torch.manual_seed(seed)\n",
    "            model = nmar_transformer.Transformer(\n",
    "                        input_dim=1,\n",
    "                        output_dim=1,\n",
    "                        max_tokens=15,\n",
    "                        nblocks=layer,\n",
    "                        nhead=attnhead,\n",
    "                        dropout=dropout,\n",
    "                        embedding_dim=hidden_size,\n",
    "                        positional_encoding=pe_param,\n",
    "                        pe_init=1.0)\n",
    "            model = model.to(device=torch.device(device))\n",
    "            model.load_state_dict(torch.load(f'{resultdir}/{modelname}/{checkpoint}.pt',map_location=torch.device(device) ))\n",
    "            if pe_param in ['absolute']:\n",
    "                model_pes[pestr].append(model.blocks[0].pe.pe.detach().cpu())\n",
    "            if pe_param in ['learn','rndpe']:\n",
    "                model_pes[pestr].append(model.blocks[0].pe.positional_encoding.detach().cpu())\n",
    "            for mask in mask_levels:\n",
    "\n",
    "                mask_tensor = torch.rand(train_batch.shape)\n",
    "                mask_tensor = mask_tensor > mask\n",
    "                masked_train_batch = torch.mul(train_batch,mask_tensor.to(device))\n",
    "                masked_valid_batch = torch.mul(valid_batch,mask_tensor.to(device))\n",
    "                train_out = model(masked_train_batch)\n",
    "                valid_out = model(masked_valid_batch)\n",
    "                flip_mask = mask_tensor==False\n",
    "                # write training data\n",
    "                row = pd.DataFrame({\n",
    "                    'Iteration': iteration,\n",
    "                    'Condition': 'Train',               \n",
    "                    'MSE': mse_metric(torch.squeeze(train_batch[flip_mask]), torch.squeeze(train_out[flip_mask])).detach().item(),               \n",
    "                    'R^2': r2_metric(torch.squeeze(train_batch[flip_mask]), torch.squeeze(train_out[flip_mask])).detach().item(),               \n",
    "                    'PE': pestr,               \n",
    "                    'Seed': seed,\n",
    "                    'Mask': mask,\n",
    "                    },index=[0])\n",
    "                df_eval = pd.concat([df_eval,row],ignore_index=True)\n",
    "                # write validation data\n",
    "                row = pd.DataFrame({\n",
    "                    'Iteration': iteration,\n",
    "                    'Condition': 'Validation',               \n",
    "                    'MSE': mse_metric(torch.squeeze(valid_batch[flip_mask]), torch.squeeze(valid_out[flip_mask])).item(),               \n",
    "                    'R^2': r2_metric(torch.squeeze(valid_batch[flip_mask]), torch.squeeze(valid_out[flip_mask])).item(),               \n",
    "                    'PE': pestr,               \n",
    "                    'Seed': seed,\n",
    "                    'Mask': mask,\n",
    "                    },index=[0])\n",
    "                df_eval = pd.concat([df_eval,row],ignore_index=True)\n",
    "                #\n",
    "                torch.mps.empty_cache()\n",
    "        except:\n",
    "            continue"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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o4UQaffq4MGDYvz48eLu3bvi+vXr4o8//hD79+9/adk55T/tnH8WGX33fv368e+//zJ37lxq1apFjx49WL16NdHR0ZQrV45jx46ZprnI5XKTf+N1lPWmycmgbt26dO3alaioKG7cuMEHH3yAjY0Nly9f5urVq7Rt25Zbt24xYcIERo8ezbfffoter2fr1q3Mnz//hYIsvb29sbW15f79+1y6dIkWLVowffp0OnTowNGjR/n1118pXbo0JUqUIDw8nN27d7N27VqCg4PR6XRER0fnSE6bNm2YPn06ffr04d1336Vu3bo0btyYqVOnmuLIDh48SKdOnQgPD+fu3bsMHTqUjh07Ai83mqpQKEz34c6dOxw9ehS5XE58fDwRERFs3ryZM2fOoFAoOHnyJKNHj0Yul7NhwwamTJmSJ1OGpK7ic9ixYwdbt27l3LlznDx5knPnzvHXX39Rt25dOnToAECvXr2oVasWHTt2pHjx4kDuHMF5JetNk5OZwMBAypYti0aj4YcffsDLyws3Nzfu3btHeHg433zzDYsXLyYsLAx3d3cqV67Mhx9+mCtZQUFBprinlStXotPpOHHiBCtXrqRy5crMmDGDfv36cebMGZYuXUrjxo1ZuHAhjRo1onr16rmSGRERQcGCBRk/fjyBgYFs2LCBR48e8b///Y9ffvmFokWL4uHhkauys+PYsWP4+fnh6+vLhQsXWLduHR4eHrRt2xa5XM65c+c4cuQIffv25eOPP2by5Mn07duX0qVLW6wO2SEprhwQFBTEnTt3cHNzY/PmzZQpU4bu3bujVqv5+eef+fPPP6lduzYXL16kdu3ajBgx4rWX9abJycqOHTs4cOAAn3zyCffv3+fmzZv07t0bb29vjh07ZrKSevfubRr2z40zPS4ujmHDhlG4cGG6d+/OsWPHOHnyJHPmzMHf358///yTsmXLUrZsWRwcHHj77bdJTk7Gycnphduk1WpRq9X8+eefjB07lqNHj+Ls7Mzw4cNxcHBg+vTpdO7cmcmTJ1O1alXAMiOPGR8DwDS9yt7entWrV3P+/Hl+/PFHJk2ahL+/P25ubgwYMIB69eq9lMznkmed0jeA48ePi6lTp4rAwEAhhBAnT54UPXv2FIcPHxZCpPlPevToIe7fv59vZL1pcjJz48YNYTAYxNdffy2WLVsmhBAiLi5OLF++XIwf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",
      "text/plain": [
       "<Figure size 300x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "plt.figure(figsize=(3.,2))\n",
    "# tmpdf = df_eval.loc[df_eval.Mask==0.9]\n",
    "# tmpdf = df_eval.loc[df_eval.Mask==0.5]\n",
    "orderdf = tmpdf.groupby('PE').mean(numeric_only=True).reset_index()\n",
    "order = orderdf.sort_values('MSE',ascending=False)\n",
    "ax = sns.boxplot(x=\"PE\",y=\"MSE\",hue=\"Condition\",data=tmpdf,fliersize=1,palette=\"Set1\",order=order.PE.values)\n",
    "sns.stripplot(x='PE',y='MSE',hue='Condition',data=tmpdf,dodge=True,palette=\"Set1\",size=3,order=order.PE.values)\n",
    "plt.legend(loc=4,prop={'size': 5})\n",
    "handles, labels = ax.get_legend_handles_labels()\n",
    "l = plt.legend(handles[-2:], labels[-2:], borderaxespad=0.2, prop={'size': 6})\n",
    "plt.yticks(fontsize=8)\n",
    "plt.xticks(fontsize=8,rotation=-35)\n",
    "plt.title(\"Predicting masked (90%) brain activity\",fontsize=10)\n",
    "# plt.legend().remove()\n",
    "sns.despine()\n",
    "plt.ylabel(\"MSE\",fontsize=8)\n",
    "plt.xlabel(\"PE\",fontsize=8)\n",
    "plt.tight_layout()\n",
    "# plt.savefig('../figures/manuscript_figures_v2/fmri_networks/train_validation_MSE.pdf',transparent=True)\n",
    "# plt.savefig(f'../figures/manuscript_figures_v2/appendix/fmri_supp/train_validation_MSE_mask{mask_train}.pdf',transparent=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compute distance metrics between positional encodings of tokens"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 166,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1d-fixed\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 15/15 [00:00<00:00, 28.21it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "learn-0.1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 15/15 [00:00<00:00, 26.42it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "learn-0.2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 15/15 [00:00<00:00, 25.89it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "learn-1.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 15/15 [00:00<00:00, 22.53it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "learn-2.0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 15/15 [00:00<00:00, 24.73it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "random\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 15/15 [00:00<00:00, 24.69it/s]\n"
     ]
    }
   ],
   "source": [
    "from scipy.linalg import orthogonal_procrustes\n",
    "from tqdm import tqdm\n",
    "\n",
    "\n",
    "n_tokens = 15\n",
    "embedding_distance = {}\n",
    "embedding_distance_corr = {}\n",
    "for pe in model_pes:\n",
    "    print(pe)\n",
    "    embedding_distance[pe] = np.zeros((len(seeds),n_tokens,n_tokens))\n",
    "    embedding_distance_corr[pe] = np.zeros((len(seeds),n_tokens,n_tokens))\n",
    "    embedding_matrix = np.squeeze(np.asarray(model_pes[pe]))\n",
    "    for i in tqdm(range(n_tokens)):\n",
    "        for j in range(n_tokens):\n",
    "            if j>i:\n",
    "                pe_i = embedding_matrix[:,i,:]\n",
    "                pe_j = embedding_matrix[:,j,:]\n",
    "                R, sca = orthogonal_procrustes(pe_i,pe_j)\n",
    "                new_A = (pe_i @ R)\n",
    "                embedding_distance[pe][:,i,j] = np.linalg.norm(new_A - pe_j,axis=1)\n",
    "\n",
    "                # correlation\n",
    "                for s in range(len(seeds)):\n",
    "                    embedding_distance_corr[pe][s,i,j] = np.corrcoef(new_A[s,:],pe_j[s,:])[0,1]\n",
    "                # covariance\n",
    "                # embedding_distance_corr[pe][i,j] = np.cov(new_A.reshape(-1),pe_j.reshape(-1))[0,1]\n",
    "\n",
    "#     ['learn-0.2'][0].shape\n",
    "\n",
    "# R, sca = orthogonal_procrustes(tmp.cpu().detach().numpy(), pe2dembedding.detach().cpu().numpy())\n",
    "\n",
    "# R, sca = orthogonal_procrustes(tmp.cpu().detach().numpy(), pe2dembedding.detach().cpu().numpy())\n",
    "# new_A = (tmp.cpu().detach().numpy() @ R)\n",
    "# df_disparity['Initialization'].append(init)\n",
    "# df_disparity['Norm'].append(np.linalg.norm(new_A - pe2dembedding.detach().cpu().numpy()))\n",
    "# df_disparity['Disparity'] = 0"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# compute modularity and segregatin *across* subjects"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 218,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 125x140 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# network_org = np.loadtxt('cortex_parcel_network_assignments.txt')\n",
    "# indsort = np.loadtxt('cortex_community_order.txt',dtype=int) - 1 \n",
    "# indsort.shape = (len(indsort),1)\n",
    "num_clusters=3\n",
    "cluster_size=5\n",
    "clusters = [list(range(i * cluster_size, (i + 1) * cluster_size)) for i in range(num_clusters)]\n",
    "network_org = []\n",
    "label = 1\n",
    "for c in range(num_clusters):\n",
    "    for i in range(cluster_size):\n",
    "        network_org.append(label)\n",
    "    label += 1\n",
    "network_org = np.asarray(network_org)\n",
    "\n",
    "#### visualize clusters\n",
    "total_nodes = num_clusters * cluster_size\n",
    "network_struct = np.zeros((total_nodes,total_nodes))\n",
    "node = 0\n",
    "for i in range(num_clusters):\n",
    "    network_struct[node:(node+cluster_size),node:(node+cluster_size)] = 1\n",
    "    node += cluster_size\n",
    "\n",
    "plt.figure(figsize=(1.25,1.4))\n",
    "plt.title(\"Network\",fontsize=8)\n",
    "ax = sns.heatmap(network_struct,square=True,center=0,cmap='seismic',cbar=False)\n",
    "ax.invert_yaxis()\n",
    "plt.xticks(np.arange(0,16,5),np.arange(0,16,5),rotation=0,fontsize=6);\n",
    "plt.yticks(np.arange(0,16,5),np.arange(0,16,5),rotation=0,fontsize=6);\n",
    "plt.ylabel('Nodes',fontsize=6);\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'../figures/manuscript_figures_v3/appendix/nmar_expt/clusters.pdf',transparent=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 222,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x140 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# visualize dataset\n",
    "train_batch_size=50\n",
    "train_dataloader = torch.utils.data.DataLoader(train_dataset, batch_size=train_batch_size, shuffle=False)\n",
    "sample_data = next(iter(train_dataloader))\n",
    "ind = np.random.choice(np.arange(15),3,replace=False)\n",
    "plt.figure(figsize=(2,1.4))\n",
    "plt.title('Sample network timeseries',fontsize=8)\n",
    "sns.lineplot(sample_data[:,ind,0],legend=False,palette='Set2')\n",
    "sns.despine(trim=True)\n",
    "plt.xticks(fontsize=6);\n",
    "plt.yticks(fontsize=6);\n",
    "plt.ylabel(r\"$x_i(t)$\",fontsize=8,labelpad=0.2)\n",
    "plt.xlabel(r\"$t$\",fontsize=8,labelpad=0.2)\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'../figures/manuscript_figures_v3/appendix/nmar_expt/data_example.pdf',transparent=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compute modularity, segregation, and plot networks"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 224,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/tito/mambaforge/envs/lstnn/lib/python3.9/site-packages/bct/algorithms/modularity.py:1653: RuntimeWarning: divide by zero encountered in scalar divide\n",
      "  d1 = 1 / s1  # dQ=dQ0/s0-dQ1/s1\n",
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_53422/2450977951.py:49: FutureWarning: The behavior of DataFrame concatenation with empty or all-NA entries is deprecated. In a future version, this will no longer exclude empty or all-NA columns when determining the result dtypes. To retain the old behavior, exclude the relevant entries before the concat operation.\n",
      "  df_netstats = pd.concat([df_netstats,row],ignore_index=True)\n"
     ]
    }
   ],
   "source": [
    "import bct \n",
    "from sklearn.metrics.pairwise import cosine_similarity\n",
    "\n",
    "\n",
    "df_netstats = pd.DataFrame(columns=['PE','Modularity','Segregation','L2Norm','Seed'])\n",
    "triu_ind = np.triu_indices(n_tokens,k=1)\n",
    "for pe in embedding_distance:\n",
    "    mat = np.zeros((len(seeds),n_tokens,n_tokens))\n",
    "    modularity = []\n",
    "\n",
    "    for s in range(len(seeds)):\n",
    "        ####\n",
    "        # Compute matrix\n",
    "        # mat[s] = embedding_distance_corr[pe][s] + embedding_distance_corr[pe][s].T\n",
    "        mat[s] = (embedding_distance[pe][s] + embedding_distance[pe][s].T)\n",
    "        # np.fill_diagonal(mat[s],np.nan)\n",
    "        # l2_norm = np.linalg.norm((mat[s])/np.linalg.norm(mat[s])- network_struct/np.linalg.norm(network_struct))\n",
    "        # l2_norm = stats.pearsonr(mat[s].reshape(-1),network_struct.reshape(-1))[0]\n",
    "        l2_norm = np.dot(mat[s].reshape(-1),network_struct.reshape(-1))/(np.linalg.norm(mat[s]*np.linalg.norm(network_struct)))\n",
    "\n",
    "\n",
    "        mat[s] = 1 - (mat[s] - np.min(mat[s][triu_ind])) / (np.max(mat[s][triu_ind]) - np.min(mat[s][triu_ind]))\n",
    "        # np.fill_diagonal(mat,1)\n",
    "\n",
    "        #### Compute modularity\n",
    "        # np.fill_diagonal(mat[s],np.mean(mat[s][triu_ind]))\n",
    "        np.fill_diagonal(mat[s],1)\n",
    "        tmp, modularity = bct.modularity_und_sign(mat[s],network_org,qtype='smp')\n",
    "\n",
    "\n",
    "        #### Compute segregation per network and then avg\n",
    "        segregation = []\n",
    "        for net in np.unique(network_org):\n",
    "            net_ind = np.where(network_org==net)[0]\n",
    "            out_ind = np.where(network_org!=net)[0]\n",
    "            net_ind.shape = (len(net_ind),1)\n",
    "            out_ind.shape = (len(out_ind),1)\n",
    "            net_in = np.nanmean(mat[s,net_ind,net_ind.T])\n",
    "            net_out = np.nanmean(mat[s,net_ind,out_ind.T])\n",
    "            segregation.append((net_in - net_out) / net_in)\n",
    "\n",
    "        row = pd.DataFrame({\n",
    "            'PE': pe,               \n",
    "            'Modularity': modularity,               \n",
    "            'Segregation': np.mean(segregation),               \n",
    "            'L2Norm': l2_norm,               \n",
    "            'Seed': seeds[s],               \n",
    "            },index=[0])\n",
    "        df_netstats = pd.concat([df_netstats,row],ignore_index=True)\n",
    "    group_mat = np.mean(mat,axis=0)\n",
    "    # np.fill_diagonal(group_mat,np.nan)\n",
    "    # plt.figure(figsize=(3,3))\n",
    "    # ax = sns.heatmap(data=group_mat,cmap='magma',square=True,cbar=False)\n",
    "    # ax.invert_yaxis()\n",
    "    # plt.title(f\"{pe}\",fontsize=10)\n",
    "    # plt.xlabel('Tokens / Brain regions',fontsize=10)\n",
    "    # plt.ylabel('Tokens / Brain regions',fontsize=10)\n",
    "    #"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Plot segregation and modularity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 226,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 185x175 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plt.figure(figsize=(4.5,1.75))\n",
    "# plt.subplot(121)\n",
    "plt.figure(figsize=(1.85,1.75))\n",
    "orderdf = df_netstats.groupby('PE').mean(numeric_only=True).reset_index()\n",
    "order = orderdf.sort_values('Modularity',ascending=False)\n",
    "ax = sns.boxplot(x=\"PE\",y=\"Modularity\",hue='PE',data=df_netstats,order=order.PE.values,palette='magma',fliersize=1)\n",
    "sns.stripplot(x='PE',y='Modularity',hue='PE',data=df_netstats,order=order.PE.values,palette='magma')\n",
    "# plt.legend(loc=4,prop={'size': 5})\n",
    "# handles, labels = ax.get_legend_handles_labels()\n",
    "# l = plt.legend(handles[len(df_ds.PE.unique()):], labels[len(df_ds.PE.unique()):], loc=1, borderaxespad=0.2, prop={'size': 6})\n",
    "plt.yticks(fontsize=6)\n",
    "plt.xticks(fontsize=6,rotation=-35)\n",
    "plt.title(\"Modularity of PE\",fontsize=8)\n",
    "plt.ylabel(\"Modularity\",fontsize=8)\n",
    "plt.xlabel(\"PE\",fontsize=8)\n",
    "# plt.legend().remove()\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'../figures/manuscript_figures_v3/appendix/nmar_expt/pe_recovery.pdf',transparent=True)\n",
    "\n",
    "# plt.figure(figsize=(1.25,1.25))\n",
    "# ax = sns.boxplot(x=\"PE\",y=\"Segregation\",data=df_netstats,palette='magma',hue='PE',fliersize=1)\n",
    "# sns.stripplot(x='PE',y='Segregation',data=df_netstats,palette='magma',hue='PE')\n",
    "# # plt.legend(loc=4,prop={'size': 5})\n",
    "# # handles, labels = ax.get_legend_handles_labels()\n",
    "# # l = plt.legend(handles[len(df_ds.PE.unique()):], labels[len(df_ds.PE.unique()):], loc=1, borderaxespad=0.2, prop={'size': 6})\n",
    "# plt.yticks(fontsize=6)\n",
    "# plt.xticks(fontsize=6,rotation=-35)\n",
    "# plt.title(\"Network clustering of PE\",fontsize=8)\n",
    "# plt.ylabel('Network segregation',fontsize=8)\n",
    "# plt.xlabel(\"PE\",fontsize=8)\n",
    "# # plt.legend().remove()\n",
    "# sns.despine()\n",
    "# plt.tight_layout()\n",
    "# # plt.savefig('../figures/manuscript_figures_v2/fmri_pes/PE_modularity_segregation.pdf',transparent=True)\n",
    "# # plt.savefig(f'../figures/manuscript_figures_v2/appendix/fmri_supp/PE_modularity_segregation_mask{mask_train}.pdf',transparent=True)\n",
    "# # plt.savefig(f'../figures/manuscript_figures_v3/appendix/nmar_expt/data_example.pdf',transparent=True)\n",
    "\n",
    "\n",
    "\n",
    "# plt.figure(figsize=(1.25,1.25))\n",
    "# ax = sns.boxplot(x=\"PE\",y=\"L2Norm\",hue='PE',data=df_netstats,palette='magma',fliersize=1)\n",
    "# sns.stripplot(x='PE',y='L2Norm',hue='PE',data=df_netstats,palette='magma')\n",
    "# # plt.legend(loc=4,prop={'size': 5})\n",
    "# # handles, labels = ax.get_legend_handles_labels()\n",
    "# # l = plt.legend(handles[len(df_ds.PE.unique()):], labels[len(df_ds.PE.unique()):], loc=1, borderaxespad=0.2, prop={'size': 6})\n",
    "# plt.yticks(fontsize=6)\n",
    "# plt.xticks(fontsize=6,rotation=-35)\n",
    "# plt.title(\"Modularity of PE\",fontsize=8)\n",
    "# plt.ylabel(\"Modularity\",fontsize=8)\n",
    "# plt.xlabel(\"PE\",fontsize=8)\n",
    "# # plt.legend().remove()\n",
    "# sns.despine()\n",
    "\n",
    "# plt.savefig(f'../figures/manuscript_figures_v3/appendix/nmar_expt/data_example.pdf',transparent=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "___"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "lstnn",
   "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.9.18"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
