import numpy as np
from mne.io import read_raw_edf
from mne.channels import make_standard_montage
from .base import _EEGDataset, DATA_PATH
from .download import data_dl
from ..utils import get_init_args
from ..tools.docs import fill_doc
URL = "https://physionet.org/files/"
[docs]
@fill_doc
class PhysioNet_MI(_EEGDataset):
"""Physionet Motor Imagery dataset.
.. admonition:: Dataset summary
====== ====== ======= ===== ======== ======
Subj Chan Time Cls Freq Sess
====== ====== ======= ===== ======== ======
109 64 3.0 s 4 160 Hz 1
====== ====== ======= ===== ======== ======
Physionet MI dataset: https://physionet.org/pn4/eegmmidb/
This data set consists of over 1500 one- and two-minute EEG recordings,
obtained from 109 volunteers [2]_.
Subjects performed different motor/imagery tasks while 64-channel EEG were
recorded using the BCI2000 system (http://www.bci2000.org) [1]_.
Each subject performed 14 experimental runs: two one-minute baseline runs
(one with eyes open, one with eyes closed), and three two-minute runs of
each of the four following tasks:
1. A target appears on either the left or the right side of the screen.
The subject opens and closes the corresponding fist until the target
disappears. Then the subject relaxes.
2. A target appears on either the left or the right side of the screen.
The subject imagines opening and closing the corresponding fist until
the target disappears. Then the subject relaxes.
3. A target appears on either the top or the bottom of the screen.
The subject opens and closes either both fists (if the target is on top)
or both feet (if the target is on the bottom) until the target
disappears. Then the subject relaxes.
4. A target appears on either the top or the bottom of the screen.
The subject imagines opening and closing either both fists
(if the target is on top) or both feet (if the target is on the bottom)
until the target disappears. Then the subject relaxes.
references
----------
.. [1] Schalk, G., McFarland, D.J., Hinterberger, T., Birbaumer, N. and
Wolpaw, J.R., 2004. BCI2000: a general-purpose brain-computer
interface (BCI) system. IEEE Transactions on biomedical engineering,
51(6), pp.1034-1043.
.. [2] Goldberger, A.L., Amaral, L.A., Glass, L., Hausdorff, J.M., Ivanov,
P.C., Mark, R.G., Mietus, J.E., Moody, G.B., Peng, C.K., Stanley,
H.E. and PhysioBank, P., PhysioNet: components of a new research
resource for complex physiologic signals Circulation 2000 Volume
101 Issue 23 pp. E215-E220.
Parameters
----------
%(subjects)s
%(epochs_tmin_tmax)s
%(baseline)s
%(picks)s
%(resample)s
imagined: bool (default True)
if True, return runs corresponding to motor imagination.
executed: bool (default False)
if True, return runs corresponding to motor execution.
%(rename)s
"""
def __init__(
self,
subjects: list[int] | None = None,
tmin: float = 0,
tmax: float | None = None,
baseline: tuple[int, int] | None = None,
picks: list[str] | None = None,
resample: float | None = None,
imagined: bool = True,
executed: bool = False,
rename: str | None = None,
) -> None:
super().__init__(
repr=get_init_args(self, locals(), rename=rename, ret_dict=True),
subject_list=list(range(1, 110)),
interval=[0.0, 3.0],
event_id={
"rest": 1,
"left_hand": 2,
"right_hand": 3,
"hands": 4,
"feet": 5,
},
subjects=subjects,
tmin=tmin,
tmax=tmax,
baseline=baseline,
picks=picks,
resample=resample,
)
self._data_path = DATA_PATH / "physionet_mi"
self._data_url = f"{URL}eegmmidb/1.0.0/"
# fmt: off
self._renames = {
"AFZ": "AFz", "PZ": "Pz", "FPZ": "Fpz", "FCZ": "FCz", "FP1": "Fp1",
"CZ": "Cz", "OZ": "Oz", "POZ": "POz", "IZ": "Iz", "CPZ": "CPz",
"FP2": "Fp2", "FZ": "Fz",
}
# fmt: on
self._montage = make_standard_montage("standard_1005")
self._feet_runs = []
self._hand_runs = []
if imagined:
self._feet_runs += [6, 10, 14]
self._hand_runs += [4, 8, 12]
if executed:
self._feet_runs += [5, 9, 13]
self._hand_runs += [3, 7, 11]
def _load_run(self, subject, run, verbose):
filename = data_dl(
f"{self._data_url}S{subject:03d}/S{subject:03d}R{run:02d}.edf",
self._data_path,
force_update=False,
)
raw = read_raw_edf(filename, preload=True, verbose=verbose)
raw.rename_channels(lambda x: x.strip("."))
raw.rename_channels(lambda x: x.upper())
raw.rename_channels(self._renames)
raw.set_montage(self._montage)
return raw
def _get_single_subject_raw(self, subject: int, verbose="ERROR"):
session = {}
# hand runs
idx = 1
for run in self._hand_runs:
raw = self._load_run(subject, run, verbose)
stim = raw.annotations.description.astype(np.dtype("<U10"))
stim[stim == "T0"] = "rest"
stim[stim == "T1"] = "left_hand"
stim[stim == "T2"] = "right_hand"
raw.annotations.description = stim
session[f"run_{idx}"] = raw
idx += 1
# feet runs
for run in self._feet_runs:
raw = self._load_run(subject, run, verbose)
stim = raw.annotations.description.astype(np.dtype("<U10"))
stim[stim == "T0"] = "rest"
stim[stim == "T1"] = "hands"
stim[stim == "T2"] = "feet"
raw.annotations.description = stim
session[f"run_{idx}"] = raw
idx += 1
return {"session_1": session}