|
23 | 23 | from mne._fiff.constants import FIFF
|
24 | 24 | from mne._fiff.meas_info import _empty_info
|
25 | 25 | from mne.channels import (
|
| 26 | + Layout, |
26 | 27 | find_layout,
|
27 | 28 | make_eeg_layout,
|
28 | 29 | make_grid_layout,
|
@@ -94,6 +95,18 @@ def _get_test_info():
|
94 | 95 | return test_info
|
95 | 96 |
|
96 | 97 |
|
| 98 | +@pytest.fixture(scope="module") |
| 99 | +def layout(): |
| 100 | + """Get a layout.""" |
| 101 | + return Layout( |
| 102 | + (0.1, 0.2, 0.1, 1.2), |
| 103 | + pos=np.array([[0, 0, 0.1, 0.1], [0.2, 0.2, 0.1, 0.1], [0.4, 0.4, 0.1, 0.1]]), |
| 104 | + names=["0", "1", "2"], |
| 105 | + ids=[0, 1, 2], |
| 106 | + kind="test", |
| 107 | + ) |
| 108 | + |
| 109 | + |
97 | 110 | def test_io_layout_lout(tmp_path):
|
98 | 111 | """Test IO with .lout files."""
|
99 | 112 | layout = read_layout(fname="Vectorview-all", scale=False)
|
@@ -224,23 +237,17 @@ def test_make_grid_layout(tmp_path):
|
224 | 237 |
|
225 | 238 | def test_find_layout():
|
226 | 239 | """Test finding layout."""
|
227 |
| - pytest.raises(ValueError, find_layout, _get_test_info(), ch_type="meep") |
| 240 | + with pytest.raises(ValueError, match="Invalid value for the 'ch_type'"): |
| 241 | + find_layout(_get_test_info(), ch_type="meep") |
228 | 242 |
|
229 | 243 | sample_info = read_info(fif_fname)
|
230 |
| - grads = pick_types(sample_info, meg="grad") |
231 |
| - sample_info2 = pick_info(sample_info, grads) |
232 |
| - |
233 |
| - mags = pick_types(sample_info, meg="mag") |
234 |
| - sample_info3 = pick_info(sample_info, mags) |
235 |
| - |
236 |
| - # mock new convention |
| 244 | + sample_info2 = pick_info(sample_info, pick_types(sample_info, meg="grad")) |
| 245 | + sample_info3 = pick_info(sample_info, pick_types(sample_info, meg="mag")) |
237 | 246 | sample_info4 = copy.deepcopy(sample_info)
|
238 |
| - for ii, name in enumerate(sample_info4["ch_names"]): |
| 247 | + for ii, name in enumerate(sample_info4["ch_names"]): # mock new convention |
239 | 248 | new = name.replace(" ", "")
|
240 | 249 | sample_info4["chs"][ii]["ch_name"] = new
|
241 |
| - |
242 |
| - eegs = pick_types(sample_info, meg=False, eeg=True) |
243 |
| - sample_info5 = pick_info(sample_info, eegs) |
| 250 | + sample_info5 = pick_info(sample_info, pick_types(sample_info, meg=False, eeg=True)) |
244 | 251 |
|
245 | 252 | lout = find_layout(sample_info, ch_type=None)
|
246 | 253 | assert lout.kind == "Vectorview-all"
|
@@ -404,3 +411,100 @@ def test_generate_2d_layout():
|
404 | 411 | # Make sure background image normalizing is correct
|
405 | 412 | lt_bg = generate_2d_layout(xy, bg_image=bg_image)
|
406 | 413 | assert_allclose(lt_bg.pos[:, :2].max(), xy.max() / float(sbg))
|
| 414 | + |
| 415 | + |
| 416 | +def test_layout_copy(layout): |
| 417 | + """Test copying a layout.""" |
| 418 | + layout2 = layout.copy() |
| 419 | + assert_allclose(layout.pos, layout2.pos) |
| 420 | + assert layout.names == layout2.names |
| 421 | + layout2.names[0] = "foo" |
| 422 | + layout2.pos[0, 0] = 0.8 |
| 423 | + assert layout.names != layout2.names |
| 424 | + assert layout.pos[0, 0] != layout2.pos[0, 0] |
| 425 | + |
| 426 | + |
| 427 | +@pytest.mark.parametrize( |
| 428 | + "picks, exclude", |
| 429 | + [ |
| 430 | + ([0, 1], ()), |
| 431 | + (["0", 1], ()), |
| 432 | + (None, ["2"]), |
| 433 | + (None, "2"), |
| 434 | + (None, [2]), |
| 435 | + (None, 2), |
| 436 | + ("all", 2), |
| 437 | + ("all", "2"), |
| 438 | + (slice(0, 2), ()), |
| 439 | + (("0", "1"), ("0", "1")), |
| 440 | + (("0", 1), ("0", "1")), |
| 441 | + (("0", 1), (0, "1")), |
| 442 | + (set(["0", 1]), ()), |
| 443 | + (set([0, 1]), set()), |
| 444 | + (None, set([2])), |
| 445 | + (np.array([0, 1]), ()), |
| 446 | + (None, np.array([2])), |
| 447 | + (np.array(["0", "1"]), ()), |
| 448 | + ], |
| 449 | +) |
| 450 | +def test_layout_pick(layout, picks, exclude): |
| 451 | + """Test selection of channels in a layout.""" |
| 452 | + layout2 = layout.copy() |
| 453 | + layout2.pick(picks, exclude) |
| 454 | + assert layout2.names == layout.names[:2] |
| 455 | + assert_allclose(layout2.pos, layout.pos[:2, :]) |
| 456 | + |
| 457 | + |
| 458 | +def test_layout_pick_more(layout): |
| 459 | + """Test more channel selection in a layout.""" |
| 460 | + layout2 = layout.copy() |
| 461 | + layout2.pick(0) |
| 462 | + assert len(layout2.names) == 1 |
| 463 | + assert layout2.names[0] == layout.names[0] |
| 464 | + assert_allclose(layout2.pos, layout.pos[:1, :]) |
| 465 | + |
| 466 | + layout2 = layout.copy() |
| 467 | + layout2.pick("all", exclude=("0", "1")) |
| 468 | + assert len(layout2.names) == 1 |
| 469 | + assert layout2.names[0] == layout.names[2] |
| 470 | + assert_allclose(layout2.pos, layout.pos[2:, :]) |
| 471 | + |
| 472 | + layout2 = layout.copy() |
| 473 | + layout2.pick("all", exclude=("0", 1)) |
| 474 | + assert len(layout2.names) == 1 |
| 475 | + assert layout2.names[0] == layout.names[2] |
| 476 | + assert_allclose(layout2.pos, layout.pos[2:, :]) |
| 477 | + |
| 478 | + |
| 479 | +def test_layout_pick_errors(layout): |
| 480 | + """Test validation of layout.pick.""" |
| 481 | + with pytest.raises(TypeError, match="must be a list, tuple, set or ndarray"): |
| 482 | + layout.pick(lambda x: x) |
| 483 | + with pytest.raises(TypeError, match="must be a list, tuple, set or ndarray"): |
| 484 | + layout.pick(None, lambda x: x) |
| 485 | + with pytest.raises(TypeError, match="must be integers or strings"): |
| 486 | + layout.pick([0, lambda x: x]) |
| 487 | + with pytest.raises(TypeError, match="must be integers or strings"): |
| 488 | + layout.pick(None, [0, lambda x: x]) |
| 489 | + with pytest.raises(ValueError, match="does not match any channels"): |
| 490 | + layout.pick("foo") |
| 491 | + with pytest.raises(ValueError, match="does not match any channels"): |
| 492 | + layout.pick(None, "foo") |
| 493 | + with pytest.raises(ValueError, match="does not match any channels"): |
| 494 | + layout.pick(101) |
| 495 | + with pytest.raises(ValueError, match="does not match any channels"): |
| 496 | + layout.pick(None, 101) |
| 497 | + with pytest.warns(RuntimeWarning, match="has duplicates which will be ignored"): |
| 498 | + layout.copy().pick(["0", "0"]) |
| 499 | + with pytest.warns(RuntimeWarning, match="has duplicates which will be ignored"): |
| 500 | + layout.copy().pick(["0", 0]) |
| 501 | + with pytest.warns(RuntimeWarning, match="has duplicates which will be ignored"): |
| 502 | + layout.copy().pick(None, ["0", "0"]) |
| 503 | + with pytest.warns(RuntimeWarning, match="has duplicates which will be ignored"): |
| 504 | + layout.copy().pick(None, ["0", 0]) |
| 505 | + with pytest.raises(RuntimeError, match="selection yielded no remaining channels"): |
| 506 | + layout.copy().pick(None, ["0", "1", "2"]) |
| 507 | + with pytest.raises(ValueError, match="must be a 1D array-like"): |
| 508 | + layout.copy().pick(None, np.array([[0, 1]])) |
| 509 | + with pytest.raises(TypeError, match="slice of integers"): |
| 510 | + layout.copy().pick(slice("2342342342", 0, 3), ()) |
0 commit comments