Skip to content
Snippets Groups Projects
Commit fc2a1e61 authored by Paul McCarthy's avatar Paul McCarthy :mountain_bicyclist:
Browse files

ENH: Finish coefficientFieldToDisplacementField implementation. Add some docs

parent 98e9ddeb
No related branches found
No related tags found
No related merge requests found
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
# Author: Paul McCarthy <pauldmccarthy@gmail.com> # Author: Paul McCarthy <pauldmccarthy@gmail.com>
# #
"""This module contains utility functions for working with affine """This module contains utility functions for working with affine
transformations. The following funcyions are available: transformations. The following functions are available:
.. autosummary:: .. autosummary::
:nosignatures: :nosignatures:
......
...@@ -88,13 +88,19 @@ def _readFnirtCoefficientField(fname, img, src, ref): ...@@ -88,13 +88,19 @@ def _readFnirtCoefficientField(fname, img, src, ref):
# The sform contains an initial # The sform contains an initial
# global src-to-ref affine # global src-to-ref affine
# (the starting point for the # (the starting point for the
# non-linear registration) # non-linear registration). This
# is encoded as a flirt matrix,
# i.e. it transforms from
# source-scaled-voxels to
# ref-scaled-voxels
srcToRefMat = img.header.get_sform() srcToRefMat = img.header.get_sform()
# The fieldToRefMat affine allows us # The fieldToRefMat affine tells
# to transform coefficient field voxel # the CoefficientField class how
# coordinates into displacement field/ # to transform coefficient field
# reference image voxel coordinates. # voxel coordinates into
# displacement field/reference
# image voxel coordinates.
fieldToRefMat = affine.scaleOffsetXform(knotSpacing, 0) fieldToRefMat = affine.scaleOffsetXform(knotSpacing, 0)
return nonlinear.CoefficientField(fname, return nonlinear.CoefficientField(fname,
......
#!/usr/bin/env python #!/usr/bin/env python
# #
# nonlinear.py - # nonlinear.py - Functions/classes for non-linear transformations.
# #
# Author: Paul McCarthy <pauldmccarthy@gmail.com> # Author: Paul McCarthy <pauldmccarthy@gmail.com>
# #
...@@ -216,7 +216,7 @@ class DisplacementField(NonLinearTransform): ...@@ -216,7 +216,7 @@ class DisplacementField(NonLinearTransform):
reference space. reference space.
:arg from_: Reference image space that ``coords`` are defined in :arg from_: Reference image space that ``coords`` are defined in
:arg to: Source image space to transform ``coords`` into :arg to: Source image space to transform ``coords`` into
:returns ``coords``, transformed into the source image space :returns ``coords``, transformed into the source image space
""" """
if from_ is None: from_ = self.refSpace if from_ is None: from_ = self.refSpace
...@@ -270,10 +270,24 @@ class DisplacementField(NonLinearTransform): ...@@ -270,10 +270,24 @@ class DisplacementField(NonLinearTransform):
class CoefficientField(NonLinearTransform): class CoefficientField(NonLinearTransform):
"""Class which represents a quadratic or cubic B-spline coefficient field """Class which represents a cubic B-spline coefficient field generated by
generated by FNIRT. FNIRT.
The :meth:`displacements` method can be used to calculate relative
displacements for a set of reference space voxel coordinates.
A FNIRT coefficient field typically contains a *premat*, a global affine
transformation from the source space to the reference space, which was
used as the starting point for the non-linear optimisation performed by
FNIRT.
This affine must be provided when creating a ``CoefficientField``, and is
subsequently accessed via the :meth:`srcToRefMat` or :meth:`premat`
attributes.
""" """
def __init__(self, def __init__(self,
image, image,
src, src,
...@@ -287,13 +301,27 @@ class CoefficientField(NonLinearTransform): ...@@ -287,13 +301,27 @@ class CoefficientField(NonLinearTransform):
**kwargs): **kwargs):
"""Create a ``CoefficientField``. """Create a ``CoefficientField``.
:arg fieldType: :arg fieldType: Must be ``'cubic'``
:arg knotSpacing:
:arg srcToRefMat: :arg knotSpacing: A tuple containing the spline knot spacings along
:arg fieldToRefMat: each axis.
:arg srcToRefMat: Initial global affine transformation from the
source image to the reference image. This is
assumed to be a FLIRT-style matrix, i.e. it
transforms from source image FSL coordinates
into reference image FSL coordinates (scaled
voxels).
:arg fieldToRefMat: Affine transformation which can transform reference
image voxel coordinates into coefficient field
voxel coordinates.
See the :class:`NonLinearTransform` class for details on the other
arguments.
""" """
if fieldType not in ('quadratic', 'cubic'): if fieldType not in ('cubic',):
raise ValueError('Unsupported field type: {}'.format(fieldType)) raise ValueError('Unsupported field type: {}'.format(fieldType))
NonLinearTransform.__init__(self, NonLinearTransform.__init__(self,
...@@ -302,7 +330,6 @@ class CoefficientField(NonLinearTransform): ...@@ -302,7 +330,6 @@ class CoefficientField(NonLinearTransform):
ref, ref,
srcSpace, srcSpace,
refSpace, refSpace,
refSpace='voxel',
**kwargs) **kwargs)
self.__fieldType = fieldType self.__fieldType = fieldType
...@@ -314,8 +341,8 @@ class CoefficientField(NonLinearTransform): ...@@ -314,8 +341,8 @@ class CoefficientField(NonLinearTransform):
@property @property
def fieldType(self): def fieldType(self):
"""Return the type of the coefficient field, either ``'cubic'`` or """Return the type of the coefficient field, currently always
``'quadratic'``. ``'cubic'``.
""" """
return self.__fieldType return self.__fieldType
...@@ -352,7 +379,20 @@ class CoefficientField(NonLinearTransform): ...@@ -352,7 +379,20 @@ class CoefficientField(NonLinearTransform):
return np.copy(self.__refToFieldMat) return np.copy(self.__refToFieldMat)
def transform(self, coords, from_=None, to=None): def transform(self, coords, from_=None, to=None, premat=True):
"""Overrides :meth:`NonLinearTransform.transform`. Transforms the
given ``coords`` from the reference image space into the source image
space.
:arg coords: A sequence of XYZ coordinates, or ``numpy`` array of shape
``(n, 3)`` containing ``n`` sets of coordinates in the
reference space.
:arg from_: Reference image space that ``coords`` are defined in
:arg to: Source image space to transform ``coords`` into
:returns ``coords``, transformed into the source image space
:arg premat: If ``True``, the inverse :meth:`srcToRefMat` is applied
to
"""
raise NotImplementedError() raise NotImplementedError()
...@@ -368,7 +408,8 @@ class CoefficientField(NonLinearTransform): ...@@ -368,7 +408,8 @@ class CoefficientField(NonLinearTransform):
raise NotImplementedError() raise NotImplementedError()
# See # See
# https://www.cs.jhu.edu/~cis/cista/746/papers/RueckertFreeFormBreastMRI.pdf # https://www.cs.jhu.edu/~cis/cista/746/papers/\
# RueckertFreeFormBreastMRI.pdf
# https://www.fmrib.ox.ac.uk/datasets/techrep/tr07ja2/tr07ja2.pdf # https://www.fmrib.ox.ac.uk/datasets/techrep/tr07ja2/tr07ja2.pdf
# Cubic b-spline basis functions # Cubic b-spline basis functions
...@@ -412,12 +453,12 @@ class CoefficientField(NonLinearTransform): ...@@ -412,12 +453,12 @@ class CoefficientField(NonLinearTransform):
il = i + l il = i + l
jm = j + m jm = j + m
kn = k + n kn = k + n
mask = ((il >= 0) & mask = (il >= 0) & \
(il < nx) & (il < nx) & \
(jm >= 0) & (jm >= 0) & \
(jm < ny) & (jm < ny) & \
(kn >= 0) & (kn >= 0) & \
(kn < nz)) (kn < nz)
il = il[mask] il = il[mask]
jm = jm[mask] jm = jm[mask]
...@@ -567,14 +608,24 @@ def convertDisplacementSpace(field, from_, to): ...@@ -567,14 +608,24 @@ def convertDisplacementSpace(field, from_, to):
dispType=field.displacementType) dispType=field.displacementType)
def coefficientFieldToDisplacementField(field): def coefficientFieldToDisplacementField(field,
"""Convert a :class:`CoefficientField` into a relative dispType='relative',
:class:`DisplacementField`. premat=True):
"""Convert a :class:`CoefficientField` into a :class:`DisplacementField`.
:arg field: :class:`CoefficientField` to convert
:arg dispType: The type of displcaement field - either ``'relative'`` (the
default) or ``'absolute'``.
:arg field: :class:`CoefficientField` to convert :arg premat: If ``True`` (the default), the :meth:`srcToRefMat` is
:return: :class:`DisplacementField` calculated from ``field`` encoded into the displacements.
:return: :class:`DisplacementField` calculated from ``field``.
""" """
# Generate coordinates for every
# voxel in the reference image
ix, iy, iz = field.ref.shape[:3] ix, iy, iz = field.ref.shape[:3]
x, y, z = np.meshgrid(np.arange(ix), x, y, z = np.meshgrid(np.arange(ix),
np.arange(iy), np.arange(iy),
...@@ -583,12 +634,85 @@ def coefficientFieldToDisplacementField(field): ...@@ -583,12 +634,85 @@ def coefficientFieldToDisplacementField(field):
y = y.flatten() y = y.flatten()
z = z.flatten() z = z.flatten()
xyz = np.vstack((x, y, z)).T xyz = np.vstack((x, y, z)).T
disps = field.displacements(xyz).reshape((ix, iy, iz, 3))
# There are three spaces to consider here:
return DisplacementField(disps, #
src=field.src, # - ref space: Reference image scaled voxels ("fsl" space)
ref=field.ref, #
srcSpace=field.srcSpace, # - aligned-src space: Source image scaled voxels, after the
refSpace=field.refSpace, # source image has been linearly aligned to
header=field.ref.header, # the reference via the field.srcToRefMat
dispType='relative') # This will typically be equivalent to ref
# space
#
# - orig-src space: Source image scaled voxels, in the coordinate
# system of the original source image, without
# linear alignment to the reference image
# The displacements method will
# return relative displacements
# from ref space to aligned-src
# space.
disps = field.displacements(xyz).reshape((ix, iy, iz, 3))
rdfield = DisplacementField(disps,
src=field.src,
ref=field.ref,
srcSpace=field.srcSpace,
refSpace=field.refSpace,
header=field.ref.header,
dispType='relative')
if (dispType == 'relative') and (not premat):
return rdfield
# Convert to absolute - the
# displacements will now be
# absolute coordinates in
# aligned-src space
disps = convertDisplacementType(rdfield)
# Apply the premat if requested -
# this will transform the coordinates
# from aligned-src to orig-src space.
if premat:
# We apply the premat in the same way
# that fnirtfileutils does - applying
# the inverse affine to every ref space
# voxel coordinate, then adding it to
# the existing displacements.
shape = disps.shape
disps = disps.reshape(-1, 3)
refToSrc = affine.invert(field.srcToRefMat)
premat = affine.concat(refToSrc - np.eye(4),
field.ref.getAffine('voxel', 'fsl'))
disps = disps + affine.transform(xyz, premat)
disps = disps.reshape(shape)
# note that convertwarp applies a premat
# differently - its method is equivalent
# to directly transforming the existing
# absolute displacements, i.e.:
#
# disps = affine.transform(disps, refToSrc)
adfield = DisplacementField(disps,
src=field.src,
ref=field.ref,
srcSpace=field.srcSpace,
refSpace=field.refSpace,
header=field.ref.header,
dispType='absolute')
# Not either return absolute displacements,
# or convert back to relative displacements
if dispType == 'absolute':
return adfield
else:
return DisplacementField(convertDisplacementType(adfield),
src=field.src,
ref=field.ref,
srcSpace=field.srcSpace,
refSpace=field.refSpace,
header=field.ref.header,
dispType='relative')
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment