-
Notifications
You must be signed in to change notification settings - Fork 299
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
[ENH] Replace FSL mcflirt
with AFNI 3dVolReg
#1283
Changes from 3 commits
682e60d
829d762
7874bc7
4450915
e2abc33
ec9da7e
0a90c83
ca7d364
add3e00
0420e3d
5d70673
27b824f
4abb15c
3d93a9c
3a8a2fa
61b094b
1ce323b
7a5fae9
c0db97f
b86e278
40a77b6
1d77247
145bde0
e9de427
bc32b8b
93c712b
649737c
93fc896
55b4236
bc48f64
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -293,6 +293,7 @@ def _run_interface(self, runtime): | |
|
||
class ValidateImageInputSpec(BaseInterfaceInputSpec): | ||
in_file = File(exists=True, mandatory=True, desc='input image') | ||
deoblique = traits.Bool(False, usedefault=True) | ||
|
||
|
||
class ValidateImageOutputSpec(TraitedSpec): | ||
|
@@ -342,6 +343,15 @@ class ValidateImage(SimpleInterface): | |
output_spec = ValidateImageOutputSpec | ||
|
||
def _run_interface(self, runtime): | ||
|
||
def deoblique(img): | ||
import os | ||
import nibabel as nb | ||
import numpy as np | ||
affine = img.affine | ||
affine[:3, :3] = np.diag(np.diag(affine[:3, :3])) | ||
return nb.Nifti1Image(np.asanyarray(img.dataobj), affine, img.header) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would prefer |
||
|
||
img = nb.load(self.inputs.in_file) | ||
out_report = os.path.join(runtime.cwd, 'report.html') | ||
|
||
|
@@ -370,7 +380,13 @@ def _run_interface(self, runtime): | |
|
||
# Both match, qform valid (implicit with match), codes okay -> do nothing, empty report | ||
if matching_affines and qform_code > 0 and sform_code > 0: | ||
self._results['out_file'] = self.inputs.in_file | ||
if self.inputs.deoblique: | ||
out_fname = fname_presuffix(self.inputs.in_file, suffix='_valid', newpath=runtime.cwd) | ||
img = deoblique(img) | ||
img.to_filename(out_fname) | ||
self._results['out_file'] = out_fname | ||
else: | ||
self._results['out_file'] = self.inputs.in_file | ||
open(out_report, 'w').close() | ||
self._results['out_report'] = out_report | ||
return runtime | ||
|
@@ -418,6 +434,9 @@ def _run_interface(self, runtime): | |
</p> | ||
""" | ||
snippet = '<h3 class="elem-title">%s</h3>\n%s\n' % (warning_txt, description) | ||
|
||
if self.inputs.deoblique: | ||
img = deoblique(img) | ||
# Store new file and report | ||
img.to_filename(out_fname) | ||
with open(out_report, 'w') as fobj: | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -10,8 +10,9 @@ | |
""" | ||
|
||
from nipype.pipeline import engine as pe | ||
from nipype.interfaces import utility as niu, fsl | ||
from nipype.interfaces import utility as niu, fsl, afni | ||
from niworkflows.interfaces import NormalizeMotionParams | ||
from fmriprep.utils.misc import afni2itk_func | ||
from ...engine import Workflow | ||
from ...interfaces import MCFLIRT2ITK | ||
|
||
|
@@ -73,24 +74,24 @@ def init_bold_hmc_wf(mem_gb, omp_nthreads, name='bold_hmc_wf'): | |
name='outputnode') | ||
|
||
# Head motion correction (hmc) | ||
mcflirt = pe.Node(fsl.MCFLIRT(save_mats=True, save_plots=True), | ||
name='mcflirt', mem_gb=mem_gb * 3) | ||
mc = pe.Node(afni.Volreg(args='-prefix NULL -twopass', | ||
zpad=4, outputtype='NIFTI_GZ'), name="mc", mem_gb=mem_gb * 3) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Do we need to specify There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. No, but we never use the output so there is no need for saving it (even if Nipype will delete it anyway). |
||
|
||
fsl2itk = pe.Node(MCFLIRT2ITK(), name='fsl2itk', | ||
mem_gb=0.05, n_procs=omp_nthreads) | ||
afni2itk = pe.Node(niu.Function(function=afni2itk_func, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If there's nothing against it, I'll convert this function node into a VOLREG2ITK fully-fledged nipype interface before merging in, similarly to MCFLIRT2ITK. I can remember that parallelizing that one was tricky but paid off with an important speed up on long datasets. |
||
input_names=["in_file"], | ||
output_names=["out_files"]), name='afni2itk', | ||
mem_gb=0.05) | ||
|
||
normalize_motion = pe.Node(NormalizeMotionParams(format='FSL'), | ||
normalize_motion = pe.Node(NormalizeMotionParams(format='AFNI'), | ||
name="normalize_motion", | ||
mem_gb=DEFAULT_MEMORY_MIN_GB) | ||
|
||
workflow.connect([ | ||
(inputnode, mcflirt, [('raw_ref_image', 'ref_file'), | ||
('bold_file', 'in_file')]), | ||
(inputnode, fsl2itk, [('raw_ref_image', 'in_source'), | ||
('raw_ref_image', 'in_reference')]), | ||
(mcflirt, fsl2itk, [('mat_file', 'in_files')]), | ||
(mcflirt, normalize_motion, [('par_file', 'in_file')]), | ||
(fsl2itk, outputnode, [('out_file', 'xforms')]), | ||
(inputnode, mc, [('raw_ref_image', 'basefile'), | ||
('bold_file', 'in_file')]), | ||
(mc, afni2itk, [('oned_matrix_save', 'in_file')]), | ||
(mc, normalize_motion, [('oned_file', 'in_file')]), | ||
(afni2itk, outputnode, [('out_files', 'xforms')]), | ||
(normalize_motion, outputnode, [('out_file', 'movpar_file')]), | ||
]) | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This will simply zero out the off-diagonal, right? That will end up producing an affine with smaller zooms than the original. Probably the correct way to do this is to do a rotation/zoom/shear/translation decomposition, and then to remove the rotations and shears.
That said, this is less a de-oblique than a removal of information that describes obliqueness. Is this what you want, and what's the reason for that?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes the intention was to remove off diagonal values (because otherwise AFNI doesn't generate transforms that can be applied by ANTs). Is there a python code snippet for rotation/zoom/shear/translation decomposition I could use?