31
31
"""
32
32
import typing as ty
33
33
34
- import nibabel as nb
35
- import numpy as np
36
34
from nipype .interfaces import utility as niu
37
35
from nipype .pipeline import engine as pe
38
36
from niworkflows .utils .connections import listify
@@ -295,44 +293,20 @@ def init_bold_wf(
295
293
fieldmap_id = fieldmap_id ,
296
294
omp_nthreads = omp_nthreads ,
297
295
)
298
- bold_anat_wf = init_bold_volumetric_resample_wf (
299
- metadata = all_metadata [0 ],
300
- fieldmap_id = fieldmap_id if not multiecho else None ,
301
- omp_nthreads = omp_nthreads ,
302
- mem_gb = mem_gb ,
303
- name = 'bold_anat_wf' ,
304
- )
305
- bold_anat_wf .inputs .inputnode .resolution = "native"
306
296
307
297
workflow .connect ([
308
298
(inputnode , bold_native_wf , [
309
299
("fmap_ref" , "inputnode.fmap_ref" ),
310
300
("fmap_coeff" , "inputnode.fmap_coeff" ),
311
301
("fmap_id" , "inputnode.fmap_id" ),
312
302
]),
313
- (inputnode , bold_anat_wf , [
314
- ("t1w_preproc" , "inputnode.target_ref_file" ),
315
- ("t1w_mask" , "inputnode.target_mask" ),
316
- ("fmap_ref" , "inputnode.fmap_ref" ),
317
- ("fmap_coeff" , "inputnode.fmap_coeff" ),
318
- ("fmap_id" , "inputnode.fmap_id" ),
319
- ]),
320
303
(bold_fit_wf , bold_native_wf , [
321
304
("outputnode.coreg_boldref" , "inputnode.boldref" ),
322
305
("outputnode.bold_mask" , "inputnode.bold_mask" ),
323
306
("outputnode.motion_xfm" , "inputnode.motion_xfm" ),
324
307
("outputnode.boldref2fmap_xfm" , "inputnode.boldref2fmap_xfm" ),
325
308
("outputnode.dummy_scans" , "inputnode.dummy_scans" ),
326
309
]),
327
- (bold_fit_wf , bold_anat_wf , [
328
- ("outputnode.coreg_boldref" , "inputnode.bold_ref_file" ),
329
- ("outputnode.boldref2fmap_xfm" , "inputnode.boldref2fmap_xfm" ),
330
- ("outputnode.boldref2anat_xfm" , "inputnode.boldref2anat_xfm" ),
331
- ]),
332
- (bold_native_wf , bold_anat_wf , [
333
- ("outputnode.bold_minimal" , "inputnode.bold_file" ),
334
- ("outputnode.motion_xfm" , "inputnode.motion_xfm" ),
335
- ]),
336
310
]) # fmt:skip
337
311
338
312
boldref_out = bool (nonstd_spaces .intersection (('func' , 'run' , 'bold' , 'boldref' , 'sbref' )))
@@ -400,6 +374,35 @@ def init_bold_wf(
400
374
if config .workflow .level == "resampling" :
401
375
return workflow
402
376
377
+ # Resample to anatomical space
378
+ bold_anat_wf = init_bold_volumetric_resample_wf (
379
+ metadata = all_metadata [0 ],
380
+ fieldmap_id = fieldmap_id if not multiecho else None ,
381
+ omp_nthreads = omp_nthreads ,
382
+ mem_gb = mem_gb ,
383
+ name = 'bold_anat_wf' ,
384
+ )
385
+ bold_anat_wf .inputs .inputnode .resolution = "native"
386
+
387
+ workflow .connect ([
388
+ (inputnode , bold_anat_wf , [
389
+ ("t1w_preproc" , "inputnode.target_ref_file" ),
390
+ ("t1w_mask" , "inputnode.target_mask" ),
391
+ ("fmap_ref" , "inputnode.fmap_ref" ),
392
+ ("fmap_coeff" , "inputnode.fmap_coeff" ),
393
+ ("fmap_id" , "inputnode.fmap_id" ),
394
+ ]),
395
+ (bold_fit_wf , bold_anat_wf , [
396
+ ("outputnode.coreg_boldref" , "inputnode.bold_ref_file" ),
397
+ ("outputnode.boldref2fmap_xfm" , "inputnode.boldref2fmap_xfm" ),
398
+ ("outputnode.boldref2anat_xfm" , "inputnode.boldref2anat_xfm" ),
399
+ ]),
400
+ (bold_native_wf , bold_anat_wf , [
401
+ ("outputnode.bold_minimal" , "inputnode.bold_file" ),
402
+ ("outputnode.motion_xfm" , "inputnode.motion_xfm" ),
403
+ ]),
404
+ ]) # fmt:skip
405
+
403
406
# Full derivatives, including resampled BOLD series
404
407
if nonstd_spaces .intersection (('anat' , 'T1w' )):
405
408
ds_bold_t1_wf = init_ds_volumes_wf (
@@ -507,7 +510,11 @@ def init_bold_wf(
507
510
]) # fmt:skip
508
511
509
512
if config .workflow .cifti_output :
510
- from .resampling import init_bold_fsLR_resampling_wf , init_bold_grayords_wf
513
+ from .resampling import (
514
+ init_bold_fsLR_resampling_wf ,
515
+ init_bold_grayords_wf ,
516
+ init_goodvoxels_bold_mask_wf ,
517
+ )
511
518
512
519
bold_MNI6_wf = init_bold_volumetric_resample_wf (
513
520
metadata = all_metadata [0 ],
@@ -518,12 +525,29 @@ def init_bold_wf(
518
525
)
519
526
520
527
bold_fsLR_resampling_wf = init_bold_fsLR_resampling_wf (
521
- estimate_goodvoxels = config .workflow .project_goodvoxels ,
522
528
grayord_density = config .workflow .cifti_output ,
523
529
omp_nthreads = omp_nthreads ,
524
530
mem_gb = mem_gb ["resampled" ],
525
531
)
526
532
533
+ if config .workflow .project_goodvoxels :
534
+ goodvoxels_bold_mask_wf = init_goodvoxels_bold_mask_wf (mem_gb ["resampled" ])
535
+
536
+ workflow .connect ([
537
+ (inputnode , goodvoxels_bold_mask_wf , [("anat_ribbon" , "inputnode.anat_ribbon" )]),
538
+ (bold_anat_wf , goodvoxels_bold_mask_wf , [
539
+ ("outputnode.bold_file" , "inputnode.bold_file" ),
540
+ ]),
541
+ (goodvoxels_bold_mask_wf , bold_fsLR_resampling_wf , [
542
+ ("outputnode.goodvoxels_mask" , "inputnode.volume_roi" ),
543
+ ]),
544
+ ]) # fmt:skip
545
+
546
+ bold_fsLR_resampling_wf .__desc__ += """\
547
+ A "goodvoxels" mask was applied during volume-to-surface sampling in fsLR space,
548
+ excluding voxels whose time-series have a locally high coefficient of variation.
549
+ """
550
+
527
551
bold_grayords_wf = init_bold_grayords_wf (
528
552
grayord_density = config .workflow .cifti_output ,
529
553
mem_gb = 1 ,
@@ -573,7 +597,6 @@ def init_bold_wf(
573
597
("midthickness_fsLR" , "inputnode.midthickness_fsLR" ),
574
598
("sphere_reg_fsLR" , "inputnode.sphere_reg_fsLR" ),
575
599
("cortex_mask" , "inputnode.cortex_mask" ),
576
- ("anat_ribbon" , "inputnode.anat_ribbon" ),
577
600
]),
578
601
(bold_anat_wf , bold_fsLR_resampling_wf , [
579
602
("outputnode.bold_file" , "inputnode.bold_file" ),
0 commit comments