Warp electrophysiological data from clock time to brain time and analyze the dynamic neural patterns of cognitive processes. A MATLAB toolbox by Sander van Bree, María Melcón, Luca Kolibius, Casper Kerrén, Maria Wimber, Simon Hanslmayr.
For the context to this project, check out the brain time paper (view full PDF here). See this video for a broad strokes primer. The full methodological details are in the paper's supplementary materials.
- MATLAB Signal Processing Toolbox
- FieldTrip
- MVPA Light
- Several functions included in the toolbox
Operation 1: Brain time warping
Operation 2: Periodicity analysis
Option 1: Download files
- Download ZIP
- Unzip to preferred folder
- Run the function setup_braintime.
Option 2: Git client or checkout with SVN
Type:
git clone https://github.com/sandervanbree/braintime.git
then run the function setup_braintime.
Toolbox term | description |
---|---|
Brain time warping | a transformation of electrophysiology data based on the phase dynamics of the warping signal |
Clock time (CT) | time as sequences of seconds |
Brain time (BT) | time as sequences of cycles of a coordinating brain oscillation |
Warping signal | an oscillatory signal that is assumed to clock the cognitive process of interest |
Warping source | data structure containing potential warping signals used for brain time warping (e.g., local field potentials, independent component analysis components, virtual channels) |
Periodicity | fluctuating patterns of a neural signature |
Neural signature | brain activity that systematically correlates with, in the present context, a cognitive process |
What's the premise behind the Brain Time Toolbox (braintime
)? Insofar a cognitive process is clocked by oscillations, analyses of the process's dynamics require a factoring in of the oscillations' dynamics. To this end, the toolbox implements brain time warping, an algorithm that transforms electrophysiological data based on relevant oscillations selected by the user — the warping signal. This changes the data from clock time (seconds), closer to brain time (cycles). This completes the first operation of braintime
: brain time warping.
The second operation is periodicity analysis. Here, braintime
uses multivariate pattern analysis (MVPA) to detect the effects of the applied data transformation. Has evidence of the rhythmicity of the cognitive patterns increased?
Below, we explain how to perform both operations step by step. To see the steps in practice, check out braintime
's tutorials. For extensive methodological documentation, check out the main paper's supplementary material. This figure gives a visual rundown of all steps:
1.1 Loading clock time data
braintime
works with FieldTrip formatted electrophysiological data. This can be electroencephalography (EEG), magnetoencephalography (MEG), or intracranial data. The starting data is called clock time data — this will be warped.
1.2 Loading warping sources data
Warping sources are the data structure containing the to-be-selected warping signal, which is used to warp clock time data. Warping sources may be obtained separately from clock time data, or extracted from it (using independent component analysis, or the selection of a few channels). Please ensure that your warping source data has 0.5s of additional time extra (before and after your time window of interest) to facilitate step 1.3.
⚠️ If your clock time data and warping sources are not independent, please read Is it circular to warp to warping sources obtained from my clock time data?
1.3 Time frequency analysis of warping sources
Each warping source contains potential warping signals. bt_analyzesources performs a time frequency analysis on all warping sources, detecting potential warping signals based on your preferences. These preferences include methods of time frequency analysis (parameters fed into FieldTrip), the frequency range of interest assumed to clock your cognitive process (e.g., 8 to 12 Hz for attention), and the time window of interest that you wish to analyze (e.g. 0 to 1 second, this should match the window you wish to warp). In addition, you can choose one of two methods to cut the data, cfg.warpmethod = 'consistenttime'
or 'cutartefact'
, both with their own relative merits (see "which cutmethod to choose?").
bt_analyzesources collects a whole bunch of information about the warping sources, such as the phase of the underlying warping signals (obtained using two methods, described in 1.5), its average waveshape and its asymmetry, and a ranking of all warping sources based on your preferences.
💡 You can adjust the ranking of warping sources by their topography by creating a topography using bt_templatetopo and changing to
cfg.rankmethod = 'temptopo'
.
1.4 Selecting a warping source and signal
Now that the toolbox has collected all the relevant time frequency details of warping sources, it's time to make a selection using bt_selectsource. If desired, you can input a layout structure to plot each warping source's topography. For each warping source, bt_selectsource plots a time frequency spectrum, a power spectrum, the topography of the warping source, and waveshape details. For each warping source, a warping signal is highlighted, set at the highest power frequency in the range of interest. Factoring in its characteristics, you can select a warping signal. braintime
uses the phase dynamics of this warping signal to brain time warp the clock time data in the next step.
1.5 Warping clock to brain time
This is where the magic happens. bt_clocktobrain takes the phase of the chosen warping signal and dynamically time warps (DTW) it to the phase of a stationary signal of the same frequency. DTW enables a readout of where clock time (stationary signal) falls out of tune with brain time (phase of warping signal) by attempting to minimize their difference. That minimization process yields a warping path, which tells braintime
the samples that need to be repeated for clock and brain time to better align. bt_clocktobrain repeats those samples in the original data, cycle-by-cycle and trial-by-trial, before squeezing things back down to the original data length. Here's a schematic:
There are two parameters you can change in bt_clocktobrain. You can choose to set the clock time signal as a basic stationary sinusoid ( cfg.warpmethod = 'sinusoid'
) or a smoothed version of the warping signal's waveshape (cfg.warpmethod = 'waveshape'
).
💡 For data with asymmetric waves such as theta oscillations in intracranial rodent data, using waveshape as a warpmethod is the natural choice.
There are also two "phase methods", which are separate phase estimations of the warping signal both automatically obtained by bt_analyzesources. One is a regular method of phase estimation (using the Fast Fourier Transform of the warping frequency, selected with cfg.phasemethod = 'fft'
), but there is also a specialized method called General Eigenvalue Decomposition (GED), selected with cfg.phasemethod = 'ged'
. The former is classical and specific to the warping source, the latter is novel and more holistically estimates phase. The latter is holistic as the phase of the warping frequency is estimated by taking a weighted combination of all warping sources. More details on when to use which? Check out "which phase should I warp to, FFT or GED?".
After bt_clocktobrain, your data has completed its transformation from clock to brain time. The time axis is now formatted as sequences of cycles, instead of seconds. You may opt to continue analyses outside of the toolbox, or test for periodic patterns in the data using braintime
's second operation described next. In case of the former, please read the previously linked circularity information.
You may wish to use braintime
to test whether brain time warping has accentuated dynamic patterns in the data. This is appropriate if your data comprises two underlying conditions which are predicted to yield fluctuating classification evidence. For example, we report a spatial attention dataset where we predict that evidence for motion direction rapidly alternates from low to high as a function of alpha phase (the warping signal). Thus, the second operation requires your brain time warped data to consist of such classes.
Before getting started, ensure the original clock time data as well as the brain time warped data have their classes labeled. Specifically, create a field called "clabel", with a vector of 1's and 2's corresponding to the condition of each trial. For example, if your trial sequence is 'left left right' conditions, the clabel field should contain [1 1 2]
.
The pipeline for this toolbox operation is that each participant's data needs to undergo step 2.1 to 2.3, separately for both clock and brain time data. Then, each participant's output needs to be added as a field to a large group structure, separately for clock and brain time results. These two group structures (e.g. ct_stats1
and bt_stats1
) can each be used as input for step 2.4 to obtain group level results, allowing you to compare periodicity in clock time and brain time data.
⚠️ A recent paper by van Driel et al. (2021) shows that high pass filtering data may cause artefacts through displacement of information. We have found that these artefacts may sometimes look periodic. Thus, when using operation 2 of the toolbox, please ensure no high pass filter was applied to the data, or do so with extreme care. van Driel et al. introduce alternative ways of getting rid of low frequencies (robust detrending using a trial-based mask), and make several suggestions.
2.1 Multivariate pattern analysis
The first step is to use MVPA-Light
to classify the data. You may opt to classify across time mv_classify_across_time.m. This method tests whether a classifier can separate both classes of data across time in the trials. Alternatively, you can apply temporal generalization by classifying using mv_classify_timextime.m. This method tests for temporal generalization of classification. That is, it tests to what extent a classifier trained on one timepoint generalizes its performance to other timepoints. Not sure whether to classify across time, or test for temporal generalization? Check out "should I classify across time or test for temporal generalization?.
💡 You can change a variety of parameters when calling
MVPA-Light
, described briefly in tutorial 2. For more information, checkMVPA-Light
's awesome tutorials.
2.2 Quantify periodicity in classifier performance
If the warping signal orchestrates the dynamics of your cognitive function, operationalized by your two classes of data, then the classifier's performance may tap into these dynamics. Thus, bt_quantify tests and quantifies periodic patterns in the classifier performance, whether that is in the classifier output obtained from mv_classify_across_time.m (1 dimensional; "diagonal"), or mv_classify_timextime.m (2 dimensional; "temporal generalization matrix" (TGM)). In both cases, you get a "periodicity spectrum" which is an average power spectrum of all the data in your 1D or 2D classifier performance.
For TGMs, you may opt to perform the periodicity analysis over either the 2 dimensional matrix itself (cfg.method = 'tgm'
) or its autocorrelation map (cfg.method = 'ac'
). Not sure what is better? Check out "should I perform bt_quantify over the TGM itself, or its autocorrelation map?".
You also need to specify a range of periodicity frequencies. In which range of rates do you expect periodic patterns in classifier performance to arise? Finally, you can choose a reference dimension. If you choose cfg.refdimension = clocktime
, periodicity in classifier performance will be displayed as a function of cycles per seconds (Hz). Alternatively, if you are quantifying brain time warped data, cfg.refdimension = clocktime
is more appropriate. This references the peroidicity as a function of cycles per second, normalized to that participants' warping frequency.
💡 Let's say you warp a participant's data to an 11 Hz warping signal. In that case, with
cfg.refdimension = clocktime
, a peak at the warping frequency will be at 11 Hz, but withcfg.refdimension = braintime
it will show up at 1 Hz, as the frequencies are normalized to the warped frequency (11/11 = 1).
2.3 First level statistics
How does the participant's quantified periodicity compare against the null distribution? bt_statslevel1 takes the output from bt_quantify and performs classification cfg.numperms1 = n1
times over, each time randomly shuffling the classification labels and obtaining n1
"permuted" periodicity spectra. This provides a null distribution that quantifies how much periodicity is in the data when the class structure is destroyed, setting things up for p-value estimation on the group level.
Now, repeat bt_statslevel1 for every participant, and send its output to a separate field in a group structure (e.g. [ct_stats1{subj}] = bt_statslevel1(cfg,data,quant)
). The next step requires this format to perform 2nd level statistics — it loops over the fields.
2.4 Second level statistics
We just got an idea of the periodicity in single participants. But what about the group level? bt_statslevel2 employs the following algorithm to generate second level statistics:
cfg.numperms2 = n2
times, do the following: randomly grab one of each participants'n1
permuted periodicity spectra and average those spectra into one spectrum. This yieldsn2
spectra.- For each frequency, compare the empirical periodicity with the distribution of permuted periodicity to derive a p-value.
- Plot the average empirical periodicity spectrum and display the p-value for each frequency.
💡 In brain time results, p-values at 0.5 Hz, 1 Hz, and 2 Hz are exempted from multiple testing correction, as periodicity is predicted at one or multiple of these rates depending on the underlying structure'.
💡
braintime
z-scores participants' spectra to enable second level statistics. For details on how it does this, check out "How does the toolbox z-score periodicity spectra?".
At this point we have tested whether periodicity in classification performance is significant for clock and brain time. However, a separate question is whether classification performance in itself is significant, without any question of periodic patterns. bt_statslevel2 also tests for this, by calling MVPA-Light
. For this, the function requires a separate structure called cfg_clus
, where you can enter a range of parameters for cluster correction. So, aside from the periodicity results, you also get a display of which datapoints show above chance classification performance.
We provide three sources of evidence for the toolbox. A simulated dataset, a rodent intracranial dataset, and a human EEG dataset. braintime
includes the script used to generate the simulated dataset. Toying around with it is a great way to both get a feel for the toolbox, and to see its effects. Check out bt_dipsim, change some parameters to your liking, and run it through the braintime
pipeline to see that the toolbox accounts for clock and brain time disharmony. Details on the rodent and human evidence are described in the main manuscript.
At the start of trials, braintime
repeats original data samples until the phase of the warping signal aligns with the stationary signal (see paragraph 1.5). This takes a while, depending on their disharmony. This data repetition may cause an artefact in further analyses, called a "first cycle artefact". In braintime
's periodicity analysis operation for example, you can sometimes see a stretched out pattern of low or high classification. Most often, this artefact is very small, and unlikely to alter subsequent analyses. Hence, braintime
's default option under bt_analyzesources is cfg.cutmethod = 'consistenttime'
. This default method is called consistenttime
because the toolbox warps exactly to the specified time window of interest.
To the extent the first cycle artefact is predicted to be an issue for further analyses, you may decide to remove it by selecting cfg.cutmethod = 'cutartefact'
under bt_analyzesources. With this parameter, brainime
warps a little bit of time before and after the specified time window of interest before it is removed in later steps. This makes it so the data reptition segment is removed from the final data. The downside to this method is that the warped data will comprise more (or less) than the specified time window of interest.
Finally, there's one more thing to consider here. braintime
allocates data samples to cycles depending on the warping path. As the two cutmehods comprise warping to a different length of data, there may be variations in which data sample is considered part of which cycle. To check how each method allocates data to cycles, check out tutorial 7.
In short, the upside of cfg.cutmethod = 'consistenttime'
is that the warped data will exactly of the specified time window, but its downside is a first cycle artefact. In contrast, the upside of cfg.cutmethod = 'curartefact'
is that the warped data will not contain a first cycle artefact, but its downside is that the data's time may not exactly match the specified window. The methods may vary in which data they allocate to which cycle.
It depends on the type of analysis carried out, and the dependence between warping signal and warped data. First, let's consider the concern. The concern is that warping data to the phase dynamics of a warping oscillation trivially introduces patterns in the data oscillating at the warping oscillation's frequency, as you've changed the data according to it.
When carrying out basic analyses on warped data outside of the Brain Time Toolbox -- such as time frequency analyses, or event-related potentials -- then most caution is warranted. Warping to any frequency in the data will reduce the nonstationarities of those oscillations, and this will have subsequent effects on a large array of basic results. Thus, oscillatory changes induced by brain time warping are not interesting in and of themselves, but set the stage for more advanced analyses that are sufficiently independent from raw oscillatory changes and tap into cognition.
Besides the type of analysis, data dependence is another factor to consider. Data dependence refers to the degree to which the warping signal is present in the warped data. For example, if a warping signal is obtained from an EEG channel that is retained in warped data, there is high data dependence. If the channel is subsequently removed, low data dependence is achieved. Warping-induced changes in highly independent data are less trivial than changes in highly dependent data (except if advanced analyses are carried out).
FFT (cfg.phasemethod = 'fft'
) is more constricted to the warping source of interest, while GED (cfg.phasemethod = 'ged'
) takes a weighted approach across all warping sources. So the best choice depends on whether you expect the oscillations of interest to be isolated or spread out across the data. To illustrate the best choice depending on circumstance let's look at some examples. If we are studying attention in the parietal cortex, or motor processes in the motor cortex, it may be better to take FFT on the warping source that shows topographies specific to our process of interest. Here, you don't want oscillations from other regions, so GED should be avoided. In contrast, if your warping sources are already restricted to a region, like the local field potential of hippocmapal regions or source localized parietal data, then a holistic estimation using GED is more appropriate as all warping sources are assumed to contain relevant phase dynamics.
It depends on the uniformity of the periodic patterns of your 2D classifier performance (TGM). In principle, the autocorrelation map (cfg.method = 'ac'
) is a more powerful approach, as it is known to accentuate periodic patterns in the data. However, when multiple periodicity rates are present in the TGM, the autocorrelation tends to pick up the strongest rate and drown out the weaker one. Similarly, when periodicity is present in only a small part of the TGM, autocorrelation map may not be able to accentuate its pattern. In these cases, it's better to analyze the TGM itself (cfg.method = 'tgm'
).
Thus, when you expect uniform periodicity patterns in the data, analyzing the autocorrelation map of TGMs is a powerful approach. When the pattern is expected to only be present partially, or when multiple rates are predicted, it is better to analyze the TGM itself.
We want to prevent that arbitrary differences in the periodicity power spectra between participants drive a group effect. To this end, braintime
z-scores each participant's empirical and permuted periodicity spectrum in the following way:
Step | first level stats (bt_statslevel1) and z-scoring algorithm |
---|---|
1 | Z-score each of n1 permuted periodicity spectra by the mean and standard deviation of that spectrum. |
2 | Z-score the single empirical dsitribution by the mean and standard deviation of all n1 permutations. |
Step | second level stats (bt_statslevel2) |
---|---|
1 | for each second level permutation n2 : Grab one (random) permuted spectrum of each participant's n1 permuted spectra and average them into one. This yields n2 permuted averages. |
2 | for each frequency, make a p-value from the percentile at which your empirical power lands in the distribution of n2 permuted power values. |
3 | apply multiple testing correction across tested frequencies using false discovery rate (FDR; Benjamini & Yekutieli, 2001). |
The paper (& preprint) have extensive supplementary material covering the toolbox, which may well give the answer to your question. For remaining questions, any issue you have found, or suggestions you would like to offer, please use the issue tracker.