Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

New feature for removing heart artifacts from EEG or ESG data using a Principal Component Analysis - Optimal Basis Sets (PCA-OBS) algorithm #13037

Merged
merged 92 commits into from
Jan 19, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
92 commits
Select commit Hold shift + click to select a range
2013fb4
feat: add initial source code
emma-bailey Sep 1, 2024
6d5f5b2
Minimum working example with local data
emma-bailey Sep 6, 2024
3622540
Implement testing dataset
emma-bailey Sep 9, 2024
6c42f33
Feat: Update examples
emma-bailey Oct 7, 2024
38b6b6a
Merge pull request #2 from mne-tools/main
steinnhauser Oct 23, 2024
2fe6a88
Merge pull request #1 from emma-bailey/feat/pca-obs
steinnhauser Oct 23, 2024
61f58b2
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Oct 23, 2024
070037d
fix: adjust import paths, add init file to module
steinnhm Oct 23, 2024
e1b6732
refactor: rearrange PCA_OBS arg structure, remove kwarg 'sr'
steinnhm Oct 23, 2024
ccd4298
refactor: move common variables to module init, further cleanup
steinnhm Oct 23, 2024
61a92ea
feat/initial-cleanup: Remove custom pchip as not in use
emma-bailey Oct 25, 2024
a1eb6f6
refactor/initial-cleanup: answer question
emma-bailey Oct 25, 2024
d75e927
refactor: rename files to match other modules in preprocessing, reduc…
steinnhm Nov 4, 2024
035a9f6
refactor: remove unused fit_ecgTemplate variable 'baselinerange'
steinnhm Nov 4, 2024
2debb61
refactor: make main methods private, import from module __init__, rem…
steinnhm Nov 4, 2024
97f9a15
refactor: add docstring templates, gather imports
steinnhm Nov 4, 2024
34a2e06
test: add placeholder tests, add multiple todos to address where we g…
steinnhm Nov 4, 2024
3760c02
Removing extra examples
emma-bailey Nov 11, 2024
0a55712
Moving example
emma-bailey Nov 11, 2024
e887ace
Adding Niazy reference
emma-bailey Nov 11, 2024
ee3b73e
Update example, put pca-obs in a single file
emma-bailey Nov 11, 2024
adae352
TODO
emma-bailey Nov 11, 2024
b3c1b4e
Update example
emma-bailey Nov 11, 2024
ba36dc2
Merge pull request #4 from mne-tools/main
steinnhauser Nov 15, 2024
e6aa17d
refactor: change way of calling pca obs method, add comments
steinnhm Nov 15, 2024
b150b6c
refactor: move pca obs method out of separate python module, change l…
steinnhm Nov 15, 2024
951e87c
Refactor: Docstrings, removed classes
emma-bailey Nov 15, 2024
c70db0a
refactor: remove unrequired index references, adjust variable namings…
steinnhm Nov 24, 2024
fb8c7ff
docs: minor edits in docstrings
steinnhm Nov 24, 2024
4c73a96
fix: add minor sanity checks for the function inputs
steinnhm Nov 25, 2024
8a0d73c
Merge pull request #3 from emma-bailey/refactor/initial-cleanup
steinnhauser Nov 25, 2024
e69039b
test: add initial test structure, missing validation of post-hear-art…
steinnhm Nov 25, 2024
4bd3a51
style: run pre-commit hooks on test file
steinnhm Nov 25, 2024
07458ba
docs: remove duplicated docstring
steinnhm Nov 25, 2024
b2be499
Merge branch 'mne-tools:main' into main
steinnhauser Nov 25, 2024
a5e68d7
Removed filter_coords from within the method
emma-bailey Nov 27, 2024
1938573
Adding info to filter to example
emma-bailey Nov 27, 2024
19e0802
style: run import sorter pre-commit hook
steinnhm Dec 4, 2024
49a96d0
refactor,test: migrate data shape to be 1d, add sanity checks for PCA…
steinnhm Dec 4, 2024
a6e11fc
Merge branch 'mne-tools:main' into main
steinnhauser Dec 4, 2024
925b293
example:Reshape ecg_events before pca_obs, tests:ecg_events is in sam…
emma-bailey Dec 6, 2024
a3d1123
example: update channel selection
emma-bailey Dec 6, 2024
e36e665
Merge branch 'mne-tools:main' into main
steinnhauser Dec 18, 2024
2ae84b0
refactor,test: change public qrs kwarg to be more clear about being i…
steinnhm Dec 18, 2024
b8d8d7c
Merge pull request #5 from emma-bailey/tests/general-pca-obs
steinnhauser Dec 18, 2024
d19049b
style: move fit_ecg_template function to bottom of file to improve re…
steinnhm Dec 18, 2024
3e9a812
Merge branch 'mne-tools:main' into main
steinnhauser Dec 19, 2024
826afec
test: add pytest importorskip for pandas lib in pca obs tests
steinnhm Dec 19, 2024
16937c8
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Dec 19, 2024
8f4dcdd
docs: add apply_pca_obs algorithm to doc/api/preprocessing.rst list
steinnhm Dec 19, 2024
35ab331
Merge branch 'main' of https://github.com/emma-bailey/mne-python
steinnhm Dec 19, 2024
121f8dd
Merge branch 'mne-tools:main' into main
steinnhauser Dec 22, 2024
03a8544
Merge branch 'mne-tools:main' into main
steinnhauser Jan 1, 2025
4a6f4e8
MAINT: Satisfy CircleCI
larsoner Jan 7, 2025
28e45ca
FIX: Better
larsoner Jan 7, 2025
6415d97
FIX: Wording
larsoner Jan 7, 2025
7508056
FIX: Docstring
larsoner Jan 7, 2025
6556418
FIX: Name
larsoner Jan 7, 2025
842132d
Merge branch 'mne-tools:main' into main
emma-bailey Jan 8, 2025
c8eb284
example: update comments
emma-bailey Jan 8, 2025
89278a6
tutorial: add ref to pca_obs in SSP tutorial section on ECG artefacts
emma-bailey Jan 8, 2025
15c7c5c
Merge branch 'mne-tools:main' into main
steinnhauser Jan 10, 2025
36e0ca0
Update mne/preprocessing/pca_obs.py
steinnhauser Jan 10, 2025
fc5bdc8
Update tutorials/preprocessing/50_artifact_correction_ssp.py
steinnhauser Jan 10, 2025
d586d62
Update mne/preprocessing/pca_obs.py
steinnhauser Jan 10, 2025
d6ae456
refactor: rework qrs_indices to qrs_times, adjust sanity checks, add …
steinnhm Jan 10, 2025
420f4fc
test,example: adjust example and tests to new qrs_times changes, add …
steinnhm Jan 10, 2025
91b0033
Merge branch 'mne-tools:main' into main
steinnhauser Jan 10, 2025
54d67aa
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 10, 2025
f4082b4
feat: add copy kwarg, optionally return new instance, add verbose dec…
steinnhm Jan 10, 2025
0e43130
Merge branch 'main' of https://github.com/emma-bailey/mne-python
steinnhm Jan 10, 2025
fd9c41b
test,example: update tests and example with new copy default behavior…
steinnhm Jan 10, 2025
a578a7d
[pre-commit.ci] auto fixes from pre-commit.com hooks
pre-commit-ci[bot] Jan 10, 2025
41559ff
lint: resolve pre-commit errors, format
steinnhm Jan 10, 2025
0079605
Merge branch 'main' of https://github.com/emma-bailey/mne-python
steinnhm Jan 10, 2025
d170c39
fix: remove conditional raw return, always return raw object regardle…
steinnhm Jan 13, 2025
b70545b
lint: resolve linter errors
steinnhm Jan 13, 2025
39c322f
FIX: Doc
larsoner Jan 14, 2025
f62f98b
[autofix.ci] apply automated fixes
autofix-ci[bot] Jan 14, 2025
ebda34e
FIX: How
larsoner Jan 14, 2025
72facf0
Merge remote-tracking branch 'upstream/main' into ebm
larsoner Jan 14, 2025
92fe25c
TST: Pre
larsoner Jan 14, 2025
d38e37f
DOC: whats new
larsoner Jan 14, 2025
2472c84
FIX: Name
larsoner Jan 14, 2025
49babe1
Merge branch 'mne-tools:main' into main
steinnhauser Jan 15, 2025
dab5623
Apply suggestions from code review
larsoner Jan 16, 2025
525a8e7
Merge branch 'mne-tools:main' into main
steinnhauser Jan 17, 2025
a8e84ff
Update examples/preprocessing/esg_rm_heart_artefact_pcaobs.py
steinnhauser Jan 17, 2025
12b3834
Update examples/preprocessing/esg_rm_heart_artefact_pcaobs.py
steinnhauser Jan 17, 2025
c1106c2
Apply suggestions from code review
steinnhauser Jan 17, 2025
a4ee593
FIX: Move
larsoner Jan 17, 2025
728a010
Merge branch 'main' into main
larsoner Jan 17, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 8 additions & 1 deletion .circleci/config.yml
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,9 @@ jobs:
- restore_cache:
keys:
- data-cache-phantom-kit
- restore_cache:
keys:
- data-cache-ds004388
- run:
name: Get data
# This limit could be increased, but this is helpful for finding slow ones
Expand Down Expand Up @@ -252,7 +255,7 @@ jobs:
name: Check sphinx log for warnings (which are treated as errors)
when: always
command: |
! grep "^.* (WARNING|ERROR): .*$" sphinx_log.txt
! grep "^.*\(WARNING\|ERROR\): " sphinx_log.txt
- run:
name: Show profiling output
when: always
Expand Down Expand Up @@ -393,6 +396,10 @@ jobs:
key: data-cache-phantom-kit
paths:
- ~/mne_data/MNE-phantom-KIT-data # (1 G)
- save_cache:
key: data-cache-ds004388
paths:
- ~/mne_data/ds004388 # (1.8 G)


linkcheck:
Expand Down
1 change: 1 addition & 0 deletions doc/api/datasets.rst
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ Datasets
brainstorm.bst_auditory.data_path
brainstorm.bst_resting.data_path
brainstorm.bst_raw.data_path
default_path
eegbci.load_data
eegbci.standardize
fetch_aparc_sub_parcellation
Expand Down
1 change: 1 addition & 0 deletions doc/api/preprocessing.rst
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,7 @@ Projections:
read_ica_eeglab
read_fine_calibration
write_fine_calibration
apply_pca_obs

:py:mod:`mne.preprocessing.nirs`:

Expand Down
1 change: 1 addition & 0 deletions doc/changes/devel/13037.newfeature.rst
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Add PCA-OBS preprocessing for the removal of heart-artefacts from EEG or ESG datasets via :func:`mne.preprocessing.apply_pca_obs`, by :newcontrib:`Emma Bailey` and :newcontrib:`Steinn Hauser Magnusson`.
2 changes: 2 additions & 0 deletions doc/changes/names.inc
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@
.. _Eberhard Eich: https://github.com/ebeich
.. _Eduard Ort: https://github.com/eort
.. _Emily Stephen: https://github.com/emilyps14
.. _Emma Bailey: https://www.cbs.mpg.de/employees/bailey
.. _Enrico Varano: https://github.com/enricovara/
.. _Enzo Altamiranda: https://www.linkedin.com/in/enzoalt
.. _Eric Larson: https://larsoner.com
Expand Down Expand Up @@ -284,6 +285,7 @@
.. _Stanislas Chambon: https://github.com/Slasnista
.. _Stefan Appelhoff: https://stefanappelhoff.com
.. _Stefan Repplinger: https://github.com/stfnrpplngr
.. _Steinn Hauser Magnusson: https://github.com/steinnhauser
.. _Steven Bethard: https://github.com/bethard
.. _Steven Bierer: https://github.com/neurolaunch
.. _Steven Gutstein: https://github.com/smgutstein
Expand Down
1 change: 1 addition & 0 deletions doc/conf.py
Original file line number Diff line number Diff line change
Expand Up @@ -355,6 +355,7 @@
"n_frequencies",
"n_tests",
"n_samples",
"n_peaks",
"n_permutations",
"nchan",
"n_points",
Expand Down
10 changes: 10 additions & 0 deletions doc/references.bib
Original file line number Diff line number Diff line change
Expand Up @@ -1335,6 +1335,16 @@ @inproceedings{NdiayeEtAl2016
year = {2016}
}

@article{NiazyEtAl2005,
author = {Niazy, R. K. and Beckmann, C.F. and Iannetti, G.D. and Brady, J. M. and Smith, S. M.},
title = {Removal of FMRI environment artifacts from EEG data using optimal basis sets},
journal = {NeuroImage},
year = {2005},
volume = {28},
pages = {720-737},
doi = {10.1016/j.neuroimage.2005.06.067.}
}

@article{NicholsHolmes2002,
author = {Nichols, Thomas E. and Holmes, Andrew P.},
doi = {10.1002/hbm.1058},
Expand Down
196 changes: 196 additions & 0 deletions examples/preprocessing/esg_rm_heart_artefact_pcaobs.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,196 @@
"""
.. _ex-pcaobs:

=====================================================================================
Principal Component Analysis - Optimal Basis Sets (PCA-OBS) removing cardiac artefact
=====================================================================================

This script shows an example of how to use an adaptation of PCA-OBS
:footcite:`NiazyEtAl2005`. PCA-OBS was originally designed to remove
the ballistocardiographic artefact in simultaneous EEG-fMRI. Here, it
has been adapted to remove the delay between the detected R-peak and the
ballistocardiographic artefact such that the algorithm can be applied to
remove the cardiac artefact in EEG (electroencephalography) and ESG
(electrospinography) data. We will illustrate how it works by applying the
algorithm to ESG data, where the effect of removal is most pronounced.

See: https://www.biorxiv.org/content/10.1101/2024.09.05.611423v1
for more details on the dataset and application for ESG data.

"""

# Authors: Emma Bailey <[email protected]>,
# Steinn Hauser Magnusson <[email protected]>
# License: BSD-3-Clause
# Copyright the MNE-Python contributors.

import glob

import numpy as np

# %%
# Download sample subject data from OpenNeuro if you haven't already.
# This will download simultaneous EEG and ESG data from a single run of a
# single participant after median nerve stimulation of the left wrist.
import openneuro
from matplotlib import pyplot as plt

import mne
from mne import Epochs, events_from_annotations
from mne.io import read_raw_eeglab
from mne.preprocessing import find_ecg_events, fix_stim_artifact

# add the path where you want the OpenNeuro data downloaded. Each run is ~2GB of data
ds = "ds004388"
target_dir = mne.datasets.default_path() / ds
run_name = "sub-001/eeg/*median_run-03_eeg*.set"
if not glob.glob(str(target_dir / run_name)):
target_dir.mkdir(exist_ok=True)
openneuro.download(dataset=ds, target_dir=target_dir, include=run_name[:-4])
block_files = glob.glob(str(target_dir / run_name))
assert len(block_files) == 1

# %%
# Define the esg channels (arranged in two patches over the neck and lower back).

esg_chans = [
"S35",
"S24",
"S36",
"Iz",
"S17",
"S15",
"S32",
"S22",
"S19",
"S26",
"S28",
"S9",
"S13",
"S11",
"S7",
"SC1",
"S4",
"S18",
"S8",
"S31",
"SC6",
"S12",
"S16",
"S5",
"S30",
"S20",
"S34",
"S21",
"S25",
"L1",
"S29",
"S14",
"S33",
"S3",
"L4",
"S6",
"S23",
]

# Interpolation window in seconds for ESG data to remove stimulation artefact
tstart_esg = -7e-3
tmax_esg = 7e-3

# Define timing of heartbeat epochs in seconds relative to R-peaks
iv_baseline = [-400e-3, -300e-3]
iv_epoch = [-400e-3, 600e-3]

# %%
# Next, we perform minimal preprocessing including removing the
# stimulation artefact, downsampling and filtering.

raw = read_raw_eeglab(block_files[0], verbose="error")
raw.set_channel_types(dict(ECG="ecg"))
# Isolate the ESG channels (include the ECG channel for R-peak detection)
raw.pick(esg_chans + ["ECG"])
# Trim duration and downsample (from 10kHz) to improve example speed
raw.crop(0, 60).load_data().resample(2000)

# Find trigger timings to remove the stimulation artefact
events, event_dict = events_from_annotations(raw)
trigger_name = "Median - Stimulation"

fix_stim_artifact(
raw,
events=events,
event_id=event_dict[trigger_name],
tmin=tstart_esg,
tmax=tmax_esg,
mode="linear",
stim_channel=None,
)

# %%
# Find ECG events and add to the raw structure as event annotations.

ecg_events, ch_ecg, average_pulse = find_ecg_events(raw, ch_name="ECG")
ecg_event_samples = np.asarray(
[[ecg_event[0] for ecg_event in ecg_events]]
) # Samples only

qrs_event_time = [
x / raw.info["sfreq"] for x in ecg_event_samples.reshape(-1)
] # Divide by sampling rate to make times
duration = np.repeat(0.0, len(ecg_event_samples))
description = ["qrs"] * len(ecg_event_samples)

raw.annotations.append(
qrs_event_time, duration, description, ch_names=[esg_chans] * len(qrs_event_time)
)

# %%
# Create evoked response around the detected R-peaks
# before and after cardiac artefact correction.

events, event_ids = events_from_annotations(raw)
event_id_dict = {key: value for key, value in event_ids.items() if key == "qrs"}
epochs = Epochs(
raw,
events,
event_id=event_id_dict,
tmin=iv_epoch[0],
tmax=iv_epoch[1],
baseline=tuple(iv_baseline),
)
evoked_before = epochs.average()

# Apply function - modifies the data in place. Optionally high-pass filter
# the data before applying PCA-OBS to remove low frequency drifts
raw = mne.preprocessing.apply_pca_obs(
raw, picks=esg_chans, n_jobs=5, qrs_times=raw.times[ecg_event_samples.reshape(-1)]
)

epochs = Epochs(
raw,
events,
event_id=event_id_dict,
tmin=iv_epoch[0],
tmax=iv_epoch[1],
baseline=tuple(iv_baseline),
)
evoked_after = epochs.average()

# %%
# Compare evoked responses to assess completeness of artefact removal.

fig, axes = plt.subplots(1, 1, layout="constrained")
data_before = evoked_before.get_data(units=dict(eeg="uV")).T
data_after = evoked_after.get_data(units=dict(eeg="uV")).T
hs = list()
hs.append(axes.plot(epochs.times, data_before, color="k")[0])
hs.append(axes.plot(epochs.times, data_after, color="green", label="after")[0])
axes.set(ylim=[-500, 1000], ylabel="Amplitude (µV)", xlabel="Time (s)")
axes.set(title="ECG artefact removal using PCA-OBS")
axes.legend(hs, ["before", "after"])
plt.show()

# %%
# References
# ----------
# .. footbibliography::
2 changes: 2 additions & 0 deletions mne/datasets/__init__.pyi
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ __all__ = [
"epilepsy_ecog",
"erp_core",
"eyelink",
"default_path",
"fetch_aparc_sub_parcellation",
"fetch_dataset",
"fetch_fsaverage",
Expand Down Expand Up @@ -70,6 +71,7 @@ from ._infant import fetch_infant_template
from ._phantom.base import fetch_phantom
from .utils import (
_download_all_example_data,
default_path,
fetch_aparc_sub_parcellation,
fetch_hcp_mmp_parcellation,
has_dataset,
Expand Down
30 changes: 29 additions & 1 deletion mne/datasets/utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
# License: BSD-3-Clause
# Copyright the MNE-Python contributors.

import glob
import importlib
import inspect
import logging
Expand Down Expand Up @@ -92,6 +93,22 @@ def _dataset_version(path, name):
return version


@verbose
def default_path(*, verbose=None):
"""Get the default MNE_DATA path.

Parameters
----------
%(verbose)s

Returns
-------
data_path : instance of Path
Path to the default MNE_DATA directory.
"""
return _get_path(None, None, None)


def _get_path(path, key, name):
"""Get a dataset path."""
# 1. Input
Expand All @@ -113,7 +130,8 @@ def _get_path(path, key, name):
return path
# 4. ~/mne_data (but use a fake home during testing so we don't
# unnecessarily create ~/mne_data)
logger.info(f"Using default location ~/mne_data for {name}...")
extra = f" for {name}" if name else ""
logger.info(f"Using default location ~/mne_data{extra}...")
path = Path(os.getenv("_MNE_FAKE_HOME_DIR", "~")).expanduser() / "mne_data"
if not path.is_dir():
logger.info(f"Creating {path}")
Expand Down Expand Up @@ -319,6 +337,8 @@ def _download_all_example_data(verbose=True):
#
# verbose=True by default so we get nice status messages.
# Consider adding datasets from here to CircleCI for PR-auto-build
import openneuro
drammock marked this conversation as resolved.
Show resolved Hide resolved

paths = dict()
for kind in (
"sample testing misc spm_face somato hf_sef multimodal "
Expand Down Expand Up @@ -375,6 +395,14 @@ def _download_all_example_data(verbose=True):
limo.load_data(subject=1, update_path=True)
logger.info("[done limo]")

# for ESG
ds = "ds004388"
target_dir = default_path() / ds
run_name = "sub-001/eeg/*median_run-03_eeg*.set"
if not glob.glob(str(target_dir / run_name)):
target_dir.mkdir(exist_ok=True)
openneuro.download(dataset=ds, target_dir=target_dir, include=run_name[:-4])
drammock marked this conversation as resolved.
Show resolved Hide resolved


@verbose
def fetch_aparc_sub_parcellation(subjects_dir=None, verbose=None):
Expand Down
2 changes: 2 additions & 0 deletions mne/preprocessing/__init__.pyi
Original file line number Diff line number Diff line change
Expand Up @@ -44,6 +44,7 @@ __all__ = [
"realign_raw",
"regress_artifact",
"write_fine_calibration",
"apply_pca_obs",
]
from . import eyetracking, ieeg, nirs
from ._annotate_amplitude import annotate_amplitude
Expand All @@ -56,6 +57,7 @@ from ._fine_cal import (
write_fine_calibration,
)
from ._lof import find_bad_channels_lof
from ._pca_obs import apply_pca_obs
from ._peak_finder import peak_finder
from ._regress import EOGRegression, read_eog_regression, regress_artifact
from .artifact_detection import (
Expand Down
Loading
Loading