diff --git a/tests/test_image_resample.py b/tests/test_image_resample.py index d9f474bf45330e5691535a545faa7ddcac9159a4..8fb488760ed74a14a0714c92d7815ac2a87f43f6 100644 --- a/tests/test_image_resample.py +++ b/tests/test_image_resample.py @@ -263,3 +263,28 @@ def test_resampleToReference3(): exp = np.array([refdata[..., 0]] * 15).transpose((1, 2, 3, 0)) resampled, xform = resample.resampleToReference(img, ref, order=0, mode='nearest') assert np.all(resampled == exp) + + +def test_resampleToReference4(): + + # the image and ref are out of + # alignment, but this affine + # will bring them into alignment + img2ref = transform.scaleOffsetXform([2, 2, 2], [10, 10, 10]) + + imgdata = np.random.randint(0, 65536, (5, 5, 5)) + refdata = np.zeros((5, 5, 5)) + img = fslimage.Image(imgdata) + ref = fslimage.Image(refdata, xform=img2ref) + + # Without the affine, the image + # will be out of the FOV of the + # reference + resampled, xform = resample.resampleToReference(img, ref) + assert np.all(resampled == 0) + + # But applying the affine will + # cause them to overlap + # perfectly in world coordinates + resampled, xform = resample.resampleToReference(img, ref, matrix=img2ref) + assert np.all(resampled == imgdata) diff --git a/tests/test_scripts/test_fsl_apply_x5.py b/tests/test_scripts/test_fsl_apply_x5.py new file mode 100644 index 0000000000000000000000000000000000000000..939fa7d47f43fa20def50a445013ede1cdd81cb7 --- /dev/null +++ b/tests/test_scripts/test_fsl_apply_x5.py @@ -0,0 +1,3 @@ +#!/usr/bin/env python + +# TODO diff --git a/tests/test_transform/test_nonlinear.py b/tests/test_transform/test_nonlinear.py index 2da07f3868b02578cb20a619f09b499d22ee7ca6..c32e7f06f62b8639f88dbc414b0fc50678cf13a2 100644 --- a/tests/test_transform/test_nonlinear.py +++ b/tests/test_transform/test_nonlinear.py @@ -5,10 +5,11 @@ import os.path as op import numpy as np -import fsl.data.image as fslimage -import fsl.transform.affine as affine -import fsl.transform.nonlinear as nonlinear -import fsl.transform.fnirt as fnirt +import fsl.data.image as fslimage +import fsl.utils.image.resample as resample +import fsl.transform.affine as affine +import fsl.transform.nonlinear as nonlinear +import fsl.transform.fnirt as fnirt datadir = op.join(op.dirname(__file__), 'testdata') @@ -40,6 +41,26 @@ def _random_field(): return nonlinear.DeformationField(field, src=src, xform=aff) + +def _affine_field(src, ref, xform, srcSpace, refSpace): + rx, ry, rz = np.meshgrid(np.arange(ref.shape[0]), + np.arange(ref.shape[1]), + np.arange(ref.shape[2]), indexing='ij') + + rvoxels = np.vstack((rx.flatten(), ry.flatten(), rz.flatten())).T + rcoords = affine.transform(rvoxels, ref.getAffine('voxel', refSpace)) + scoords = affine.transform(rcoords, xform) + + field = np.zeros(list(ref.shape[:3]) + [3]) + field[:] = (scoords - rcoords).reshape(*it.chain(ref.shape, [3])) + field = nonlinear.DeformationField(field, src, ref, + srcSpace=srcSpace, + refSpace=refSpace, + header=ref.header, + defType='relative') + return field + + def _random_affine_field(): src = _random_image() @@ -310,3 +331,62 @@ def test_coefficientFieldToDeformationField(): assert np.all(np.isclose(acnv.data, adf .data, **tol)) assert np.all(np.isclose(rcnvnp.data, rdfnp.data, **tol)) assert np.all(np.isclose(acnvnp.data, adfnp.data, **tol)) + + +def test_applyDeformation(): + + src2ref = affine.compose( + np.random.randint(2, 5, 3), + np.random.randint(1, 10, 3), + np.random.random(3)) + ref2src = affine.invert(src2ref) + + srcdata = np.random.randint(1, 65536, (10, 10, 10)) + refdata = np.random.randint(1, 65536, (10, 10, 10)) + + src = fslimage.Image(srcdata) + ref = fslimage.Image(refdata, xform=src2ref) + field = _affine_field(src, ref, ref2src, 'world', 'world') + + expect, xf = resample.resampleToReference( + src, ref, matrix=src2ref, order=1, mode='nearest') + result = nonlinear.applyDeformation( + src, field, order=1, mode='nearest') + + assert np.all(np.isclose(expect, result)) + + + +def test_applyDeformation_altref(): + src2ref = affine.compose( + np.random.randint(2, 5, 3), + np.random.randint(1, 10, 3), + np.random.random(3)) + ref2src = affine.invert(src2ref) + + srcdata = np.random.randint(1, 65536, (10, 10, 10)) + refdata = np.random.randint(1, 65536, (10, 10, 10)) + + src = fslimage.Image(srcdata) + ref = fslimage.Image(refdata, xform=src2ref) + field = _affine_field(src, ref, ref2src, 'world', 'world') + + altrefxform = affine.concat( + src2ref, + affine.scaleOffsetXform([1, 1, 1], [5, 0, 0])) + + altref = fslimage.Image(refdata, xform=altrefxform) + + expect, xf = resample.resampleToReference( + src, altref, matrix=src2ref, order=1, mode='constant', cval=0) + result = nonlinear.applyDeformation( + src, field, ref=altref, order=1, mode='constant', cval=0) + + # We can get imprecision/rounding errors + # which may cause affine_transform and + # map_coordinates to determine that + # boundary voxels are out of bounds + expect = expect[1:-1, 1:-1, 1:-1] + result = result[1:-1, 1:-1, 1:-1] + + assert np.all(np.isclose(expect, result))