Visualising statistical significance thresholds on EEG data#

MNE-Python provides a range of tools for statistical hypothesis testing and the visualisation of the results. Here, we show a few options for exploratory and confirmatory tests - e.g., targeted t-tests, cluster-based permutation approaches (here with Threshold-Free Cluster Enhancement); and how to visualise the results.

The underlying data comes from [1]; we contrast long vs. short words. TFCE is described in [2].

# License: BSD-3-Clause
# Copyright the MNE-Python contributors.
import matplotlib.pyplot as plt
import numpy as np
from scipy.stats import ttest_ind

import mne
from mne.channels import find_ch_adjacency, make_1020_channel_selections
from mne.stats import spatio_temporal_cluster_test

np.random.seed(0)

# Load the data
path = mne.datasets.kiloword.data_path() / "kword_metadata-epo.fif"
epochs = mne.read_epochs(path)
# These data are quite smooth, so to speed up processing we'll (unsafely!) just
# decimate them
epochs.decimate(4, verbose="error")
name = "NumberOfLetters"

# Split up the data by the median length in letters via the attached metadata
median_value = str(epochs.metadata[name].median())
long_words = epochs[name + " > " + median_value]
short_words = epochs[name + " < " + median_value]
Reading /home/circleci/mne_data/MNE-kiloword-data/kword_metadata-epo.fif ...
Isotrak not found
    Found the data of interest:
        t =    -100.00 ...     920.00 ms
        0 CTF compensation matrices available
Adding metadata with 8 columns
960 matching events found
No baseline correction applied
0 projection items activated

If we have a specific point in space and time we wish to test, it can be convenient to convert the data into Pandas Dataframe format. In this case, the mne.Epochs object has a convenient mne.Epochs.to_data_frame() method, which returns a dataframe. This dataframe can then be queried for specific time windows and sensors. The extracted data can be submitted to standard statistical tests. Here, we conduct t-tests on the difference between long and short words.

time_windows = ((0.2, 0.25), (0.35, 0.45))
elecs = ["Fz", "Cz", "Pz"]
index = ["condition", "epoch", "time"]

# display the EEG data in Pandas format (first 5 rows)
print(epochs.to_data_frame(index=index)[elecs].head())

report = "{elec}, time: {tmin}-{tmax} s; t({df})={t_val:.3f}, p={p:.3f}"
print("\nTargeted statistical test results:")
for tmin, tmax in time_windows:
    long_df = long_words.copy().crop(tmin, tmax).to_data_frame(index=index)
    short_df = short_words.copy().crop(tmin, tmax).to_data_frame(index=index)
    for elec in elecs:
        # extract data
        A = long_df[elec].groupby("condition").mean()
        B = short_df[elec].groupby("condition").mean()

        # conduct t test
        t, p = ttest_ind(A, B)

        # display results
        format_dict = dict(
            elec=elec, tmin=tmin, tmax=tmax, df=len(epochs.events) - 2, t_val=t, p=p
        )
        print(report.format(**format_dict))
channel                       Fz        Cz        Pz
condition epoch time
film      0     -0.096  0.453939  0.232879  0.222424
                -0.080  0.518939  0.214091 -0.371515
                -0.064  0.811667  0.793636  0.250152
                -0.048  0.039697  0.188636  0.318030
                -0.032 -1.163030 -1.018939 -0.425152

Targeted statistical test results:
Fz, time: 0.2-0.25 s; t(958)=-0.661, p=0.509
Cz, time: 0.2-0.25 s; t(958)=-2.682, p=0.007
Pz, time: 0.2-0.25 s; t(958)=-3.238, p=0.001
Fz, time: 0.35-0.45 s; t(958)=5.304, p=0.000
Cz, time: 0.35-0.45 s; t(958)=5.684, p=0.000
Pz, time: 0.35-0.45 s; t(958)=6.508, p=0.000

Absent specific hypotheses, we can also conduct an exploratory mass-univariate analysis at all sensors and time points. This requires correcting for multiple tests. MNE offers various methods for this; amongst them, cluster-based permutation methods allow deriving power from the spatio-temoral correlation structure of the data. Here, we use TFCE.

# Calculate adjacency matrix between sensors from their locations
adjacency, _ = find_ch_adjacency(epochs.info, "eeg")

# Extract data: transpose because the cluster test requires channels to be last
# In this case, inference is done over items. In the same manner, we could
# also conduct the test over, e.g., subjects.
X = [
    long_words.get_data(copy=False).transpose(0, 2, 1),
    short_words.get_data(copy=False).transpose(0, 2, 1),
]
tfce = dict(start=0.4, step=0.4)  # ideally start and step would be smaller

# Calculate statistical thresholds
t_obs, clusters, cluster_pv, h0 = spatio_temporal_cluster_test(
    X, tfce, adjacency=adjacency, n_permutations=100
)  # a more standard number would be 1000+
significant_points = cluster_pv.reshape(t_obs.shape).T < 0.05
print(str(significant_points.sum()) + " points selected by TFCE ...")
Could not find a adjacency matrix for the data. Computing adjacency based on Delaunay triangulations.
-- number of adjacent vertices : 29
stat_fun(H1): min=7.176438127884552e-07 max=80.91729757691468
Running initial clustering …
Using 202 thresholds from 0.40 to 80.80 for TFCE computation (h_power=2.00, e_power=0.50)
Found 1856 clusters

  0%|          | Permuting : 0/99 [00:00<?,       ?it/s]
  1%|          | Permuting : 1/99 [00:00<00:09,    9.86it/s]
  2%|▏         | Permuting : 2/99 [00:00<00:06,   14.98it/s]
  3%|▎         | Permuting : 3/99 [00:00<00:06,   14.90it/s]
  4%|▍         | Permuting : 4/99 [00:00<00:06,   14.89it/s]
  5%|▌         | Permuting : 5/99 [00:00<00:07,   13.39it/s]
  6%|▌         | Permuting : 6/99 [00:00<00:07,   12.53it/s]
  7%|▋         | Permuting : 7/99 [00:00<00:07,   12.86it/s]
  8%|▊         | Permuting : 8/99 [00:00<00:06,   14.04it/s]
  9%|▉         | Permuting : 9/99 [00:00<00:06,   14.14it/s]
 10%|█         | Permuting : 10/99 [00:00<00:06,   14.22it/s]
 11%|█         | Permuting : 11/99 [00:00<00:06,   14.29it/s]
 12%|█▏        | Permuting : 12/99 [00:00<00:06,   14.34it/s]
 13%|█▎        | Permuting : 13/99 [00:00<00:06,   13.71it/s]
 14%|█▍        | Permuting : 14/99 [00:01<00:06,   13.21it/s]
 15%|█▌        | Permuting : 15/99 [00:01<00:06,   12.31it/s]
 16%|█▌        | Permuting : 16/99 [00:01<00:06,   12.50it/s]
 17%|█▋        | Permuting : 17/99 [00:01<00:06,   12.67it/s]
 18%|█▊        | Permuting : 18/99 [00:01<00:06,   12.82it/s]
 19%|█▉        | Permuting : 19/99 [00:01<00:06,   12.96it/s]
 20%|██        | Permuting : 20/99 [00:01<00:06,   13.08it/s]
 21%|██        | Permuting : 21/99 [00:01<00:06,   12.77it/s]
 22%|██▏       | Permuting : 22/99 [00:01<00:05,   13.33it/s]
 23%|██▎       | Permuting : 23/99 [00:01<00:05,   13.43it/s]
 24%|██▍       | Permuting : 24/99 [00:01<00:05,   13.52it/s]
 25%|██▌       | Permuting : 25/99 [00:01<00:05,   13.61it/s]
 26%|██▋       | Permuting : 26/99 [00:01<00:05,   14.12it/s]
 27%|██▋       | Permuting : 27/99 [00:01<00:05,   13.73it/s]
 28%|██▊       | Permuting : 28/99 [00:02<00:04,   14.23it/s]
 29%|██▉       | Permuting : 29/99 [00:02<00:04,   14.27it/s]
 30%|███       | Permuting : 30/99 [00:02<00:04,   14.30it/s]
 31%|███▏      | Permuting : 31/99 [00:02<00:04,   14.33it/s]
 32%|███▏      | Permuting : 32/99 [00:02<00:04,   14.81it/s]
 33%|███▎      | Permuting : 33/99 [00:02<00:04,   15.27it/s]
 34%|███▍      | Permuting : 34/99 [00:02<00:04,   15.71it/s]
 35%|███▌      | Permuting : 35/99 [00:02<00:04,   15.18it/s]
 36%|███▋      | Permuting : 36/99 [00:02<00:04,   15.63it/s]
 37%|███▋      | Permuting : 37/99 [00:02<00:04,   15.11it/s]
 38%|███▊      | Permuting : 38/99 [00:02<00:04,   14.66it/s]
 39%|███▉      | Permuting : 39/99 [00:02<00:04,   14.26it/s]
 40%|████      | Permuting : 40/99 [00:02<00:04,   13.90it/s]
 41%|████▏     | Permuting : 41/99 [00:02<00:04,   13.59it/s]
 42%|████▏     | Permuting : 42/99 [00:03<00:04,   13.30it/s]
 43%|████▎     | Permuting : 43/99 [00:03<00:04,   13.73it/s]
 44%|████▍     | Permuting : 44/99 [00:03<00:03,   13.78it/s]
 45%|████▌     | Permuting : 45/99 [00:03<00:03,   13.81it/s]
 46%|████▋     | Permuting : 46/99 [00:03<00:03,   13.86it/s]
 47%|████▋     | Permuting : 47/99 [00:03<00:03,   13.56it/s]
 48%|████▊     | Permuting : 48/99 [00:03<00:03,   13.63it/s]
 49%|████▉     | Permuting : 49/99 [00:03<00:03,   13.69it/s]
 51%|█████     | Permuting : 50/99 [00:03<00:03,   13.75it/s]
 52%|█████▏    | Permuting : 51/99 [00:03<00:03,   14.16it/s]
 53%|█████▎    | Permuting : 52/99 [00:03<00:03,   14.19it/s]
 54%|█████▎    | Permuting : 53/99 [00:03<00:03,   14.22it/s]
 55%|█████▍    | Permuting : 54/99 [00:03<00:03,   14.25it/s]
 56%|█████▌    | Permuting : 55/99 [00:03<00:03,   14.28it/s]
 57%|█████▋    | Permuting : 56/99 [00:04<00:03,   13.95it/s]
 58%|█████▊    | Permuting : 57/99 [00:04<00:03,   13.65it/s]
 59%|█████▊    | Permuting : 58/99 [00:04<00:03,   13.06it/s]
 60%|█████▉    | Permuting : 59/99 [00:04<00:03,   12.56it/s]
 61%|██████    | Permuting : 60/99 [00:04<00:03,   12.92it/s]
 62%|██████▏   | Permuting : 61/99 [00:04<00:02,   13.01it/s]
 63%|██████▎   | Permuting : 62/99 [00:04<00:02,   13.40it/s]
 64%|██████▎   | Permuting : 63/99 [00:04<00:02,   13.46it/s]
 65%|██████▍   | Permuting : 64/99 [00:04<00:02,   13.53it/s]
 66%|██████▌   | Permuting : 65/99 [00:04<00:02,   13.92it/s]
 67%|██████▋   | Permuting : 66/99 [00:04<00:02,   14.31it/s]
 68%|██████▊   | Permuting : 67/99 [00:04<00:02,   13.99it/s]
 69%|██████▊   | Permuting : 68/99 [00:04<00:02,   14.38it/s]
 70%|██████▉   | Permuting : 69/99 [00:04<00:02,   14.40it/s]
 71%|███████   | Permuting : 70/99 [00:05<00:02,   14.07it/s]
 72%|███████▏  | Permuting : 71/99 [00:05<00:01,   14.11it/s]
 73%|███████▎  | Permuting : 72/99 [00:05<00:01,   13.81it/s]
 74%|███████▎  | Permuting : 73/99 [00:05<00:01,   13.86it/s]
 75%|███████▍  | Permuting : 74/99 [00:05<00:01,   13.91it/s]
 76%|███████▌  | Permuting : 75/99 [00:05<00:01,   14.30it/s]
 77%|███████▋  | Permuting : 76/99 [00:05<00:01,   13.98it/s]
 78%|███████▊  | Permuting : 77/99 [00:05<00:01,   14.36it/s]
 79%|███████▉  | Permuting : 78/99 [00:05<00:01,   14.04it/s]
 80%|███████▉  | Permuting : 79/99 [00:05<00:01,   14.08it/s]
 81%|████████  | Permuting : 80/99 [00:05<00:01,   14.11it/s]
 82%|████████▏ | Permuting : 81/99 [00:05<00:01,   14.14it/s]
 83%|████████▎ | Permuting : 82/99 [00:05<00:01,   14.53it/s]
 84%|████████▍ | Permuting : 83/99 [00:05<00:01,   14.54it/s]
 85%|████████▍ | Permuting : 84/99 [00:05<00:01,   14.55it/s]
 86%|████████▌ | Permuting : 85/99 [00:06<00:00,   14.94it/s]
 87%|████████▋ | Permuting : 86/99 [00:06<00:00,   14.56it/s]
 88%|████████▊ | Permuting : 87/99 [00:06<00:00,   14.21it/s]
 89%|████████▉ | Permuting : 88/99 [00:06<00:00,   14.24it/s]
 90%|████████▉ | Permuting : 89/99 [00:06<00:00,   14.26it/s]
 91%|█████████ | Permuting : 90/99 [00:06<00:00,   14.65it/s]
 92%|█████████▏| Permuting : 91/99 [00:06<00:00,   14.65it/s]
 93%|█████████▎| Permuting : 92/99 [00:06<00:00,   14.66it/s]
 94%|█████████▍| Permuting : 93/99 [00:06<00:00,   13.98it/s]
 95%|█████████▍| Permuting : 94/99 [00:06<00:00,   14.34it/s]
 96%|█████████▌| Permuting : 95/99 [00:06<00:00,   14.36it/s]
 97%|█████████▋| Permuting : 96/99 [00:06<00:00,   14.39it/s]
 98%|█████████▊| Permuting : 97/99 [00:06<00:00,   14.77it/s]
 99%|█████████▉| Permuting : 98/99 [00:06<00:00,   15.15it/s]
100%|██████████| Permuting : 99/99 [00:06<00:00,   15.52it/s]
100%|██████████| Permuting : 99/99 [00:06<00:00,   14.28it/s]
362 points selected by TFCE ...

The results of these mass univariate analyses can be visualised by plotting mne.Evoked objects as images (via mne.Evoked.plot_image) and masking points for significance. Here, we group channels by Regions of Interest to facilitate localising effects on the head.

# We need an evoked object to plot the image to be masked
evoked = mne.combine_evoked(
    [long_words.average(), short_words.average()], weights=[1, -1]
)  # calculate difference wave
time_unit = dict(time_unit="s")
evoked.plot_joint(
    title="Long vs. short words", ts_args=time_unit, topomap_args=time_unit
)  # show difference wave

# Create ROIs by checking channel labels
selections = make_1020_channel_selections(evoked.info, midline="12z")

# Visualize the results
fig, axes = plt.subplots(nrows=3, figsize=(8, 8))
axes = {sel: ax for sel, ax in zip(selections, axes.ravel())}
evoked.plot_image(
    axes=axes,
    group_by=selections,
    colorbar=False,
    show=False,
    mask=significant_points,
    show_names="all",
    titles=None,
    **time_unit,
)
plt.colorbar(axes["Left"].images[-1], ax=list(axes.values()), shrink=0.3, label="µV")

plt.show()
  • Long vs. short words, 0.208 s, 0.320 s, 0.432 s
  • Left (8 channels, 82.8% of points masked), Midline (13 channels, 79.1% of points masked), Right (8 channels, 80.5% of points masked)
No projector specified for this dataset. Please consider the method self.add_proj.

References#

Total running time of the script: (0 minutes 13.397 seconds)

Estimated memory usage: 117 MB

Gallery generated by Sphinx-Gallery