{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This demo aims to provide the basic codes for loading experimental sessions in which the neural data is recorded by Cerebus and the EMGs are recorded by DSPW wireless system."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**1. Data files**"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "The data files are put in a folder on FSM server"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['Pop_20210902_PG_baseline_001.mat', 'Pop_20210902_PG_baseline_001.nev', 'Pop_20210902_PG_baseline_001.ns3', 'Pop_20210902_PG_baseline_001.pkl', 'Pop_20210902_PG_baseline_001.rhd']\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "base_path = 'Z:/limblab/lab_folder/Projects/wireless_EMG/Pop_demo/'\n",
    "file_list = os.listdir(base_path)\n",
    "print(file_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "* The *.nev* and *.ns3* files are recorded by the Cerebus system. The *.nev* file stores cortical signals, while the *.ns3* stores forces, video sync pulses etc.\n",
    "* The *.rhd* files stores the EMGs collected by the DSPW wireless system. *.rhd* is also the data format created by INTAN and used by their products.\n",
    "* The *.mat* file is generated by [xds_matlab](https://github.com/limblab/xds/tree/master/xds_matlab), containing all necessary information in the *.nev* and *.ns3* files.\n",
    "\n",
    "**Notice** Sometimes if there's no *.nsx* file accoompanying the *.nev* file the xds_matlab script would throw an error when trying to bin the data. When binning the data xds calls [this function](https://github.com/limblab/ClassyDataAnalysis/blob/master/%40commonDataStructure/sanitizeTimeWindows.m) in [cds](https://github.com/limblab/ClassyDataAnalysis). The *cds* tries to find the proper aligned time window for all data types based on the start and end time for certain types of data in the *.nsx* file (could be EMGs, forces or kinematics). So if it fails to find these types of data in *.nsx* file it will assume the time window is [0, inf], therefore leading to errors."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**2. Download the xds Python codes for wireless EMG and include it in your Python path**"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Use **git bash** to download the codes and switch to the right branch\n",
    "\n",
    "    cd F:\n",
    "   \n",
    "    git clone https://github.com/limblab/xds\n",
    "   \n",
    "    git checkout wireless_EMG\n",
    "\n",
    "Use the codes below to include xds in your path. For example, here the xds-Python path should be `F:\\\\xds\\\\xds_python`"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append('F:\\\\xds\\\\xds_python')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**3. Load the .mat file and the .rhd file and save the data in .pkl for future use**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['Pop_20210902_PG_baseline_001.mat']\n",
      "['Pop_20210902_PG_baseline_001.rhd']\n",
      "\n",
      "Reading Intan Technologies RHD2000 Data File, Version 1.5\n",
      "\n",
      "n signal groups 7\n",
      "Found 32 amplifier channels.\n",
      "Found 0 auxiliary input channels.\n",
      "Found 0 supply voltage channels.\n",
      "Found 0 board ADC channels.\n",
      "Found 2 board digital input channels.\n",
      "Found 0 board digital output channels.\n",
      "Found 0 temperature sensors channels.\n",
      "\n",
      "File contains 906.298 seconds of data.  Amplifiers were sampled at 2.01 kS/s.\n",
      "\n",
      "Allocating memory for data...\n",
      "Reading data from file...\n",
      "10% done...\n",
      "20% done...\n",
      "30% done...\n",
      "40% done...\n",
      "50% done...\n",
      "60% done...\n",
      "70% done...\n",
      "80% done...\n",
      "90% done...\n",
      "Parsing data...\n",
      "Warning: 21 gaps in timestamp data found.  Time scale will not be uniform!\n",
      "Done!  Elapsed time: 5.7 seconds\n",
      "The numbers of these bad channels are []\n",
      "For noisy channel 1DI, use only one single end channel.\n",
      "For noisy channel BI, use only one single end channel.\n",
      "For noisy channel TRI, use only one single end channel.\n",
      "For noisy channel EDC1, use only one single end channel.\n",
      "Rejecting high amplitude EMG artifacts.\n",
      "Applying notch filter.\n",
      "All EMG channels have been filtered.\n",
      "Bin data from both Cerebus recorded nev file and DSPW recorded rhd file\n",
      "Binning spikes with 0.0010 s\n",
      "Filtered EMGs have been downsampled\n",
      "Data have been binned.\n",
      "Save to Z:/limblab/lab_folder/Projects/wireless_EMG/Pop_demon/Pop_20210902_PG_baseline_001.pkl successfully\n"
     ]
    }
   ],
   "source": [
    "from xds import lab_data_DSPW_EMG\n",
    "import fnmatch\n",
    "import numpy as np\n",
    "\n",
    "mat_list = np.sort(fnmatch.filter(os.listdir(base_path+'/'), \"*.mat\"))\n",
    "rhd_list = np.sort(fnmatch.filter(os.listdir(base_path+'/'), \"*.rhd\"))\n",
    "print(mat_list)\n",
    "print(rhd_list)\n",
    "\n",
    "bad_EMG = [16, 24, 25, 26] \n",
    "# Known bad EMG channels. Channels 24, 25, 26 have been proved to be mistakenly shorted on the adaptor board\n",
    "# Channel 16 could be related to an electrode failure\n",
    "# When putting the channel numbers here, the processing codes will automatically remove them. If 2 channels in here belong to a\n",
    "# pair of EMG, the codes will remove the pair, otherwise the remaining EMG will be treated as a single-ended channel.\n",
    "\n",
    "for each in zip(mat_list, rhd_list):\n",
    "    data = lab_data_DSPW_EMG(base_path, each[0], each[1], bad_EMG, comb_filter = 0, art_reject = 1)\n",
    "    data.save_to_pickle(base_path) "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Now both the neural data and wireless EMGs were taken care of and saved in the *.pkl*. The single-ended EMGs were paired and differentiated. The bad channels were removed. High amplitude artifacts were rejected. The good channels were filtered to create envelops.\n",
    "\n",
    "You can load the *.pkl* file to do more work at any time."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**4. Load the .pkl file**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The original bin size is 0.001000 s\n",
      "The new bin size is 0.0250 s\n"
     ]
    }
   ],
   "source": [
    "import pickle\n",
    "\n",
    "file_name = 'Pop_20210902_PG_baseline_001.pkl'\n",
    "\n",
    "with open ( base_path + file_name, 'rb' ) as fp:\n",
    "    dataset = pickle.load(fp)\n",
    "print('The original bin size is %f s'%(dataset.bin_width))\n",
    "    \n",
    "EMG_names = dataset.EMG_names\n",
    "    \n",
    "bin_size = 0.025 # Change the bin size here\n",
    "smooth_size = 0.05 # Change the smooth window size here\n",
    "\n",
    "dataset.update_bin_data(bin_size)  \n",
    "dataset.smooth_binned_spikes(bin_size, 'gaussian', smooth_size)\n",
    "\n",
    "# Getting the data in trials\n",
    "trial_info = dataset.get_trial_info('R')\n",
    "spike = dataset.get_trials_data_spike_counts('R', 'gocue_time', 0.5, 'gocue_time', 1.5)\n",
    "EMG = dataset.get_trials_data_EMG('R', 'gocue_time', 0.5, 'gocue_time', 1.5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['APB', 'FPB', 'Lum', 'PT', 'FDS1', 'FCR2', 'FCU1', 'FDP2', '1DI', '4DI', 'EPM', 'ECR', 'ECU', 'EDC1', 'TRI', 'BI']\n"
     ]
    }
   ],
   "source": [
    "print(EMG_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "There are 98 trials in this session.\n"
     ]
    }
   ],
   "source": [
    "print('There are %d trials in this session.'%(len(trial_info)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here are trial information for the 10th trial:\n",
      "{'trial_result': 'R', 'trial_target_dir': 90.0, 'trial_gocue_time': 90.0853, 'trial_start_time': 89.31, 'trial_end_time': 91.659, 'trial_target_corners': array([-12.5 ,  16.75,  12.5 ,  13.25]), 'trial_target_center_x': 0.0, 'trial_target_center_y': 15.0}\n"
     ]
    }
   ],
   "source": [
    "print('Here are trial information for the 10th trial:')\n",
    "print(trial_info[10])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "In this session the y position of the targets are:\n",
      "{11.0, 13.0, 15.0}\n"
     ]
    }
   ],
   "source": [
    "print('In this session the y position of the targets are:')\n",
    "print(set(dataset.trial_target_center_y))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Here is the shape of the data within one trial\n",
      "Spikes:\n",
      "(80, 96)\n",
      "EMGs:\n",
      "(80, 16)\n"
     ]
    }
   ],
   "source": [
    "print('Here is the shape of the data within one trial')\n",
    "print('Spikes:')\n",
    "print(spike[0].shape)\n",
    "print('EMGs:')\n",
    "print(EMG[0].shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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.7.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
