diff --git a/CHANGELOG.rst b/CHANGELOG.rst index 51fa80ef936da0845a293c8b1813d1f9479b1b40..78b627b8ec8c27fdb1070606fb2590798ca179bf 100644 --- a/CHANGELOG.rst +++ b/CHANGELOG.rst @@ -3,8 +3,8 @@ order. -2.8.0 (Monday 27th January 2020) ---------------------------------- +2.8.0 (Wednesday 29th January 2020) +----------------------------------- Added @@ -15,6 +15,8 @@ Added header with adjusted shape, pixdims, and affine. This can be useful for creating a resampling reference. * New :func:`.affine.rescale` function, for adjusting a scaling matrix. +* New :func:`.mghimage.voxToSurfMat` function, for creating a + voxel-to-freesurfer affine transform from any image. Changed @@ -37,6 +39,8 @@ Fixed * Fixed a bug in the :class:`.Mesh` class to prevent indices from being loaded as floating point type. * Fixed a bug in the :func:`.resample` function. +* Fixed a bug in the :class:`.MGHImage` class, which was causing pixdims to + be overridden by scales derived from the affine. Deprecated diff --git a/fsl/data/mghimage.py b/fsl/data/mghimage.py index be29b5cad5ed56c789f528b478b19d9823093bb9..38f854446d3333cd7f150bd082d80d76df6cc236 100644 --- a/fsl/data/mghimage.py +++ b/fsl/data/mghimage.py @@ -38,7 +38,7 @@ class MGHImage(fslimage.Image): - http://nipy.org/nibabel/reference/nibabel.freesurfer.html """ - def __init__(self, image, *args, **kwargs): + def __init__(self, image, **kwargs): """Create a ``MGHImage``. :arg image: Name of MGH file, or a @@ -57,13 +57,32 @@ class MGHImage(fslimage.Image): data = np.asanyarray(image.dataobj) xform = image.affine + pixdim = image.header.get_zooms() vox2surf = image.header.get_vox2ras_tkr() + # the image may have an affine which + # transforms the data into some space + # with a scaling that is different to + # the pixdims. So we create a header + # object with both the affine and the + # pixdims, so they are both preserved. + # + # Note that we have to set the zooms + # after the s/qform, otherwise nibabel + # will clobber them with zooms gleaned + # fron the affine. + header = nib.nifti1.Nifti1Header() + header.set_data_shape(data.shape) + header.set_sform(xform) + header.set_qform(xform) + header.set_zooms(pixdim) + fslimage.Image.__init__(self, data, - xform=xform, + header=header, name=name, - dataSource=filename) + dataSource=filename, + **kwargs) if filename is not None: self.setMeta('mghImageFile', filename) @@ -128,3 +147,30 @@ class MGHImage(fslimage.Image): coordinates into the surface coordinate system for this image. """ return self.__worldToSurfMat + + +def voxToSurfMat(img): + """Generate an affine which can transform the voxel coordinates of + the given image into a corresponding Freesurfer surface coordinate + system (known as "Torig", or "vox2ras-tkr"). + + See https://surfer.nmr.mgh.harvard.edu/fswiki/CoordinateSystems + + :arg img: An :class:`.Image` object. + + :return: A ``(4, 4)`` matrix encoding an affine transformation from the + image voxel coordinate system to the corresponding Freesurfer + surface coordinate system. + """ + + zooms = np.array(img.pixdim[:3]) + dims = img.shape[ :3] * zooms / 2 + + xform = np.zeros((4, 4), dtype=np.float32) + xform[ 0, 0] = -zooms[0] + xform[ 1, 2] = zooms[2] + xform[ 2, 1] = -zooms[1] + xform[ 3, 3] = 1 + xform[:3, 3] = [dims[0], -dims[2], dims[1]] + + return xform diff --git a/tests/test_mghimage.py b/tests/test_mghimage.py index e2ad7739f4796b41297090d09882d0ef5cac6dfe..f91ff31dd0e4c4818fa6a02e442a77357b26c57f 100644 --- a/tests/test_mghimage.py +++ b/tests/test_mghimage.py @@ -67,3 +67,9 @@ def test_MGHImage_save(): expfile = op.abspath(fslimage.addExt('example', mustExist=False)) assert img.dataSource == op.abspath(expfile) + + +def test_voxToSurfMat(): + testfile = op.join(datadir, 'example.mgz') + img = fslmgh.MGHImage(testfile) + assert np.all(np.isclose(img.voxToSurfMat, fslmgh.voxToSurfMat(img)))